// compile with c++20 or higher

#include <bits/stdc++.h>

#include "graph_cc_cost_working.hpp"

using namespace std;

void run_k_vs_error_full(string input_file, float threshold) {
  cout << "reading in graph from " << input_file << "..." << endl;

  auto in = deserialize(input_file, threshold);
  vector<element> G;
  G.reserve(in.size());
  for (int i = 0; i < in.size(); i++)
    G.push_back({i, in[i]});

  cout << "finished reading graph!" << endl;

  // main big one
  auto res = run_experiment<100>(G, k_range<1,20>{}, space_seq<.04>{});

  // auto res = run_experiment<150>(G, k_seq<5>{}, space_seq<.01>{});
  
  auto [res_pivot, res_pruned_pivot, res_cc_cost] = res;

  string out_folder = "./results/";
  assert(folder_exists(out_folder));

  std::filesystem::path input_path(input_file);
  string basename = input_path.stem().string();

  if (threshold > 0) {
    std::stringstream ss;
    ss << std::fixed << std::setprecision(3) << threshold;
    std::string thresh_str = ss.str().substr(2);
    basename += "_threshold_" + thresh_str;
  }
  string base_file = out_folder + basename + "_k_vs_error_";
  string latest_file = base_file + "latest.txt";
  auto output_file = timestamp_filename(base_file, ".txt");

  std::ofstream of(output_file);

  of << "[\n";

  for (auto x : res_pivot)
    write_baseline_stats_json(of, x, true);
  for (auto x : res_pruned_pivot)
    write_baseline_stats_json(of, x, true);
  for (int i = 0; i < res_cc_cost.size(); i++)
    write_exec_stats_json(of, res_cc_cost[i], i+1 < res_cc_cost.size());

  of << "]" << endl;

  of.close();

  stringstream command;
  command << "cp " << output_file << " " << latest_file;
  std::system(command.str().c_str());
}

int main(int argc, char* argv[]) {
  cin.tie(0)->sync_with_stdio(0);
  cin.exceptions(cin.failbit);

  if (argc < 2) {
    cerr << "usage: " << argv[0] << " graph_file" << endl;
    return 1;
  }

  string filename = argv[1];
  if (!filename.ends_with(".bin") && !filename.ends_with(".npz")) {
    cerr << "input graph must be in binary format (.bin) or numpy format (.npz)" << endl;
    return 1;
  }

  float threshold = argc < 3 ? -1 : std::stof(argv[2]);

  run_k_vs_error_full(filename, threshold);

  return 0;
}
