

// mod static_coreset;
mod d_ary_static_coreset;
pub mod faster;

use faster::wrapper::FasterDynamicCoreset;

use pyo3::{ffi::{PyTupleObject, CO_FUTURE_UNICODE_LITERALS}, prelude::*, types::{PyList, PyTuple}, IntoPyObject, IntoPyObjectExt};
use std::{cell::UnsafeCell, collections::{BTreeMap, HashSet}, fs::File, ops::AddAssign, sync::Arc, vec};
use std::fmt;
use std::fmt::Write as WriteFmt;
use std::io::Write;
use rand::{rngs::{SmallRng, StdRng}, Rng, SeedableRng};
use rand::seq::IteratorRandom;

use rayon::{prelude::*, ThreadPoolBuilder};
use pid::Pid;

use rustc_hash::{FxBuildHasher, FxHashMap, FxHashSet};

use faer::sparse::{
    csr_symbolic::generic::SymbolicSparseRowMat, SparseColMat, SparseRowMat, SymbolicSparseColMat};
use numpy::{IntoPyArray, PyArray1, PyReadonlyArray1};
// use static_coreset::DefaultCoresetSampler as DefaultCoresetSampler;
// use static_coreset::unstable::TreeNode as TreeNode;
// use static_coreset::SelfAffinity as SelfAffinity;
// pub use static_coreset::Float as Float;



use d_ary_static_coreset::DefaultCoresetSampler as DefaultCoresetSampler;
use d_ary_static_coreset::unstable::TreeNode as TreeNode;
use d_ary_static_coreset::SelfAffinity as SelfAffinity;
pub use d_ary_static_coreset::Float as Float;


// Floating point tolerance to decide if we should remove an edge (edge weight within TOLERANCE of 0)
const TOLERANCE: Float = 1e-10;
const SAMPLING_TREE_ARITY: usize = 16;
const CORESET_TREE_ARITY: usize = 8;




// Some cursed nonsense to get mutable references to a node and immutable references to its children, if they exist
// only safe if the indices are all from the same level
unsafe fn get_mut_and_child_refs<'a, T>(
    vec: &'a mut [T],
    indices: &[usize],
) -> Option<Vec<(&'a mut T, Option<&'a T>, Option<&'a T>)>>{
    // We assume the indices are unique and within bounds
    let len = vec.len();

    let mut result = Vec::with_capacity(indices.len());

    // get a mutable pointer to the vector
    let ptr = vec.as_mut_ptr();
    for &index in indices {
        // First we get immutable references to the left and right children if they exist
        let left_index = 2 * index + 1;
        let right_index = 2 * index + 2;

        let left_child = if left_index < len {
            Some(&*ptr.add(left_index))
        } else {
            None
        };
        let right_child = if right_index < len {
            Some(&*ptr.add(right_index))
        } else {
            None
        };
        // Now we construct a mutable pointer to the current node and then convert it into a mutable reference (unsafe)

        // Aliasing: We are ok because we use pointers to grab the children instead of taking a reference to the entire vec
        // This means we are safe to get a mutable reference to the (independent) node below:
        let current_node = unsafe { &mut *ptr.add(index) };
        // Now we push the mutable reference to the current node and the immutable references to the children into the result vector
        result.push((current_node, left_child, right_child));
    }
    Some(result)
}

// the same but works for arbitrary arities
unsafe fn get_mut_and_child_refs_with_arity<'a, T>(
    vec: &'a mut [T],
    indices: &[usize],
) -> Option<Vec<(&'a mut T, [Option<&'a T>;CORESET_TREE_ARITY])>>{
    // We assume the indices are unique and within bounds
    let len = vec.len();

    let mut result = Vec::with_capacity(indices.len());

    let ptr = vec.as_mut_ptr();
    for &index in indices{
        // First we get immutable references to all the children, if they exist
        let mut children = [None; CORESET_TREE_ARITY];
        (0..CORESET_TREE_ARITY).for_each(|i|{
            let child_idx = CORESET_TREE_ARITY * index + i + 1;
            if child_idx < len {
                children[i] = Some(&*ptr.add(child_idx));
            }
        });
        // Now we construct a mutable pointer to the current and the convert it to a mutable reference (unsafe)
        // Aliasing: Because indices are disjoint and from the same level,
        // we will never alias nodes with mutable references more than once.
        let current_node = unsafe { &mut *ptr.add(index) };
        result.push((current_node, children));
    }
    Some(result)
}

// some newtypes so we don't get confused over random integers/strings flying around
#[derive(Debug, Clone, PartialEq, Eq, Hash, Copy)]
pub struct NodeIdx(usize);


impl NodeIdx {
    fn parent(&self) -> Option<NodeIdx> {
        DynamicCoreset::parent_idx(*self)
    }
}

#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]

pub struct NodeName(usize);

impl fmt::Display for NodeName {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "{}", self.0)
    }
}

// when we remove an edge from the adjacency there are 3 cases which we capture here
enum EdgeDeletionResult{
    BothNodesStillConnected,
    SingleNodeDisconnected(NodeName, NodeName), //first node is the disconnected one, second one is still connected
    BothNodesDisconnected(NodeName, NodeName),
}

// Error type
#[derive(Debug)]
pub enum CoresetError{
    NoData,
    InvalidEdge(String,String),
    NodeNotFound(String),
    NodeAlreadyExists(String),
    NoSelfLoopsAllowed(String),
}




impl fmt::Display for CoresetError {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match self {
            CoresetError::NoData => write!(f, "No data in the dynamic coreset"),
            CoresetError::InvalidEdge(u,v) => write!(f, "Invalid edge between {} and {}", u, v),
            CoresetError::NodeNotFound(u) => write!(f, "Node not found: {}", u),
            CoresetError::NodeAlreadyExists(u) => write!(f, "Node already exists: {}", u),
            CoresetError::NoSelfLoopsAllowed(u) => write!(f, "Self loops not allowed: {}", u),
        }
    }
}
impl std::error::Error for CoresetError {}



/// A struct to hold the data for each node in the coreset tree
/// We store a vector of indices and a vector of weights
#[repr(align(64))]
#[derive(Debug, Clone, Default)]
pub struct CoresetNodeData{
    pub indices: Vec<NodeName>, // indices of the points in the coreset at this node
    pub weights: Vec<Float>, // weights of the points in the coreset at this node
}

impl CoresetNodeData{
    // A node is a leaf if it has only one index (and one weight)
    pub fn is_leaf(&self) -> bool{
        self.indices.len() == 1
    }

    pub fn new(indices: Vec<NodeName>, weights: Vec<Float>) -> Self{
        Self{
            indices,
            weights,
        }
    }

    // If we hold more than cutoff points, we assume we are in coreset mode and need to recompute the coreset on an update
    pub fn in_coreset_mode(&self, cutoff: usize) -> bool{
        self.indices.len() >= cutoff
    }
}


// A struct holding the thread local data for each thread.
// We use this to avoid unnecessary allocations and to speed up the sampling process

// #[repr(align(64))]
pub struct ThreadLocalData<const ARITY: usize>{
    rng: SmallRng,
    input_indices: Vec<NodeName>,
    input_weights: Vec<Float>,
    input_degrees: Vec<Float>,
    output_indices: Vec<usize>,
    output_weights: Vec<Float>,
    unique_indices: Vec<NodeName>,
    unique_weights: Vec<Float>,
    seen: FxHashMap<usize,usize>,

    // data for the coreset
    self_affinities: Vec<SelfAffinity>,
    names_to_indices: FxHashMap<NodeName, usize>,
    pub sampling_tree_storage: Vec<TreeNode<ARITY>>,
}


impl <const ARITY: usize> ThreadLocalData<ARITY>{
    fn clear(&mut self){
        self.input_indices.clear();
        self.input_weights.clear();
        self.input_degrees.clear();
        self.output_indices.clear();
        self.output_weights.clear();
        self.unique_indices.clear();
        self.unique_weights.clear();
        self.seen.clear();
        self.self_affinities.clear();
        self.names_to_indices.clear();
        self.sampling_tree_storage.clear();
    }
}


#[repr(align(64))]
pub struct Padded<T>(pub T);

pub struct ThreadLocalPool<T> {
    data: Vec<UnsafeCell<Padded<T>>>,
}

unsafe impl<T> Sync for ThreadLocalPool<T> where T: Send {}

impl<T> ThreadLocalPool<T> {
    pub fn new<F: Fn() -> T>(make: F, n: usize) -> Self {
        let data = (0..n).map(|_| UnsafeCell::new(Padded(make()))).collect();
        Self { data }
    }

    pub fn get(&self) -> &mut T {
        let tid = rayon::current_thread_index().expect("not in Rayon thread");
        unsafe { &mut (*self.data[tid].get()).0 }
    }

    pub unsafe fn get_serial(&self) -> &mut T {
        &mut (*self.data[0].get()).0
    }
}

/// A struct to hold the data for the Dynamic Coreset Algorithm
/// - We store the coreset tree in a vector using a heap layout
/// - We store the mapping from node names (strings) to node indices (usizes) in a FxHashMap (node name -> node index)
/// - We store the degrees of each node in a FxHashMap (node name -> degree)
/// - We store the adjacency list of each node in a FxHashMap (node name -> adjacency list). 
///  The adjacency list is a BTreeMap of node names (strings) to weights (Float).
/// - We store the size of the coreset in a usize. This determines the maximum number of points we can hold in each node of the coreset tree.
///
/// We will implicitly assume each node has an edge to itself of at least weight 1.0 
/// TODO: add an implicit shift to self affinities
#[pyclass]
pub struct DynamicCoreset{
    pub coreset_tree: Vec<CoresetNodeData>, 
    pub leaves: FxHashMap<NodeName, NodeIdx>, // map of node ids (strings) to node ids
    pub degrees: FxHashMap<NodeName, Float>, // map of node ids (strings) to degrees
    pub old_degrees: FxHashMap<NodeName, Float>, // map of node ids (strings) to old degrees
    pub degree_threshold_factor: Float, // Reduce number of updates id degrees change by less than this factor
    pub adjacency: FxHashMap<NodeName, BTreeMap<NodeName, Float>>, // map of node ids (strings) to a BTreeMap of adjacent nodes and their edge weights
    pub string_map: FxHashMap<String, NodeName>, // map for string to NodeName conversion
    pub string_map_rev: FxHashMap<NodeName, String>, // map for NodeName to string conversion
    pub node_name_counter: usize, // counter for the number of nodes in the coreset tree
    pub coreset_size: usize, // the size of the final and every intermediate coreset 
    pub update_thread_pool: rayon::ThreadPool, // thread pool for parallel updates
    pub num_clusters: usize, // estimate of number of clusters used to seed coreset construction
    pub affinity_shift: Float, // shift to be added to the self affinities (multiple of 1/(degree))
    pub thread_buffers: ThreadLocalPool<ThreadLocalData<SAMPLING_TREE_ARITY>>, // thread local data for each thread
    pub num_coresets_computed: std::sync::atomic::AtomicUsize, // number of coreset computed
    pub num_distance_warnings: std::sync::atomic::AtomicUsize, // number of distance warnings after last update
    pub num_distance_updates: std::sync::atomic::AtomicUsize, // number of distance updates after last update
    pub filtered_average_distance_error: Float, // We apply a low pass filter to the average ratio of distance errors
    pub filtering_constant: Float, // alpha in y_t = (1-alpha) x_{t} + alpha y_{t-1}
    pub shift_pid: Pid<Float>, // PID to set the shift to reach a fixed average ratio of distance warnings to updates.
    pub update_buffer: HashSet<String>, // buffer used to defer updates until we querry or hit a max_buffer size
    pub update_buffer_size: usize, // size of the update buffer
}

// MARK: Python Interface

#[pymethods]
impl DynamicCoreset{
    #[new]
    pub fn py_new(
        coreset_size: usize,
        num_clusters: usize, 
        degree_threshold: Float, 
        pid_target: Float,
        update_threads: usize,
        update_buffer_size: usize) -> Self{
            Self::new(
                coreset_size,
                num_clusters, 
                degree_threshold, 
                pid_target,
                update_threads,
                update_buffer_size,
            )
        }
    
    
    pub fn insert_edge(&mut self,u: &str, v: &str,w: Float){
        self.rust_insert_edge(u, v, w).unwrap();
    }

    pub fn delete_edge_weighted(&mut self, u: &str, v: &str, w: Float
    ){
        self.rust_delete_edge(u, v, w).unwrap();
    }

    pub fn delete_edge(&mut self, u: &str, v: &str){
        self.rust_delete_entire_edge(u, v).unwrap();
    }

    pub fn rust_get_coreset_graph<'py>(&mut self, py: Python<'py>) -> PyResult<Bound<'py, PyTuple>>{

        // Get the coreset graph as a sparse matrix
        let coreset_csr = self.rust_extract_coreset_graph().unwrap();

        let (symbolic, data) = coreset_csr.parts();
        let (row_size, col_size, indptr, nnz, indices) = symbolic.parts();

        let n = row_size.into_py_any(py)?;
        let indptr = indptr.to_vec().into_pyarray(py).into_py_any(py)?;
        let indices = indices.to_vec().into_pyarray(py).into_py_any(py)?;
        let data = data.to_vec().into_pyarray(py).into_py_any(py)?;
        let nnz = nnz.unwrap().to_vec().into_pyarray(py).into_py_any(py)?;

        let tuple = PyTuple::new(py,
            &[n, indptr,indices, data, nnz])?;
        PyResult::Ok(tuple)
    }

    pub fn extract_filtered_average_distance_error(&self) -> Float{
        self.filtered_average_distance_error
    }
    pub fn extract_affinity_shift(&self) -> Float{
        self.affinity_shift
    }

    pub fn extract_coreset_indices_and_weights<'py>(&self, py: Python<'py>) -> PyResult<Bound<'py, PyTuple>>{
        // Get the coreset indices and weights as a tuple
        let indices = self.coreset_tree[0].indices.iter().map(|x| self.string_map_rev.get(&x).unwrap().clone());
        let weights = self.coreset_tree[0].weights.clone();

        let indices_list = PyList::new(py,
            &indices.map(|x| x).collect::<Vec<_>>())?.into_py_any(py)?;
        let weights = weights.into_pyarray(py).into_py_any(py)?;

        let tuple = PyTuple::new(py, &[indices_list, weights])?;
        PyResult::Ok(tuple)
    }

    pub fn label_entire_graph<'py>(&self, labels: PyReadonlyArray1<usize>,num_clusters: usize, py: Python<'py>) -> PyResult<Bound<'py,PyTuple>>{

        let (names, labels,distances) = self.rust_label_full_graph(
            self.coreset_tree[0].indices.as_slice(),
            self.coreset_tree[0].weights.as_slice(),
            labels.as_slice().unwrap(),
            num_clusters
        );

        

        let names = PyList::new(
            py,names.iter().map(|x|self.string_map_rev.get(x).unwrap())
        )?.into_py_any(py)?;
        let labels = labels.into_pyarray(py).into_py_any(py)?;
        let distances = distances.into_pyarray(py).into_py_any(py)?;

        let tuple = PyTuple::new(
            py,
            &[names, labels,distances]
        )?;
        PyResult::Ok(tuple)
    }

    pub fn get_names_in_graph<'py>(&self, py: Python<'py>) -> PyResult<Bound<'py,PyList>>{
        // Get the names of the nodes in the graph
        let names = self.string_map.keys().map(|x| x.clone()).collect::<Vec<_>>();
        let names_list = PyList::new(py, &names)?;
        PyResult::Ok(names_list)
    }
}

impl DynamicCoreset{
    pub fn new(
        coreset_size: usize,
        num_clusters: usize,
        degree_threshold: Float, 
        pid_target: Float,
        update_threads: usize,
        update_buffer_size: usize) -> Self{

        let core_ids = Arc::new(core_affinity::get_core_ids().unwrap());
        // println!("Core IDs: {:?}", core_ids);

        let max_input_size = CORESET_TREE_ARITY*coreset_size;
        let internals = ((max_input_size as f32 - 1.0)/(SAMPLING_TREE_ARITY as f32 - 1.0)).ceil() as usize;

        let setpoint = pid_target;
        let pid_setpoint = 1.0-setpoint;
        let mut shift_pid = Pid::new(pid_setpoint, 0.5);
        shift_pid.p(10.0, 0.5);
        shift_pid.i(5.0,0.5);


        Self{
            coreset_tree: vec![],
            leaves: FxHashMap::default(),
            degrees: FxHashMap::default(),
            old_degrees: FxHashMap::default(),
            degree_threshold_factor: degree_threshold,
            adjacency: FxHashMap::default(),
            string_map: FxHashMap::default(),
            string_map_rev: FxHashMap::default(),
            node_name_counter: 0,
            coreset_size: coreset_size,
            update_thread_pool: ThreadPoolBuilder::new()
                .num_threads(update_threads)
                .start_handler(move |i|{
                    // space out the threads across the cores:
                    let core_ids = core_ids.clone();
                    let id = (i as f32 * core_ids.len() as f32 / update_threads as f32).round() as usize;
                    // println!("Thread {} is running on core {:?}", i, &core_ids[id]);
                    core_affinity::set_for_current(core_ids[id]);
                })
                .build()
                .unwrap(),
            num_clusters: num_clusters,
            affinity_shift: 0.0,
            thread_buffers: ThreadLocalPool::new(
                || ThreadLocalData{
                    rng: SmallRng::from_os_rng(),
                    input_indices: Vec::with_capacity(max_input_size),
                    input_weights: Vec::with_capacity(max_input_size),
                    input_degrees: Vec::with_capacity(max_input_size),
                    output_indices: Vec::with_capacity(coreset_size),
                    output_weights: Vec::with_capacity(coreset_size),
                    unique_indices: Vec::with_capacity(coreset_size),
                    unique_weights: Vec::with_capacity(coreset_size),
                    seen: FxHashMap::with_capacity_and_hasher(coreset_size, FxBuildHasher::default()),
                    self_affinities: Vec::with_capacity(max_input_size),
                    names_to_indices: FxHashMap::with_capacity_and_hasher(max_input_size, FxBuildHasher::default()),
                    sampling_tree_storage: Vec::with_capacity(max_input_size + internals),
                },
                update_threads
            ),
            num_coresets_computed: std::sync::atomic::AtomicUsize::new(0),
            num_distance_warnings: std::sync::atomic::AtomicUsize::new(0),
            num_distance_updates: std::sync::atomic::AtomicUsize::new(0),
            filtered_average_distance_error: Float::NEG_INFINITY,
            filtering_constant: 0.001,
            shift_pid, 
            update_buffer: HashSet::with_capacity(update_buffer_size),
            update_buffer_size,
        }
    }

    /// Helper functions to access child and parent nodes in the coreset tree
    fn child_idx(&self, node_idx: NodeIdx, child_offset: usize) -> Option<NodeIdx>{
        let parent_idx = node_idx.0;
        let child_idx = CORESET_TREE_ARITY* parent_idx + child_offset + 1;
        if child_idx < self.coreset_tree.len() {
            Some(NodeIdx(child_idx))
        } else {
            None
        }
    }

    fn parent_idx(node_index: NodeIdx) -> Option<NodeIdx>{
        let child_idx = node_index.0;
        if child_idx == 0 {
            None
        } else {
            Some(NodeIdx((child_idx - 1) / CORESET_TREE_ARITY))
        }
    }

    fn build_internal_node(&mut self, children: [Option<NodeIdx>;CORESET_TREE_ARITY]) -> CoresetNodeData{
        // - If the children have a total of at most self.coreset_size points, we can just merge them
        // - If the children have a total of at least self.coreset_size points, we need to recompute a coreset of the 
        // union of the children
        let indices = children.iter().filter_map(|x|{
            x.map(|x| self.coreset_tree[x.0].indices.clone())
        }).flatten().collect::<Vec<_>>();
        let weights = children.iter().filter_map(|x|{
            x.map(|x| self.coreset_tree[x.0].weights.clone())
        }).flatten().collect::<Vec<_>>();
        CoresetNodeData{
            indices: indices,
            weights: weights,
        }
    } 


    fn construct_coreset<'a>(
        adj_matrix: &'a FxHashMap<NodeName, BTreeMap<NodeName, Float>>,
        num_clusters: usize,
        coreset_size: usize,
        shift: Float,
        verbose: bool,
        worker_buffer: &mut ThreadLocalData<SAMPLING_TREE_ARITY>) ->(usize,usize){

        
        let indices = worker_buffer.input_indices.as_slice();
        let weights = worker_buffer.input_weights.as_slice();
        let tree_storage = &mut worker_buffer.sampling_tree_storage;
        let degree_vector: &[Float] = &worker_buffer.input_degrees.as_slice();
        let rng = &mut worker_buffer.rng;

        let output_indices = &mut worker_buffer.output_indices;
        let output_weights = &mut worker_buffer.output_weights;


        
        
        // set self affinities with the shift
        degree_vector
        .iter().for_each(|d| worker_buffer.self_affinities.push(SelfAffinity((shift/d) +1.0/(d*d))));

        let self_affinities = worker_buffer.self_affinities.as_slice();

        indices.iter().enumerate().for_each(|(i,n)| {
            worker_buffer.names_to_indices.insert(n.clone(), i);
        });

        let names_to_indices = &worker_buffer.names_to_indices;
        let unique_indices = &mut worker_buffer.unique_indices;
        let unique_weights = &mut worker_buffer.unique_weights;
        let seen = &mut worker_buffer.seen;

        let mut coreset_sampler: DefaultCoresetSampler<'_, TreeNode<SAMPLING_TREE_ARITY>, SAMPLING_TREE_ARITY> = DefaultCoresetSampler::new(
            adj_matrix,
            degree_vector,
            self_affinities,
            output_indices,
            output_weights,
            names_to_indices,
            unique_indices,
            unique_weights,
            seen,
            indices,
            weights,
            num_clusters,
            coreset_size,
            rng,
            tree_storage,
        );
        
        // puts the output in unique_indices and unique_weights
        coreset_sampler.sample_and_deduplicate().unwrap();

        // get the number of distance warnings and distance updates
        let num_distance_warnings = coreset_sampler.total_distance_update_warnings();
        let num_distance_updates = coreset_sampler.total_distance_updates();
        (num_distance_warnings, num_distance_updates)

    }

    fn update_internal_node_standalone(
        node: &mut CoresetNodeData,
        children: [Option<&CoresetNodeData>; CORESET_TREE_ARITY],
        adjacency: &FxHashMap<NodeName, BTreeMap<NodeName, Float>>,
        degrees: &FxHashMap<NodeName, Float>,
        coreset_size: usize,
        num_clusters: usize,
        shift: Float,
        worker_buffer: &mut ThreadLocalData<SAMPLING_TREE_ARITY>,
        ) ->(usize,usize){
        // - If the children have a total of at most coreset_size points, we can just merge them
        // - If the children have a total of at least coreset_size points, we need to recompute a coreset of the 
        // union of the children

        // clear the node's data
        node.indices.truncate(0);
        node.weights.truncate(0);

        // unsafe {
        //     node.indices.set_len(0);
        //     node.weights.set_len(0);
        // }


        let mut extend_input = |src_indices: &[NodeName], src_weights: &[Float]| {
            worker_buffer.input_indices.extend_from_slice(&src_indices);
            worker_buffer.input_weights.extend_from_slice(&src_weights);
        };


        // extend the input with the children's data
        children.iter().for_each(|child|{
            if let Some(child) = child {
                extend_input(&child.indices, &child.weights);
            }
        });
    

        // write the degrees of the input nodes to the degrees buffer:
        worker_buffer.input_degrees.truncate(0);
        for idx in worker_buffer.input_indices.iter(){

            if !degrees.contains_key(idx){
                panic!("Node {} not found in degrees", idx);
            }
            let degree = degrees.get(idx).unwrap();
            worker_buffer.input_degrees.push(*degree + 1.0); // Add 1.0 for the self-loops
        }

        // If we have more than coreset_size points, we need to compute a coreset:
        if worker_buffer.input_indices.len() > coreset_size {
            // We need to recompute the coreset
            let (warnings,total_updates) = DynamicCoreset::construct_coreset(
                adjacency,
                num_clusters,
                coreset_size,
                shift,
                false,
                worker_buffer,
            );

            // Now we need to copy the output data into the node
            node.indices.extend_from_slice(worker_buffer.unique_indices.as_slice());
            node.weights.extend_from_slice(worker_buffer.unique_weights.as_slice());

            assert!(warnings <= total_updates);

            return (warnings, total_updates);
        }else{
            // We just return the children's data
            node.indices.extend_from_slice(&worker_buffer.input_indices);
            node.weights.extend_from_slice(&worker_buffer.input_weights);

            return (0,0);
        }
    } 



    fn insert_leaf_node_into_the_tree(
        &mut self,
        node_id: NodeName,
    ) -> Result<NodeIdx, CoresetError> {
        //  Check if node already exists
        if self.leaves.contains_key(&node_id) {
            let name = self.string_map_rev.get(&node_id).unwrap().clone();
            return Err(CoresetError::NodeAlreadyExists(name));
        }
    
        // Determine where new leaf will go
        let new_idx = NodeIdx(self.coreset_tree.len());
    
        // Empty-tree case
        if self.coreset_tree.is_empty() {
            self.coreset_tree.push(CoresetNodeData::new(vec![node_id.clone()], vec![0.0]));
            self.leaves.insert(node_id.clone(), new_idx);
            // string maps were already updated in rust_insert_edge
            return Ok(new_idx);
        }
    
        // Non‐empty: compute parent slot
        let parent = DynamicCoreset::parent_idx(new_idx).unwrap();
    
        // 5 If parent is still a leaf, turn it into an internal node
        if self.coreset_tree[parent.0].is_leaf() {
            // Extract old leaf data
            let old_leaf_data =
                std::mem::replace(&mut self.coreset_tree[parent.0], CoresetNodeData::default());
            let old_leaf_id = old_leaf_data.indices[0].clone();
            let old_leaf_idx = new_idx;
    
            // Push both children: old leaf first, then new leaf
            self.coreset_tree.push(old_leaf_data);
            self.coreset_tree.push(CoresetNodeData::new(vec![node_id.clone()], vec![0.0]));
            let new_leaf_idx = NodeIdx(self.coreset_tree.len() - 1);
    
            // Build and install the internal node at `parent`
            let mut children = [None; CORESET_TREE_ARITY];
            children[0] = Some(old_leaf_idx);
            children[1] = Some(new_leaf_idx);
            let internal = self.build_internal_node(children);
            self.coreset_tree[parent.0] = internal;
    
            // Update leaves map for both leaves
            self.leaves.insert(old_leaf_id, old_leaf_idx);
            self.leaves.insert(node_id.clone(), new_leaf_idx);
    
            Ok(new_leaf_idx)
        } else {
            // Parent already internal: just append the leaf
            self.coreset_tree.push(CoresetNodeData::new(vec![node_id.clone()], vec![0.0]));
            let leaf_idx = NodeIdx(self.coreset_tree.len() - 1);
            self.leaves.insert(node_id.clone(), leaf_idx);
            Ok(leaf_idx)
        }
    }


    fn _insert_edge_into_adjacency(&mut self,u: NodeName, v: NodeName, w: Float){
        // (u,v)
        self.adjacency
            .entry(u.clone()) // get the neighour BTreeMap for u
            .or_insert(BTreeMap::new()) // create a new BTreeMap if it doesn't exist
            .entry(v.clone()) // get the entry for v
            .or_insert(0.0) // create a new entry with value zero if it doesn't exist
            .add_assign(w); // add the weight to the entry
        // (v,u)
        self.adjacency
            .entry(v.clone()) 
            .or_insert(BTreeMap::new()) 
            .entry(u.clone()) 
            .or_insert(0.0) 
            .add_assign(w);
    }

    fn _delete_edge_from_adjacency(
        &mut self,
        u: NodeName,
        v: NodeName,
        w: Float,
    ) -> Result<EdgeDeletionResult, CoresetError> {
        // 1) Mutate (u → v)
        let new_u_w = {
            let u_neigh = self.adjacency
                .get_mut(&u)
                .ok_or_else(|| CoresetError::NodeNotFound(self.string_map_rev[&u].clone()))?;
            let uv = u_neigh
                .get_mut(&v)
                .ok_or_else(|| CoresetError::InvalidEdge(
                    self.string_map_rev[&u].clone(),
                    self.string_map_rev[&v].clone(),
                ))?;
            *uv -= w;
            *uv
        };
        // 2) Mutate (v → u)
        let new_v_w = {
            let v_neigh = self.adjacency
                .get_mut(&v)
                .ok_or_else(|| CoresetError::NodeNotFound(self.string_map_rev[&v].clone()))?;
            let vu = v_neigh
                .get_mut(&u)
                .ok_or_else(|| CoresetError::InvalidEdge(
                    self.string_map_rev[&v].clone(),
                    self.string_map_rev[&u].clone(),
                ))?;
            *vu -= w;
            *vu
        }; 
    
        // Decide once if the edge is gone
        let gone = new_u_w < TOLERANCE || new_v_w < TOLERANCE;
    
        let mut u_disc = false;
        let mut v_disc = false;
    
        if gone {
            // Remove u → v
            {
                let u_neigh = self.adjacency.get_mut(&u);
                if let Some(map) = u_neigh {
                    map.remove(&v);
                    if map.is_empty() {
                        self.adjacency.remove(&u);
                        u_disc = true;
                    }
                } else {
                    u_disc = true;
                }
            } 
    
            // Remove v → u
            {
                let v_neigh = self.adjacency.get_mut(&v);
                if let Some(map) = v_neigh {
                    map.remove(&u);
                    if map.is_empty() {
                        self.adjacency.remove(&v);
                        v_disc = true;
                    }
                } else {
                    v_disc = true;
                }
            }
        }
    
        // 5) Return result
        let result = match (u_disc, v_disc) {
            (true, true)   => EdgeDeletionResult::BothNodesDisconnected(u, v),
            (true, false)  => EdgeDeletionResult::SingleNodeDisconnected(u, v),
            (false, true)  => EdgeDeletionResult::SingleNodeDisconnected(v, u),
            (false, false) => EdgeDeletionResult::BothNodesStillConnected,
        };
        Ok(result)
    }


    fn neighbourhood(&self, node_id: NodeName) -> Vec<NodeName>{
        // Get the neighbours of a node
        let neighbours = self.adjacency.get(&node_id).unwrap();
        let mut result = Vec::new();
        for (neighbour, _) in neighbours.iter(){
            result.push(neighbour.clone());
        }
        result
    }

    fn _increase_degree(&mut self, node_id: NodeName, w: Float) -> bool{
        // Update the degree of the node
        let deg = self.degrees.entry(node_id).or_insert(0.0);
        // clip the degree to be non-negative
        *deg += w;

        let old_deg = self.old_degrees.entry(node_id).or_insert(0.0);
        if *deg > *old_deg * self.degree_threshold_factor || *deg < *old_deg / self.degree_threshold_factor{
            // update the old degree
            *old_deg = *deg;
            // update the value of the leaf:
            let leaf = self.leaves.get(&node_id).unwrap();
            self.coreset_tree[leaf.0].weights[0] = *deg;

            return true;
        }else{
            // don't update the old degree and return false
            return false;
        }
    }

    fn _decrease_degree(&mut self, node_id: &NodeName, w: Float) -> Result<bool, CoresetError>{
        // Update the degree of the node
        let deg = self.degrees.get_mut(node_id).ok_or(
            CoresetError::NodeNotFound(self.string_map_rev.get(&node_id).unwrap().clone()))?;
        // clip the degree to be non-negative
        *deg = (*deg - w).max(0.0);

        let old_deg = self.old_degrees.entry(node_id.clone()).or_insert(0.0);
        if *deg > *old_deg * self.degree_threshold_factor || *deg < *old_deg / self.degree_threshold_factor{
            // update the old degree
            *old_deg = *deg;
            // update the value of the leaf:
            let leaf = self.leaves.get(node_id).unwrap();
            self.coreset_tree[leaf.0].weights[0] = *deg;
            return Ok(true);
        }else{ 
            // don't update the old degree and return false
            return Ok(false);
        }
    }

    fn  get_set_to_update(&self, nodes: Vec<(NodeName,bool, bool)>) -> HashSet<NodeName>{
        // First bool says add self, second bool says add neighbours


        let mut to_update = HashSet::new();


        // add all the neighbours of the nodes in the set to the set
        nodes.iter().for_each(|(node,_,get_neighbours)|{
            // if we don't want to get the neighbours, we just return
            if *get_neighbours{
                // get the neighbours of the node
                let neighbours = self.adjacency.get(node).unwrap();
                for (neighbour, _) in neighbours.iter(){
                    to_update.insert(neighbour.clone());
                }
            }
        });
        // Now we need to add the nodes themselves to the set
        nodes.iter().for_each(|(node,add_self,_)| {
            if *add_self{
                to_update.insert(node.clone());
            }
        });

        to_update
    }

    fn propagate_update_to_buffer(&mut self, to_update: HashSet<NodeName>, force_update: bool){

        // first we fill the update buffer with the nodes to update (after mapping to strings)
        to_update.iter().for_each(|x| {
            let string_name = self.string_map_rev.get(x).unwrap();
            self.update_buffer.insert(string_name.clone());
        });

        // Now we flush the buffer if it is full or if we are forcing an update:
        if self.update_buffer.len() >= self.update_buffer_size || force_update{
            let updates = self.update_buffer.drain().filter_map(|x|{
                match self.string_map.get(&x){
                    Some(node_id) => Some(node_id.clone()),
                    None => None,
                }
            }).collect::<HashSet<NodeName>>();

            // Now we need to propagate the updates up the tree
            self.propagate_updates(updates);
        }
        


    }


    fn propagate_updates(&mut self, to_update: HashSet<NodeName>){
        // We need to compute coresets along the paths from each node to update to the root of the tree.
        // To avoid recomputing the same coreset multiple times,
        // we work our way up the tree from the bottom level 

        // clear the number of distance warnings and updates
        self.num_distance_warnings.store(0, std::sync::atomic::Ordering::Relaxed);
        self.num_distance_updates.store(0, std::sync::atomic::Ordering::Relaxed);

        // Hold out the leaves that are on the second level for the first iteration:
        let tree_len = self.coreset_tree.len();

        // heap height (starting from zero) = floor( log_d((d-1)*n+1) )
        // This comes from the equation for the total number of nodes in a complete d-ary tree of height h:
        // N = (d^(h+1)-1)/(d-1)
        // The index of the first leaf on the bottom level will then correspond 
        // to the number of nodes in a complete d-ary tree of height h-1
        // aka first_leaf_on_bottom_level = (d^h -1)/(d-1)

        let heap_height = (((CORESET_TREE_ARITY-1)*tree_len +1) as Float).log(CORESET_TREE_ARITY as Float).floor() as usize;
        let first_leaf_on_bottom_level = (CORESET_TREE_ARITY.pow(heap_height as u32)-1) / (CORESET_TREE_ARITY-1);

        let level = to_update.into_iter().map(|x| self.leaves.get(&x).unwrap().0.clone()).collect::<Vec<_>>();
        let (mut level, deferred) = level.into_iter().partition::<Vec<_>, _>(|x| *x >= first_leaf_on_bottom_level);

        let mut deferred_added = false;
        let parallel_threshold = (self.coreset_size as Float).log2().ceil() as usize;
        let mut level_count = 0;


        // Edge case when level is empty and everything is on the second level
        while !level.is_empty() || !deferred_added {
            level_count += 1;
    
            let next_level: HashSet<Option<NodeIdx>> = if (level_count >= parallel_threshold) && (self.update_thread_pool.current_num_threads() > 1) && (level.len() >= 2) {
                // Parallel: use custom thread pool and unsafe access pattern
                let refs: Vec<(&mut CoresetNodeData, [Option<&CoresetNodeData>; CORESET_TREE_ARITY])> = unsafe { get_mut_and_child_refs_with_arity(&mut self.coreset_tree, &level).unwrap() };
                let refs_and_indices = level.iter().zip(refs.into_iter()).collect::<Vec<_>>();
            
                // let chunk_size = refs_and_indices.len() / self.update_thread_pool.current_num_threads();

                self.update_thread_pool.install(|| {
                    refs_and_indices
                        .into_par_iter()
                        .map(|(node_idx, (node, maybe_children))| {
                            if node.is_leaf() {
                                DynamicCoreset::parent_idx(NodeIdx(*node_idx))
                            } else {

                                let worker_buffer = self.thread_buffers.get();
                                worker_buffer.clear();

                                let (warn,total) = DynamicCoreset::update_internal_node_standalone(
                                    node,
                                    maybe_children,
                                    &self.adjacency,
                                    &self.degrees,
                                    self.coreset_size,
                                    self.num_clusters,
                                    self.affinity_shift,
                                    worker_buffer,
                                );
                                self.num_coresets_computed.fetch_add(1, std::sync::atomic::Ordering::Relaxed);
                                self.num_distance_warnings.fetch_add(warn, std::sync::atomic::Ordering::Relaxed);
                                self.num_distance_updates.fetch_add(total, std::sync::atomic::Ordering::Relaxed);
                                DynamicCoreset::parent_idx(NodeIdx(*node_idx))
                            }
                        })
                        .collect()
                })
            } else {
                // Serial: 
                level
                    .iter()
                    .map(|&node_idx| {
                        if self.coreset_tree[node_idx].is_leaf() {
                            DynamicCoreset::parent_idx(NodeIdx(node_idx))
                        } else {

                            let worker_buffer = unsafe{self.thread_buffers.get_serial()};
                            worker_buffer.clear();
                            let (node,maybe_children) = unsafe { get_mut_and_child_refs_with_arity(&mut self.coreset_tree, &[node_idx]).unwrap()}.into_iter().next().unwrap(); 
                            let (warn,total) = DynamicCoreset::update_internal_node_standalone(
                                node,
                                maybe_children,
                                &self.adjacency,
                                &self.degrees,
                                self.coreset_size,
                                self.num_clusters,
                                self.affinity_shift,
                                worker_buffer,
                            );
                            self.num_coresets_computed.fetch_add(1, std::sync::atomic::Ordering::Relaxed);
                            self.num_distance_warnings.fetch_add(warn, std::sync::atomic::Ordering::Relaxed);
                            self.num_distance_updates.fetch_add(total, std::sync::atomic::Ordering::Relaxed);
                            DynamicCoreset::parent_idx(NodeIdx(node_idx))
                        }
                    })
                    .collect()
            };
    
            level = next_level
                .into_iter()
                .filter_map(|x| x)
                .map(|x| x.0)
                .collect::<Vec<_>>();
    
            if !deferred_added {
                level.extend(deferred.iter().cloned());
                deferred_added = true;
            }
        }
    }


    pub fn rust_insert_edge(&mut self, u: &str, v: &str, w: Float) -> Result<(), CoresetError>{
        // Part of the public API


        // Can't accept self loops:
        if u == v {
            return Err(CoresetError::NoSelfLoopsAllowed(u.to_owned()));
        }

        // First we intern the strings if it is not already in the maps:
        let u = self.string_map.entry(u.to_owned()).or_insert_with(||{
            let new_node = NodeName(self.node_name_counter);
            self.node_name_counter += 1;
            self.string_map_rev.insert(new_node.clone(), u.to_owned());
            new_node
        }).clone();

        let v = self.string_map.entry(v.to_owned()).or_insert_with(||{
            let new_node = NodeName(self.node_name_counter);
            self.node_name_counter += 1;
            self.string_map_rev.insert(new_node.clone(), v.to_owned());
            new_node
        }).clone();

        // We need to add the edge and it's weight to our datastructure and perform the necessary updates

        assert!(w > 0.0, "Weight must be positive");
        // First update the adjacency list entries for u and v:
        self._insert_edge_into_adjacency(u.clone(), v.clone(), w);




        // Now we need to check if u and v are already in the coreset tree:
        if !self.leaves.contains_key(&u){
            // If u is not in the coreset tree, we insert it and add it to the leaves map
            self.insert_leaf_node_into_the_tree(u.clone()).unwrap();

        }
        if !self.leaves.contains_key(&v){
            // If v is not in the coreset tree, we insert it and add it to the leaves map
            self.insert_leaf_node_into_the_tree(v.clone()).unwrap();
        }

        // Now update the degrees of u and v since inserting doesn't set the degree:
        let update_u_neighbours = self._increase_degree(u.clone(), w);
        let update_v_neighbours = self._increase_degree(v.clone(), w);

        // Now that everything is in the coreset tree, we collect a set of all nodes whose kernel values may have changed:
        let to_update = self.get_set_to_update(
            vec!(
                    (u.clone(), update_u_neighbours, update_u_neighbours),
                    (v.clone(), update_v_neighbours, update_v_neighbours),
                )
            );

        // Now we need to go through the nodes in the set and propage the updates up the tree:
        self.propagate_update_to_buffer(to_update, false);

        // update filtered distance error:
        self.update_filtered_average_dist_error();
        
        Ok(())
    }


    /// After deletion, return zero, one, or two node indices 
    /// that might need updating.
    fn delete_node_from_coreset_tree(
        &mut self,
        target_node_idx: NodeIdx
    ) -> Result<Vec<NodeIdx>, CoresetError> {
        if self.coreset_tree.is_empty() {
            return Ok(vec![]);
        }

        let old_end_idx = NodeIdx(self.coreset_tree.len() - 1);
        let mut changed_positions = vec![];

        // If target is not the last node, swap 
        if target_node_idx != old_end_idx {
            self.coreset_tree.swap(target_node_idx.0, old_end_idx.0);
            // The occupant from end_idx is now at target_node_idx
            let moved = self.coreset_tree[target_node_idx.0].indices[0].clone();
            self.leaves.insert(moved, target_node_idx);
            // We'll need to re-check or update the occupant now at target_node_idx
            changed_positions.push(target_node_idx);
        }

        // Remove the node at end_idx
        self.coreset_tree.pop();

        // If that was the only node, we’re done and don't need to trigger any updates
        if self.coreset_tree.is_empty() {
            return Ok(changed_positions);
        }


        // If the node wasn't the only node, we may need to trigger updates for a sibling,
        // or even promote the parent to a leaf and trigger and update for it.


        if let Some(parent_idx) = DynamicCoreset::parent_idx(old_end_idx) {
            // Check if there's a single-child parent to fix
            // we can check this by seeing if the parent index is valid and:
            // 1. The first child index is less than the length of the tree
            // 2. The second child index is at least the length of the tree (aka it doesn't exist)

            // This is equivalent to checking if the first child index is at the last position in the tree
            if let Some(first) = self.child_idx(parent_idx, 0){
                if first.0 == self.coreset_tree.len()-1{
        
                    let only_child_idx = first;
                    // We need to swap the only child with the parent, then pop the parent
                    self.coreset_tree.swap(parent_idx.0, only_child_idx.0);
                    // update the leaves map
                    let moved = self.coreset_tree[parent_idx.0].indices[0].clone();
                    self.leaves.insert(moved, parent_idx);
                    // Now we need to remove the parent from the tree
                    self.coreset_tree.pop();

                    // if we had previously added only_child_idx to changed, we remove it (since we've just removed it)
                    if changed_positions.first() == Some(&only_child_idx) {
                        changed_positions.remove(0);
                    }
                    // We need to add the leaf now at parent_idx to the changed positions
                    changed_positions.push(parent_idx);
                }else{
                    // In this case, we have a parent that is not a leaf with at least one child.
                    // We need to trigger an update for it so we select the first child:
                    if !changed_positions.contains(&first) {
                        // We only add the first child if it is not already in the changed positions
                        changed_positions.push(first);
                    }
                }
            }
        } else {
            // In this case, we presumably deleted the root
            assert!(self.coreset_tree.is_empty(), "Tree should be empty if root was removed");
        }

        Ok(changed_positions)
    }

    pub fn rust_delete_entire_edge(&mut self, u: &str, v: &str) -> Result<(), CoresetError>{
        // Part of the public API

        // We need to remove the edge completely.

        // First check if the nodes exists in the cache:

        if !self.string_map.contains_key(u){
            return Err(CoresetError::NodeNotFound(u.to_owned()));
        }
        if !self.string_map.contains_key(v){
            return Err(CoresetError::NodeNotFound(v.to_owned()));
        }

        // get the node names from the cache:
        let u_id = self.string_map.get(u).unwrap().clone();
        let v_id = self.string_map.get(v).unwrap().clone();

        
        // Now we get the weight of the edge if it exists else return an error
        let w = self.adjacency.get(&u_id).unwrap().get(&v_id);
        match w{
            None =>{
                return Err(CoresetError::InvalidEdge(u.to_owned(), v.to_owned()))
            },
            Some(w) => {
                // Now we call the delete edge function with this weight:
                return self.rust_delete_edge(u, v, *w);
            }
        }

    }


    pub fn rust_delete_edge(&mut self, u: &str, v: &str, w: Float) -> Result<(), CoresetError>{

        // Part of the public API

        // first we check if the nodes u and v are even present in the map
        // First check if the nodes exists in the cache:
        if !self.string_map.contains_key(u){
            return Err(CoresetError::NodeNotFound(u.to_owned()));
        }
        if !self.string_map.contains_key(v){
            return Err(CoresetError::NodeNotFound(v.to_owned()));
        }
        // get the node names from the cache:
        let u = self.string_map.get(u).unwrap().clone();
        let v = self.string_map.get(v).unwrap().clone();
        let u_neighbours = self.neighbourhood(u.clone());
        let v_neighbours = self.neighbourhood(v.clone());

        // start building the set of nodes to update. We add the neighbours of u and v but not themselves
        let mut to_update = HashSet::new();

        // Now we update the adjacency list, propagating any errors with ?:
        let edge_deletion_result = self._delete_edge_from_adjacency(u, v, w)?;

        // Handle any deletions/updates to the degrees and leaves map
        match edge_deletion_result{
            EdgeDeletionResult::BothNodesStillConnected => {
                // only need to decrement degrees:
                let u_degree_threshold_hit = self._decrease_degree(&u, w)?;
                let v_degree_threshold_hit = self._decrease_degree(&v, w)?;

                to_update = self.get_set_to_update(
                    vec!(
                        (u.clone(),u_degree_threshold_hit, u_degree_threshold_hit),
                        (v.clone(), u_degree_threshold_hit, v_degree_threshold_hit),
                    )
                );
            },
            EdgeDeletionResult::SingleNodeDisconnected(to_be_deleted,to_keep) => {

                let to_keep_neighbours = {
                    match u == to_keep{
                        true => u_neighbours.iter().cloned(),
                        false => v_neighbours.iter().cloned()
                    }
                };
                // We need to remove the node from the leaves map and the degrees map

                let to_be_deleted_idx = self.leaves.get(&to_be_deleted).unwrap().clone();


                // update the degree of the remaining node
                let to_keep_threshold_hit = self._decrease_degree(&to_keep, w)?;
                self.leaves.remove(&to_be_deleted);
                self.degrees.remove(&to_be_deleted);
                self.old_degrees.remove(&to_be_deleted);

                // remove the disconnected node from the coreset tree
                let changed_positions = self.delete_node_from_coreset_tree(to_be_deleted_idx)?;

                // We only update the node to keep and its neighbours if it hits the degree change threshold:
                if to_keep_threshold_hit{
                    to_update.extend(to_keep_neighbours);
                    to_update.insert(to_keep.clone());
                    to_update.remove(&to_be_deleted);
                }
                // However by deleting this node, we may have moved up to two other nodes that will also need 
                // updating. However we don't need to add their neighbours since 
                // it's just a structural change in the tree.
                for &idx in changed_positions.iter() {
                    let occupant_name = self.coreset_tree[idx.0].indices[0].clone();
                    to_update.insert(occupant_name);
                }

                // We now remove the node from the string maps
                let string_name = self.string_map_rev.get(&to_be_deleted).unwrap().clone();
                self.string_map.remove(&string_name);
                self.string_map_rev.remove(&to_be_deleted);
            },
            EdgeDeletionResult::BothNodesDisconnected(x, y) => {

                // both nodes now have no neighbours

                // We can remove both nodes from the to_update set

                let x_idx = self.leaves.get(&x).unwrap().clone();
                let changed_positions = self.delete_node_from_coreset_tree(x_idx)?;
                self.leaves.remove(&x);
                self.degrees.remove(&x);
                self.old_degrees.remove(&x);


                // By deleting this node, we may have moved up to two other nodes that will also need 
                // updating. However we don't need to add their neighbours since 
                // it's just a structural change in the tree.
                for &idx in changed_positions.iter() {
                    let occupant_name = self.coreset_tree[idx.0].indices[0].clone();

                    // don't add the name of the other node we are about to delete
                    if &occupant_name == &y{
                        continue;
                    }
                    to_update.insert(occupant_name);
                }

                let y_idx = self.leaves.get(&y).unwrap().clone();
                let changed_positions = self.delete_node_from_coreset_tree(y_idx)?;
                self.leaves.remove(&y);
                self.degrees.remove(&y);
                self.old_degrees.remove(&y);

                // By deleting this node, we may have moved up to two other nodes that will also need 
                // updating. However we don't need to add their neighbours since 
                // it's just a structural change in the tree.
                for &idx in changed_positions.iter() {
                    let occupant_name = self.coreset_tree[idx.0].indices[0].clone();
                    to_update.insert(occupant_name);
                }

                // We now remove the nodes from the string maps:
                let x_name = self.string_map_rev.get(&x).unwrap();
                self.string_map.remove(x_name);
                self.string_map_rev.remove(&x);
                let y_name = self.string_map_rev.get(&y).unwrap();
                self.string_map.remove(y_name);
                self.string_map_rev.remove(&y);


            }
        }

        // Now we need to update the coreset tree using the node names in the to_update set
        self.propagate_update_to_buffer(to_update, false);
    
        // update filtered distance error:
        self.update_filtered_average_dist_error();

        Ok(())
    }


    pub fn rust_extract_coreset_graph(&mut self) -> Result<SparseRowMat<usize,Float>, CoresetError>{

        if self.coreset_tree.is_empty(){
            return Err(CoresetError::NoData);
        }
        // trigger any updates in the buffer:
        self.propagate_update_to_buffer(HashSet::new(), true);

        let coreset_indices = &self.coreset_tree[0].indices;
        let coreset_weights = &self.coreset_tree[0].weights;
        let degrees = coreset_indices.iter()
            .map(|name| *self.degrees.get(name).unwrap())
            .collect::<Vec<_>>();

        let n = coreset_indices.len();

        // First we build a map from the NodeNames to indices in the coreset graph:
        let node_name_to_index = coreset_indices
            .iter()
            .enumerate()
            .map(|(idx, name)| (*name, idx))
            .collect::<FxHashMap<NodeName, usize>>();

        let shift = self.affinity_shift;

        let W_D_inv = (0..n).into_iter()
        .map(|idx|{
            coreset_weights[idx]/degrees[idx]
        }).collect::<Vec<Float>>();


        // guess the number of non-zero entries in the coreset graph:
        let mut data = Vec::<Float>::with_capacity(n*200);
        let mut indices = Vec::<usize>::with_capacity(n*200);
        let mut indptr = Vec::<usize>::with_capacity(n+1);
        let mut nnz_per_row = Vec::<usize>::with_capacity(n);

        let mut indptr_counter = 0;
        for (i, &node_name) in coreset_indices.iter().enumerate(){
            let neighbours = self.adjacency.get(&node_name).unwrap();
    
            // get the neighbours of index that are in the coreset and transform the data
            // We are computing
            // A_C = W_CD^{-1}_C A_C D^{-1}_C W_C + W_C shift*D^{-1}_C W_C
            //     = W_CD^{-1}_C A_C D^{-1}_C W_C + shift* W_C*D^{-1}_C W_C
            // where:
            //  -A_C is the submatrix of A corresponding to the coreset indices,
            //  -W_C is the diagonal matrix of coreset weights,
            //  -D is the diagonal matrix of A and D_C is the submatrix of D corresponding to the coreset indices.
            let W_D_inv_i = W_D_inv[i];
            let mut good_indices_and_data_transformed = neighbours.iter().filter_map(|(&neighbour_name,&data)|{
                if node_name == neighbour_name{
                    node_name_to_index.get(&neighbour_name).map(|&coreset_j|{
                    (coreset_j, data*W_D_inv_i*W_D_inv[i] + shift*coreset_weights[i]*W_D_inv_i)
                })}
                else{
                    node_name_to_index.get(&neighbour_name).map(|&coreset_j|{
                    (coreset_j, data*W_D_inv_i*W_D_inv[coreset_j])
                })}
    
            }).collect::<Vec<(usize,Float)>>();

            good_indices_and_data_transformed.sort_unstable_by_key(|&(idx,_)| idx);

            // push the data and indices to the data and indices vectors:
            data.extend(good_indices_and_data_transformed.iter().map(|x| x.1));
            indices.extend(good_indices_and_data_transformed.iter().map(|x| x.0));
            let nnz = good_indices_and_data_transformed.len();
            nnz_per_row.push(nnz);
            // push the indptr counter to the indptr vector and bump by nnz
            indptr.push(indptr_counter);
            indptr_counter += nnz;
        }
        // push the last indptr counter to the indptr vector
        indptr.push(indptr_counter);

        // Now we need to convert the data and indices vectors to a SparseColMat
        Ok(SparseRowMat::new(
            SymbolicSparseRowMat::<faer::sparse::csr_symbolic::Own<_,_,_>>::new_checked(
                n,
                n,
                indptr,
                Some(nnz_per_row), 
                indices,
            ),
            data
        ))
    }

    pub fn rust_label_full_graph(&self,
        coreset_indices: &[NodeName],
        coreset_weights: &[Float],
        coreset_labels: &[usize],
        num_clusters: usize,
    ) -> (Vec<NodeName>, Vec<usize>, Vec<Float>){

        let shift = self.affinity_shift;
        let coreset_size = coreset_indices.len();
        let n = self.adjacency.len();
        assert!(coreset_size == coreset_weights.len() && coreset_size == coreset_labels.len());
        
        // group the coreset and coreset weights by label and process each group in parallel:
        let coreset_grouped = coreset_indices.iter().zip(coreset_labels).zip(coreset_weights).fold(
            vec![(vec![],vec![]);num_clusters], |mut acc, ((&i, &label), &weight)|{
                acc[label].0.push(i);
                acc[label].1.push(weight);
                acc
            }
        );

        // Now we compute the center norms and center denoms for each cluster
        let result = coreset_grouped.into_par_iter().enumerate().map(|(_, (indices,weights))|{
            // return zero if the cluster is empty
            if indices.is_empty(){
                return (0.0,0.0);
            }

            let indices_set: HashSet<&NodeName> = indices.iter().collect();
            let index_to_weight: FxHashMap<_,_> = indices.iter().zip(weights.iter()).collect();

            // compute the denominator:
            let denom = weights.iter().sum::<Float>();
            // compute the center norm sum
            let mut center_norm_sum = 0.0;
            indices.iter().for_each(|i|{
                let weight = index_to_weight[i];
                let neighbour_indices = self.adjacency[i].keys();
                let neighbour_values = self.adjacency[i].iter().map(|(j,v)|{
                    if i!=j{
                        v/(self.degrees[i]*self.degrees[j])
                    }else{
                        v/(self.degrees[i]*self.degrees[j]) + shift/(self.degrees[i])
                    }

                });
                center_norm_sum += neighbour_indices.zip(neighbour_values).fold(
                    Float::from(0.0), |acc, (j, value)|{
                        match indices_set.contains(&j){
                            true => acc + value*weight*index_to_weight[&j],
                            false => acc
                        }
                    });
            });
            center_norm_sum /= denom*denom;
            (center_norm_sum,denom)
        }).collect::<Vec<(Float,Float)>>();

        let (center_norms, center_denoms): (Vec<Float>,Vec<Float>) = result.into_iter().unzip();

            // Now find the cluster with the smallest center norm - this will be the "default" cluster

        let smallest_center_by_norm = center_norms.iter().enumerate()
        .min_by(|(_,a),(_,b)| a.partial_cmp(b).unwrap()).unwrap().0;
        let smallest_center_by_norm_value = center_norms[smallest_center_by_norm];

        // Now prepare to label everything in parallel:

        let coreset_set = coreset_indices.iter().collect::<FxHashSet<_>>();
        let label_map = coreset_indices.iter().zip(coreset_labels).collect::<FxHashMap<_,_>>();
        let weight_map = coreset_indices.iter().zip(coreset_weights).collect::<FxHashMap<_,_>>();

        let node_names = self.leaves.keys().into_iter().cloned().collect::<Vec<_>>();

        let labels_and_distances2: (Vec<usize>,Vec<Float>) = node_names.par_iter().map(|i|{

            let vertex_degree = self.degrees[i];
            // store the inner product to all the centers
            let mut x_to_c_is = FxHashMap::default();


            // let neighbour_indices = self.adjacency[i].keys();
            // let neighbour_edge_weights = self.adjacency[i].values();
            // let neighbour_values = adj_mat.values_of_row(i).iter().enumerate().map(|(j,v)|{
            //     v/(degree_vector[i]*degree_vector[j])
            // });

            self.adjacency[i].iter().for_each(|(indx,weight)|{
                if coreset_set.contains(&indx){
                        let label = label_map[&indx];
                        let neighbour_weight = weight_map[&indx];
                        let inner_prod_with_vertex = {
                            if i!=indx{
                                weight/(vertex_degree*self.degrees[indx])
                            }else{
                                weight/(vertex_degree*self.degrees[indx])
                                /(vertex_degree*self.degrees[indx]) + shift/(vertex_degree)
                            }
                        };
                        x_to_c_is.entry(label).and_modify(|e|{
                            *e += neighbour_weight*inner_prod_with_vertex;
                        }).or_insert(neighbour_weight*inner_prod_with_vertex);
                    }
            });

            // normalize the inner products to each cluster by each center denominator
            x_to_c_is.iter_mut().for_each(|(k,v)| *v /= center_denoms[**k]);

            let mut best_center = smallest_center_by_norm;
            let mut best_center_value = smallest_center_by_norm_value;

            x_to_c_is.iter().for_each(|(center,v)|{
                // right now v is just the inner product to each center, not the distance

                // When we compute the (smallest) distance, we can ignore the contribution of the vertex
                let distance = center_norms[**center] - 2.0*v;
                if distance < best_center_value{
                    best_center = **center;
                    best_center_value = distance;
                }
            });
            (best_center,best_center_value + 1.0/(vertex_degree*vertex_degree) + shift/(vertex_degree))
        }).unzip();
        
        (
            node_names,
            labels_and_distances2.0,
            labels_and_distances2.1
        )

    }



    pub fn get_average_distance_error(&self) -> Option<Float>{
        // Get the average distance error
        let num_warnings = self.num_distance_warnings.load(std::sync::atomic::Ordering::Relaxed);
        let num_updates = self.num_distance_updates.load(std::sync::atomic::Ordering::Relaxed);
        if num_warnings > num_updates{
            println!("warnings: {}, updates: {}", num_warnings, num_updates);
            panic!();
        }
        assert!(num_warnings <= num_updates);
        if num_updates == 0{
            return None;
        }
        Some((num_warnings as Float) / (num_updates as Float))
    }
    
    pub fn get_filtered_average_distance_error(&self) -> Float{
        self.filtered_average_distance_error
    }

    pub fn update_filtered_average_dist_error(&mut self){


        match  self.get_average_distance_error(){
            Some(error) => {

                // set the filtered value to the first value on startup
                if self.filtered_average_distance_error == Float::NEG_INFINITY{
                    self.filtered_average_distance_error = error;
                    return;
                }


                // tukey filtering 
                let alpha = self.filtering_constant;
                let threshold = 0.15;
                
                let r = error - self.filtered_average_distance_error;
                let weight = (threshold / r.abs()).min(1.0).powi(2); // (threshold / abs(r))^2 clipped at 1
                
                self.filtered_average_distance_error = alpha * weight * error + (1.0 - alpha * weight) * self.filtered_average_distance_error;

                // use the pid to get the new shift
                // since the pid outputs values from -0.5 to 0.5, we shift up by 0.5 to get 0 to 1:
                let pid_output = self.shift_pid.next_control_output(
                    1.0-self.filtered_average_distance_error
                );

                // shift the output to be between 0 and 1
                let new_shift = pid_output.output+0.5;
                assert!(new_shift >= 0.0 && new_shift <= 1.0, "Shift must be between 0 and 1");
                self.affinity_shift = new_shift;

            },
            None =>{}
        }
    }

    /// Generate a Graphviz DOT string for debugging/visualization.
    /// - Each node is labeled with the concatenated indices of its `CoresetNodeData`.
    /// - Edges show the parent -> child relationship in the heap layout.
    pub fn to_dot(&self, graph_name: &str) -> String {
        fn escape_label(s: &str) -> String {
            s.replace('\\', "\\\\")
             .replace('"', "\\\"")
             .replace('{', "\\{")
             .replace('}', "\\}")
        }
    
        let mut out = String::new();
        writeln!(out, "digraph {} {{", graph_name).unwrap();
        writeln!(out, "    node [shape=record];\n").unwrap();
    
        // Node declarations
        for (i, node_data) in self.coreset_tree.iter().enumerate() {
            let joined = node_data.indices
                .iter()
                .filter_map(|id| {
                    self.string_map_rev
                        .get(id)
                        .map(|s| s.clone())   // keep only those still in the map
                })
                .collect::<Vec<_>>()
                .join(", ");
            let label = escape_label(&joined);
            writeln!(out, "    Node{} [label=\"{{ {} }}\"];", i, label).unwrap();
        }
    
        writeln!(out).unwrap();
    
        // Optional: cluster leaves
        writeln!(out, "    subgraph cluster_leaves {{").unwrap();
        writeln!(out, "        label = \"Leaves\";").unwrap();
        for &leaf_idx in self.leaves.values() {
            writeln!(out, "        Node{};", leaf_idx.0).unwrap();
        }
        writeln!(out, "    }}\n").unwrap();
    
        // Edges
        for i in 0..self.coreset_tree.len() {
            let parent = NodeIdx(i);
            for k in 0..CORESET_TREE_ARITY {
                if let Some(child) = self.child_idx(parent, k) {
                    writeln!(out, "    Node{} -> Node{};", i, child.0).unwrap();
                }
            }
        }
    
        writeln!(out, "}}").unwrap();
        out
    }

}



/// A Python module implemented in Rust.
#[pymodule]
#[pyo3(name="dynamic_csc")]
fn dynamic_csc(m: &Bound<'_, PyModule>) -> PyResult<()> {
    m.add_class::<DynamicCoreset>()?;
    m.add_class::<FasterDynamicCoreset>()?;
    Ok(())
}



pub fn random_operations_seeded(){

    // seed rng
    let seed: u64 = 501;

    let mut rng = StdRng::seed_from_u64(seed);

    let mut coreset = DynamicCoreset::new(
        100, 
        5, 2.0,
        1.0, 
        4, 
        1);

    const NUM_NODES: usize = 10_000;
    const NUM_UPDATES: usize = 5_000;

    let nodes: Vec<String> = (0..NUM_NODES)
        .map(|i| format!("Node{}", i))
        .collect();

    // Store the edges we inserted/updated
    let mut known_edges = FxHashMap::<(String,String), Float>::default();

    // simulate adding and removing edges. Insert withing 
    let insert_prob=0.5;


    let mut commands = vec!();

    for i in 0..NUM_UPDATES {
        match rng.random_range(0.0..1.0) < insert_prob{
            true => {
                // Insert an edge
                let u = nodes[rng.random_range(0..NUM_NODES)].clone();
                let v = nodes[rng.random_range(0..NUM_NODES)].clone();
                if u == v{
                    // don't insert self loops
                    continue;
                }
                let w = rng.random_range(1.0..10.0);

                println!("Inserting edge ({}, {}) with weight {}", &u, &v, &w);
                commands.push(format!("Insert ({}, {}), weight {}", &u, &v, w));
                coreset.rust_insert_edge(&u, &v, w).unwrap();
                known_edges.entry((u.clone(),v.clone())).and_modify(|e| *e += w).or_insert(w);
                known_edges.entry((v.clone(),u.clone())).and_modify(|e| *e += w).or_insert(w);

            },
            false => {
                // Delete an edge
                if !known_edges.is_empty(){

                    let mut known_edges_list = known_edges.iter().map(
                        |(k,v)| (k.clone(), *v)
                    ).collect::<Vec<_>>();
                    known_edges_list.sort_by( |a,b| a.0.partial_cmp(&b.0).unwrap() );
                    
                    let ((u,v),w) = known_edges_list.into_iter().choose(&mut rng).unwrap();
                    // with probability 0.5 delete the entire edge
                    if rng.random_range(0.0..1.0) < 0.5{
                        println!("Deleting edge ({}, {}) with weight {}", &u, &v, &w);
                        commands.push(format!("Delete ({}, {}), weight {}", &u, &v, w));
                        coreset.rust_delete_entire_edge(&u, &v).unwrap();
                        known_edges.remove(&(u.clone(),v.clone()));
                        known_edges.remove(&(v.clone(),u.clone()));
                    }else{
                        println!("Deleting half edge ({}, {}) with weight {}", &u, &v, w/2.0);
                        commands.push(format!("Delete half ({}, {}), weight {}", &u, &v, w/2.0));
                        // delete half the weight
                        let new_w: Float = w/2.0;
                        coreset.rust_delete_edge(&u, &v, w/2.0).unwrap();
                        // update the known edges
                        known_edges.entry((u.clone(),v.clone())).and_modify(|e| *e -= new_w).or_insert(new_w);
                        known_edges.entry((v.clone(),u.clone())).and_modify(|e| *e -= new_w).or_insert(new_w);
                    }
                }
            }
        }

        // produce the dot file
        let dot = coreset.to_dot("test_graph");
        let mut file = File::create(format!("test_output/graph_{:04}.dot", i)).unwrap();
        file.write_all(dot.as_bytes()).unwrap();

        // now check the adjacency list
        for (k,w) in known_edges.iter(){
            let u = &k.0;
            let v = &k.1;

            let u_name = coreset.string_map.get(u).unwrap();
            let v_name = coreset.string_map.get(v).unwrap();
            assert_eq!(coreset.adjacency.get(u_name).unwrap().get(v_name).unwrap(), w);
            assert_eq!(coreset.adjacency.get(v_name).unwrap().get(u_name).unwrap(), w);
        }

        let root_names = coreset.coreset_tree.iter().map(|x| x.indices[0].clone()).collect::<Vec<_>>();

        // check that the root indices are in the leaves map
        // check the nodes in the known_edges match exactly with the indices in the root:
        let mut expected_root_set = HashSet::new();
        for (u,v) in known_edges.keys(){
            expected_root_set.insert(u.to_string());
            expected_root_set.insert(v.to_string());
        }
        let mut actual_root_set: HashSet<String> = HashSet::new();
        for name in root_names{
            actual_root_set.insert(coreset.string_map_rev.get(&name).unwrap().clone());
        }

        // if the root sets differ, print the differences between them
        if expected_root_set != actual_root_set{
            println!("Expected root set: {:?}", expected_root_set);
            println!("Actual root set: {:?}", actual_root_set);
            // print the difference
            let diff = expected_root_set.symmetric_difference(&actual_root_set).collect::<Vec<_>>();
            println!("Difference: {:?}", diff);
            panic!("Root sets differ");
        }



        // // panic after 100 iterations
        // if i > 100{
        //     // write the commands to a file
        //     let mut file = File::create("test_output/commands.txt").unwrap();
        //     for command in commands{
        //         file.write_all(command.as_bytes()).unwrap();
        //         file.write_all(b"\n").unwrap();
        //     }
        //     panic!();
        // }

        // dbg!(&coreset.coreset_tree);
        // dbg!(&coreset.adjacency);
        // dbg!(&coreset.degrees);
        // dbg!(&coreset.leaves);

        // if coreset.adjacency.get(&NodeName("Node15".to_string())).and_then(|x| x.get(&NodeName("Node7".to_string()))).is_some(){
        //     panic!();
        // }

        // print edge weight of (Node24,Node15) if it exists:
        // println!("Edge weight of (Node15,Node7): {:?}", coreset.adjacency.get(&NodeName("Node15".to_string())).and_then(|x| x.get(&NodeName("Node7".to_string()))));
    }
}






#[cfg(test)]
mod tests {


    use super::*;

    #[test]
    fn test_insert_single_edge() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0, 1.0,1, 1);
        let u = "A".to_string();
        let v = "B".to_string();

        // Insert edge (A, B) with weight 2.0
        coreset.rust_insert_edge(&u, &v, 2.0).unwrap();

        let u_name = coreset.string_map.get(&u).unwrap();
        let v_name = coreset.string_map.get(&v).unwrap();

        // Check adjacency
        assert_eq!(coreset.adjacency[u_name].get(v_name), Some(&2.0));
        assert_eq!(coreset.adjacency[v_name].get(u_name), Some(&2.0));

        // Check degrees
        assert_eq!(*coreset.degrees.get(u_name).unwrap(), 2.0);
        assert_eq!(*coreset.degrees.get(v_name).unwrap(), 2.0);

        // Check leaves
        assert!(coreset.leaves.contains_key(u_name));
        assert!(coreset.leaves.contains_key(v_name));
    }

    #[test]
    fn test_insert_multiple_edges() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0, 1.0,1,1);
    
        let a = "A";
        let b = "B";
        let c = "C";
    
        // Insert edges
        coreset.rust_insert_edge(a, b, 1.0).unwrap();
        coreset.rust_insert_edge(a, c, 2.0).unwrap();
    
        let a_name = coreset.string_map.get(a).unwrap();
        let b_name = coreset.string_map.get(b).unwrap();
        let c_name = coreset.string_map.get(c).unwrap();
    
        // Check adjacency & degrees
        assert_eq!(coreset.adjacency[a_name].get(b_name), Some(&1.0));
        assert_eq!(coreset.adjacency[a_name].get(c_name), Some(&2.0));
        assert_eq!(*coreset.degrees.get(a_name).unwrap(), 3.0);
        assert_eq!(*coreset.degrees.get(b_name).unwrap(), 1.0);
        assert_eq!(*coreset.degrees.get(c_name).unwrap(), 2.0);
    }

    #[test]
    fn test_delete_partial_edge() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0, 1.0,1, 1);
    
        let a = "A";
        let b = "B";
    
        // Insert edge with weight 3.0
        coreset.rust_insert_edge(a, b, 3.0).unwrap();
    
        let a_name = coreset.string_map.get(a).unwrap().clone();
        let b_name = coreset.string_map.get(b).unwrap().clone();
    
        assert_eq!(coreset.adjacency[&a_name].get(&b_name), Some(&3.0));
    
        // Delete part of the edge weight (1.0)
        coreset.rust_delete_edge(a, b, 1.0).unwrap();
    
        // Check adjacency and degrees
        assert_eq!(coreset.adjacency[&a_name].get(&b_name), Some(&2.0));
        assert_eq!(coreset.adjacency[&b_name].get(&a_name), Some(&2.0));
        assert_eq!(*coreset.degrees.get(&a_name).unwrap(), 2.0);
        assert_eq!(*coreset.degrees.get(&b_name).unwrap(), 2.0);
    }

    #[test]
    fn test_delete_entire_edge() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0, 1.0,1, 1);
    
        let x = "X";
        let y = "Y";
    
        // Insert edge
        coreset.rust_insert_edge(x, y, 4.0).unwrap();
    
        // Interned names
        let x_name = coreset.string_map.get(x).unwrap().clone();
        let y_name = coreset.string_map.get(y).unwrap().clone();
    
        assert_eq!(coreset.adjacency[&x_name].get(&y_name), Some(&4.0));
    
        // Delete the entire edge in one call
        coreset.rust_delete_entire_edge(x, y).unwrap();
    
        // Edge should be gone
        assert!(!coreset.adjacency.contains_key(&x_name));
        assert!(!coreset.adjacency.contains_key(&y_name));
        // Leaves should be removed as well
        assert!(!coreset.leaves.contains_key(&x_name));
        assert!(!coreset.leaves.contains_key(&y_name));
        // Degrees should be removed
        assert!(!coreset.degrees.contains_key(&x_name));
        assert!(!coreset.degrees.contains_key(&y_name));
    }

    #[test]
    fn test_insert_duplicate_node() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0, 1.0,1,1 );
        let dup = "Dup";
        let other = "Other";
    
        // Insert once via an edge
        coreset.rust_insert_edge(dup, other, 1.0).unwrap();
    
        // Get the interned name
        let dup_name = coreset.string_map.get(dup).unwrap().clone();
    
        // Try inserting the same node explicitly again
        let res = coreset.insert_leaf_node_into_the_tree(dup_name.clone());
    
        assert!(matches!(res, Err(CoresetError::NodeAlreadyExists(n)) if coreset.string_map.get(&n).unwrap() == &dup_name));
    }

    #[test]
    fn test_delete_non_existent_edge() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0, 1.0,1,1 );
        let p = "P";
        let q = "Q";
        let r = "R";
    
        // Insert a single edge (P, R)
        coreset.rust_insert_edge(p, r, 1.0).unwrap();
    
        // Try to delete an edge (P, Q) that doesn't exist
        let result = coreset.rust_delete_entire_edge(p, q);
        assert!(matches!(result, Err(CoresetError::NodeNotFound(_))));
    }
    
    #[test]
    fn test_delete_node_not_present() {
        let mut coreset = DynamicCoreset::new(10, 5, 0.0,1.0, 1,1);
    
        // Attempt to delete an edge referencing nodes that were never inserted
        let result = coreset.rust_delete_edge("NotThere", "AlsoNotThere", 1.0);
    
        assert!(matches!(result, Err(CoresetError::NodeNotFound(_))));
    }
    

    #[test]
    fn test_complex_insertion_deletion() {
        let mut coreset = DynamicCoreset::new(20, 2, 0.0, 1.0,1,1);
    
        // Insert multiple edges
        coreset.rust_insert_edge("A", "B", 2.0).unwrap();
        coreset.rust_insert_edge("B", "C", 3.0).unwrap();
        coreset.rust_insert_edge("A", "D", 1.0).unwrap();
    
        let a = coreset.string_map.get("A").unwrap().clone();
        let b = coreset.string_map.get("B").unwrap().clone();
        let c = coreset.string_map.get("C").unwrap().clone();
        let d = coreset.string_map.get("D").unwrap().clone();
    
        // Adjacency checks
        assert_eq!(coreset.adjacency[&a].get(&b), Some(&2.0));
        assert_eq!(coreset.adjacency[&b].get(&c), Some(&3.0));
        assert_eq!(coreset.adjacency[&a].get(&d), Some(&1.0));
    
        // Degree checks
        assert_eq!(*coreset.degrees.get(&a).unwrap(), 3.0);
        assert_eq!(*coreset.degrees.get(&b).unwrap(), 5.0);
        assert_eq!(*coreset.degrees.get(&c).unwrap(), 3.0);
        assert_eq!(*coreset.degrees.get(&d).unwrap(), 1.0);
    
        // Partial delete (B,C) by 1.0
        coreset.rust_delete_edge("B", "C", 1.0).unwrap();
        assert_eq!(coreset.adjacency[&b].get(&c), Some(&2.0));
        assert_eq!(coreset.adjacency[&c].get(&b), Some(&2.0));
        assert_eq!(*coreset.degrees.get(&b).unwrap(), 4.0);
        assert_eq!(*coreset.degrees.get(&c).unwrap(), 2.0);
    
        // Delete entire edge (A,B)
        coreset.rust_delete_entire_edge("A", "B").unwrap();
        assert!(!coreset.adjacency[&a].contains_key(&b));
        assert!(!coreset.adjacency[&b].contains_key(&a));
        assert!(coreset.leaves.contains_key(&a));
        assert!(coreset.leaves.contains_key(&b));
        assert_eq!(*coreset.degrees.get(&a).unwrap(), 1.0);
        assert_eq!(*coreset.degrees.get(&b).unwrap(), 2.0);
    
        // Insert (B, E)
        coreset.rust_insert_edge("B", "E", 5.0).unwrap();
        let e = coreset.string_map.get("E").unwrap().clone();
        assert_eq!(coreset.adjacency[&b].get(&e), Some(&5.0));
        assert_eq!(*coreset.degrees.get(&b).unwrap(), 7.0);
        assert_eq!(*coreset.degrees.get(&e).unwrap(), 5.0);
    
        // Insert (A, C)
        coreset.rust_insert_edge("A", "C", 2.0).unwrap();
        assert_eq!(coreset.adjacency[&a].get(&c), Some(&2.0));
        assert_eq!(*coreset.degrees.get(&a).unwrap(), 3.0);
        assert_eq!(*coreset.degrees.get(&c).unwrap(), 4.0);
    
        // Delete (B, E)
        coreset.rust_delete_entire_edge("B", "E").unwrap();
        assert!(!coreset.adjacency[&b].contains_key(&e));
        assert!(!coreset.leaves.contains_key(&e));
        assert!(!coreset.degrees.contains_key(&e));
        assert_eq!(coreset.adjacency[&b].len(), 1);
        assert_eq!(*coreset.degrees.get(&b).unwrap(), 2.0);
    
        // Delete (C, B)
        coreset.rust_delete_entire_edge("C", "B").unwrap();
        assert!(!coreset.leaves.contains_key(&b));
        assert!(!coreset.degrees.contains_key(&b));
        assert_eq!(coreset.adjacency[&c].len(), 1);
        assert_eq!(*coreset.degrees.get(&c).unwrap(), 2.0);
    
        // Partial delete (A,C) by 1
        coreset.rust_delete_edge("A", "C", 1.0).unwrap();
        assert_eq!(coreset.adjacency[&a].get(&c), Some(&1.0));
        assert_eq!(coreset.adjacency[&c].get(&a), Some(&1.0));
        assert_eq!(*coreset.degrees.get(&a).unwrap(), 2.0);
        assert_eq!(*coreset.degrees.get(&c).unwrap(), 1.0);
    
        // Delete entire edge (A,C)
        coreset.rust_delete_entire_edge("A", "C").unwrap();
        assert!(!coreset.adjacency[&a].contains_key(&c));
        assert!(!coreset.adjacency.get(&c).map_or(false, |neigh| neigh.contains_key(&a)));
        assert!(coreset.leaves.contains_key(&a));
        assert_eq!(*coreset.degrees.get(&a).unwrap(), 1.0);
        assert!(!coreset.leaves.contains_key(&c));
        assert!(!coreset.degrees.contains_key(&c));
    
        // Final: A <-> D
        assert_eq!(coreset.adjacency[&a].get(&d), Some(&1.0));
        assert_eq!(coreset.adjacency[&d].get(&a), Some(&1.0));
        assert_eq!(*coreset.degrees.get(&a).unwrap(), 1.0);
        assert_eq!(*coreset.degrees.get(&d).unwrap(), 1.0);
        assert!(coreset.leaves.contains_key(&a));
        assert!(coreset.leaves.contains_key(&d));
        assert_eq!(coreset.leaves.len(), 2);
    }

    #[test]
    fn test_random_operations_seeded(){

        // This is a test to check that the random operations seeded function works
        // It will run the random operations seeded function and check that the adjacency list
        // is consistent with the degrees and leaves map.

        random_operations_seeded();
        
    }

}