use crate::faster::DynamicCorest;
use crate::faster::common::*;

use std::ops::{AddAssign};
use rustc_hash::{FxHashMap};

impl<const ARITY: usize> DynamicCorest<ARITY> {

    pub fn num_vertices_in_graph(&self) -> usize {
        let n = self.string_map.len();
        debug_assert!(
            n == self.string_map_reverse.len() &&
            n == self.node_location_map.len() &&
            n == self.adjacency.len() &&
            n == self.degrees.len(),
            "Inconsistent node counts across data structures: \
             string_map: {}, string_map_reverse: {}, node_location_map: {}, \
             adjacency: {}, degrees: {}",
            n,
            self.string_map_reverse.len(),
            self.node_location_map.len(),
            self.adjacency.len(),
            self.degrees.len()
        );
        n
    }

    //MARK: Adjacency List and Degree Methods

    pub fn insert_into_adjacency_and_degrees(&mut self, from: NodeIdentity, to: NodeIdentity, weight: EdgeWeight,){


        let degree_on_change: NodeDegree = convert(weight);
        let initial_degree = degree_on_change + Float::from(1.0).into();

        // insert the edge into the adjacency list

        for &(src, dst) in &[(from, to), (to, from)] {
            self.adjacency
                .entry(src)
                .or_default()
                .entry(dst)
                .and_modify(|w| *w += weight)
                .or_insert_with(|| weight.clone());
        }


        for &node in &[from, to] {
            if !self
                .degrees
                .change_priority_by(&node, |p| *p += degree_on_change)
            {
                self.degrees.push(node, initial_degree);
            }
        }

    }

    pub fn delete_from_adjacency_and_degrees(&mut self, from: NodeIdentity, to: NodeIdentity, weight: EdgeWeight) -> Result<EdgeDeletionResult, DynamicCoresetError>{

        
        if from == to {
            return Err(DynamicCoresetError::NoSelfLoopsAllowed(
                self.string_map_reverse.get(&from).unwrap().clone(),
            ));
        }

        if let [Some(from_neighbours), Some(to_neighbours)] = self.adjacency.get_disjoint_mut(
            [&from,&to]
        ){
            if let (Some(edge_weight), Some(reverse_edge_weight)) = (
                from_neighbours.get_mut(&to),
                to_neighbours.get_mut(&from)
            ){
                self.degrees.change_priority_by(&from, |p| *p -= convert(weight));
                self.degrees.change_priority_by(&to, |p| *p -= convert(weight));
                edge_weight.add_assign(-weight);
                reverse_edge_weight.add_assign(-weight);
                if *edge_weight < FP_EPSILON.into() || *reverse_edge_weight < FP_EPSILON.into() {
                    from_neighbours.remove(&to);
                    to_neighbours.remove(&from);

                    let from_empty = from_neighbours.is_empty();
                    let to_empty   = to_neighbours.is_empty();
                    
                    return match (from_empty, to_empty){
                        (true,true) => {
                            // Remove both nodes from the adjacency list and degrees:
                            self.adjacency.remove(&from);
                            self.adjacency.remove(&to);
                            self.degrees.remove(&from);
                            self.degrees.remove(&to);
                            Ok(EdgeDeletionResult::BothNodesDisconnected(
                                self.string_map_reverse.get(&from).unwrap().clone(),
                                self.string_map_reverse.get(&to).unwrap().clone()))
                            },
                        (true,false) => {
                            // Remove the `from` node from the adjacency list and degrees:
                            self.adjacency.remove(&from);
                            self.degrees.remove(&from);
                            Ok(EdgeDeletionResult::OneNodeDisconnected(
                            self.string_map_reverse.get(&from).unwrap().clone()))
                        },
                        (false,true) => {
                            // Remove the `to` node from the adjacency list and degrees:
                            self.adjacency.remove(&to);
                            self.degrees.remove(&to);
                            Ok(EdgeDeletionResult::OneNodeDisconnected(
                            self.string_map_reverse.get(&to).unwrap().clone()))
                        },
                        (false,false) => Ok(EdgeDeletionResult::BothNodesStillConnected),
                    }
                } else {
                    // If not, we just update the edge weights:
                    return Ok(EdgeDeletionResult::BothNodesStillConnected);
                }
            } else{
                return Err(DynamicCoresetError::InvalidEdge(
                    self.string_map_reverse.get(&from).unwrap().clone(),
                    self.string_map_reverse.get(&to).unwrap().clone(),
                ));
            }
        }else{
            return Err(DynamicCoresetError::InvalidEdge(
                self.string_map_reverse.get(&from).unwrap().clone(),
                self.string_map_reverse.get(&to).unwrap().clone(),
            ));
        }
    }

}