use criterion::{criterion_group, criterion_main, Criterion};

use dynamic_csc::faster::{DynamicCorest};
use dynamic_csc::faster::test_utils::{
    generate_commands,
    process_commands
};
use rand::distr::Uniform;
use rand::prelude::*;

use std::fs::File;
use std::io::Write;
use std::io::BufWriter;

// use jemallocator::Jemalloc;

// #[global_allocator]
// static GLOBAL: Jemalloc = Jemalloc;
const arity: usize = 64;

fn bench_faster(c: &mut Criterion) {

    let mut group = c.benchmark_group("faster_benchmarks");
    group.sample_size(10);

    println!("Running benchmark");

    let num_nodes = 100_000;
    let capacity = (num_nodes as f64 * 1.33) as usize;
    let num_updates = 10_000_000;
    let num_rounds = 2;


    let commands = (0..num_rounds).map(|_|{generate_commands(
        42424242,
        num_nodes,
        num_updates,
        1.0.into(),
        0.25,
    )}).collect::<Vec<_>>();

    group.bench_function("faster", |b| {
        b.iter(|| {
            println!("######################################");
            let mut coreset: DynamicCorest<arity> = DynamicCorest::new_with_capacities(
                0.02.into(), 
                0.9.into(),
                None,
                capacity, 4);
            
            // let mut info = Vec::<(usize, f64, f64, f64)>::new();

            commands.iter().enumerate().for_each(|(i,cmds)|{
                process_commands(&mut coreset, cmds).unwrap();
                // Show the capacity info:
                let capacity_info = coreset.get_capacities();
                let mut capacity_info = capacity_info
                    .iter()
                    .map(|(node, cap)| (node.clone(), cap.clone()))
                    .collect::<Vec<_>>();
                capacity_info.sort_by_key(|(node, _)| node.clone());
                println!("Capacity Info: {:?}", (&capacity_info[0],&capacity_info[1]));

                // let (indices, weights, graph, stats) = coreset.extract_coreset_graph(4096*2, 256).unwrap();
                // println!("indices sample: {:?}", &indices[0..10]);
                // println!("weights sample: {:?}", &weights[0..10]);
                // println!("graph cols: {:?}", graph.ncols());

                // let raw_frac = DynamicCorest::<128>::get_average_distance_fraction(&stats).unwrap();
                // let filtered_frac = coreset.filtered_average_distance_fraction;
                // let shift = coreset.shift;
                // info.push((i, raw_frac.0, filtered_frac.0, shift.0));


                // let num_clusters = 8;
                // let mut rng = rand::rng();

                // let random_labels = (0..indices.len())
                //     .map(|_| rng.sample(Uniform::new(0,num_clusters).unwrap()) )
                //     .collect::<Vec<usize>>();

                // let result = coreset.rust_label_full_graph(
                //     indices.as_slice(),
                //      &weights.iter().map(|x|x.0).collect::<Vec<f64>>().as_slice(), &random_labels, num_clusters);

                // println!("labelling sample");
                // println!("{:?}", &result.0[0..10]);
                // println!("{:?}", &result.1[0..10]);
                // println!("pid info: {:?}", coreset.pid);
                // println!("shift: {:?}", coreset.shift);
            });

            // // dump the info to average_distance_errors.txt
            // let mut file = BufWriter::new(File::create("average_distance_errors.txt").unwrap());
            // for (i, raw_frac, filtered_frac, shift) in info {
            //     writeln!(file, "{},{},{},{}", i, raw_frac, filtered_frac, shift).unwrap();
            // }
            // file.flush().unwrap();
            // panic!();

        })
    });
}




// The Criterion macros: define the group of benchmarks and the main entry point
criterion_group!(benches, bench_faster);
criterion_main!(benches);