use super::*;
use super::test_utils::*;



#[test]
fn test_dynamic_corest_creation() {
    let corest: DynamicCorest<4> = DynamicCorest::new(0.01.into(), 0.9.into());
    assert_eq!(corest.num_vertices_in_graph(), 0);
    assert_eq!(corest.num_nodes_in_tree(), 0);
}

#[test]
fn test_insert_edge() {
    let mut coreset: DynamicCorest<4> = DynamicCorest::new(0.0.into(), 0.9.into());
    coreset.insert_edge("A", "B", EdgeWeight::new(1.0.into())).unwrap();

    assert_eq!(coreset.num_vertices_in_graph(), 2);
    assert_eq!(coreset.adjacency.len(), 2);
    assert!(coreset.adjacency.contains_key(&NodeIdentity(0)));
    assert!(coreset.adjacency.contains_key(&NodeIdentity(1)));
}

#[test]
fn test_insert_edges(){
    let commands = generate_commands(
        42424242,
        50_000,
        1_000_000,
        1.0.into(),
        0.0,
    );

    let mut coreset: DynamicCorest<4> = DynamicCorest::new(0.0.into(), 0.9.into());

    let result = process_commands(&mut coreset, &commands);
    assert!(result.is_ok(), "Failed to process commands: {:?}", result.err());
}

#[test]
fn test_delete_edge() {
    let edges = vec![
        ("A", "B", EdgeWeight::new(1.0.into())),
        ("B", "C", EdgeWeight::new(2.0.into())),
        ("C", "D", EdgeWeight::new(3.0.into())),
        ("D", "E", EdgeWeight::new(4.0.into())),
        ("E", "A", EdgeWeight::new(5.0.into())),
    ];
    let num_verts_after_insert = vec![
        2, 3, 4, 5, 5
    ];
    let nodes_present_after_insert = vec![
        vec!["A", "B"],
        vec!["A", "B", "C"],
        vec!["A", "B", "C", "D"],
        vec!["A", "B", "C", "D", "E"],
        vec!["A", "B", "C", "D", "E"],
    ];
    let edges_to_remove = vec![
        ("B", "C"),
        ("C", "D"),
        ("D", "E"),
        ("E", "A"),
        ("A", "B"),
    ];
    let num_verts_after_delete = vec![
        5, 4, 3, 2, 0
    ];
    let nodes_present_after_delete = vec![
        vec!["A", "B", "C", "D", "E"],
        vec!["A", "B", "D", "E"],
        vec!["A", "B", "E"],
        vec!["A", "B"],
        vec![],
    ];
    let mut coreset: DynamicCorest<4> = DynamicCorest::new(0.0.into(), 0.9.into());
    for (i,(u, v, w)) in edges.iter().enumerate() {
        println!("Inserting edge: {} - {} with weight {}", u, v, w);
        coreset.insert_edge(u, v, *w).unwrap();
        let mut  degree_info = get_node_degrees(&mut coreset,nodes_present_after_insert[i].as_slice()).unwrap();
        degree_info.sort_by_key(|(_, _, degree)| *degree);
        assert_eq!(coreset.num_vertices_in_graph(), num_verts_after_insert[i]);
    }
    // deletions
    for (i, (u, v)) in edges_to_remove.iter().enumerate() {
        println!("Deleting edge: {} - {}", u, v);
        let edge_weight = coreset.adjacency.get(&coreset.string_map.get(*u).unwrap()).unwrap()
            .get(&coreset.string_map.get(*v).unwrap()).unwrap().clone();
        println!("Edge weight: {}", edge_weight);
        coreset.delete_edge(u, v, edge_weight/(Float::from(2.0).into())).unwrap();
        coreset.delete_edge(u, v, edge_weight/(Float::from(2.0).into())).unwrap();
        let degree_info = get_node_degrees(&mut coreset, nodes_present_after_delete[i].as_slice()).unwrap();
        dbg!(degree_info);
        assert_eq!(coreset.num_vertices_in_graph(), num_verts_after_delete[i]);
    }

    dbg!(coreset.tree_data.sizes);
    dbg!(coreset.tree_data.volumes);
    dbg!(coreset.tree_data.inv_volumes);



}


#[test]
fn test_tree_values_on_updates(){
    let edges = vec![
        ("A", "B", EdgeWeight::new(1.0.into())),
        ("B", "C", EdgeWeight::new(2.0.into())),
        ("C", "D", EdgeWeight::new(3.0.into())),
        ("D", "E", EdgeWeight::new(4.0.into())),
        ("E", "A", EdgeWeight::new(5.0.into())),
    ];

    let mut coreset: DynamicCorest<4> = DynamicCorest::new(0.0.into(), 0.9.into());

    for (u, v, w) in edges.iter() {
        println!("Inserting edge: {} - {} with weight {}", u, v, w);
        coreset.insert_edge(u, v, *w).unwrap();
        
        let info = ["A", "B", "C", "D", "E"].into_iter().filter_map(|n|{
            let node_id = coreset.string_map.get(n);
            if node_id.is_none() {
                return None;
            }
            let node_index = coreset.node_location_map.get(node_id.unwrap()).unwrap();
            Some((n, coreset.tree_data.volumes[node_index.0].clone(), coreset.tree_data.inv_volumes[node_index.0].clone()))
        }).collect::<Vec<_>>();
        dbg!(info);
    }

    let leaf_node_degrees = vec![
        ("A", Volume::new(7.0.into())),
        ("B", Volume::new(4.0.into())),
        ("C", Volume::new(6.0.into())),
        ("D", Volume::new(8.0.into())),
        ("E", Volume::new(10.0.into())),
    ];

    let leaf_node_inv_degrees = leaf_node_degrees.iter()
        .map(|(name, volume)| (*name, convert::<Volume, InvVolume>(*volume).inv()))
        .collect::<Vec<_>>();

    // check the values of the tree at each leaf node:
    for (node_name, expected_volume) in leaf_node_degrees.iter() {
        let node_id = coreset.string_map.get(*node_name).unwrap();
        let node_index = coreset.node_location_map.get(node_id).unwrap();
        assert_eq!(coreset.tree_data.volumes[node_index.0], *expected_volume, "Volume mismatch for node {}", node_name);
    }

    for (node_name, expected_inv_volume) in leaf_node_inv_degrees.iter() {
        let node_id = coreset.string_map.get(*node_name).unwrap();
        let node_index = coreset.node_location_map.get(node_id).unwrap();
        assert_eq!(coreset.tree_data.inv_volumes[node_index.0], *expected_inv_volume, "InvVolume mismatch for node {}", node_name);
    }
}