// Module imports
mod common;
mod graph_impls;
mod tree_impls;
mod sampling_impls;
mod extraction_impls;
mod pid_impls;
pub mod wrapper;



#[cfg(test)]
mod tests;
pub mod test_utils;
use std::collections::HashMap;

// imports from the common module
use common::*;

use ordered_float::Float as FloatTrait;
// use pid::Pid;
use pid_control::{PIDController, Controller};
// 3rd party imports
use rustc_hash::{FxHashMap, FxBuildHasher};
use priority_queue::PriorityQueue;


//MARK: Tree Data Structures

pub struct TreeData<const ARITY: usize> {
    pub sizes: Vec<usize>,
    pub volumes: Vec<Volume>,
    pub inv_volumes: Vec<InvVolume>,
    pub deltas: Vec<Delta>,
    pub smoothing_term_deltas: Vec<SmoothingTermDelta>,
    pub generational_flags: Vec<usize>,
}



pub struct DynamicCorest<const ARITY: usize>{

    // maps for node string names to unique identifiers
    pub string_map: FxHashMap<String, NodeIdentity>,
    pub string_map_reverse: FxHashMap<NodeIdentity, String>,

    pub node_degree_guess: usize,
    // counter for unique node identifiers
    pub node_name_counter: NodeIdentity,
    // counter for query generation (used to decide if a delta is valid)
    pub node_generation_counter: usize,

    // map from node identities to their location in the leaves
    pub node_location_map: FxHashMap<NodeIdentity, ShiftedIndex>,
    // and the reverse map:
    pub node_location_map_reverse: FxHashMap<ShiftedIndex, NodeIdentity>,

    // Adjacency list and degree priority queue (backed by a hash map for fast lookups/updates)
    pub adjacency: FxHashMap<NodeIdentity, FxHashMap<NodeIdentity, EdgeWeight>>,
    pub degrees: PriorityQueue<NodeIdentity, NodeDegree, FxBuildHasher>,

    //Arrays to hold the Tree (Struct of Array style)
    pub tree_data: TreeData<ARITY>,

    pub shift: Float,
    pub pid: PIDController,
    pub filtered_average_distance_fraction: Float,
    pub filtering_constant: Float,
    pub constant_output: Option<Float>,
}


impl<const ARITY: usize> DynamicCorest<ARITY> 
    where ConstPow2<ARITY>: PowerOfTwo,
{
    // ARITY constants:
    const ARITY_SHIFT: u8 = ARITY.trailing_zeros() as u8;

    // MARK: Constructors
    pub fn new(pid_target: Float, filtering_constant: Float) -> Self {

        // setup PID: 
        let mut shift_pid = PIDController::new(
            -0.1, // proportional gain
            -0.75, // integral gain,
            0.01, // derivative gain
        );

        shift_pid.set_target(pid_target.0 as f64);
        shift_pid.set_limits(0.0, 5.0);


        DynamicCorest {
            string_map: FxHashMap::default(),
            string_map_reverse: FxHashMap::default(),
            node_degree_guess: 8, // A guess for the average degree of a node
            node_name_counter: NodeIdentity(0),
            node_generation_counter: 0,
            node_location_map: FxHashMap::default(),
            node_location_map_reverse: FxHashMap::default(),
            adjacency: FxHashMap::default(),
            degrees: PriorityQueue::with_hasher(FxBuildHasher::default()),
            tree_data: TreeData{
                sizes: Vec::new(),
                volumes: Vec::new(),
                inv_volumes: Vec::new(),
                deltas: Vec::new(),
                smoothing_term_deltas: Vec::new(),
                generational_flags: Vec::new(),
            },
            shift: Float::from(1.0),
            pid: shift_pid,
            filtered_average_distance_fraction: Float::neg_infinity(),
            filtering_constant,
            constant_output: None
        }
    }

    pub fn new_with_capacities(
        pid_target: Float,
        filtering_constant: Float,
        pid_constant_output: Option<Float>,
        num_nodes: usize,
        node_degree_guess: usize,
    ) -> Self{

        // work out the number of nodes in a D-arity tree with `num_nodes` leaves
        let num_tree_nodes = ((num_nodes as f32 - 1.0)/ (ARITY as f32 - 1.0)) as usize;

        // setup PID:
        let mut shift_pid = PIDController::new(
            -0.1, // proportional gain
            -0.75, // integral gain,
            0.00, // derivative gain
        );
        shift_pid.set_target(pid_target.0 as f64);
        shift_pid.set_limits(0.0, 5.0);


        DynamicCorest {
            string_map: FxHashMap::with_capacity_and_hasher(num_nodes, FxBuildHasher::default()),
            string_map_reverse: FxHashMap::with_capacity_and_hasher(num_nodes, FxBuildHasher::default()),
            node_degree_guess,
            node_name_counter: NodeIdentity(0),
            node_generation_counter: 0,
            node_location_map: FxHashMap::with_capacity_and_hasher(num_nodes, FxBuildHasher::default()),
            node_location_map_reverse: FxHashMap::with_capacity_and_hasher(num_nodes, FxBuildHasher::default()),
            adjacency: FxHashMap::with_capacity_and_hasher(num_nodes, FxBuildHasher::default()),
            degrees: PriorityQueue::with_capacity_and_hasher(num_nodes, FxBuildHasher::default()),
            tree_data: TreeData{
                sizes: Vec::with_capacity(num_tree_nodes),
                volumes: Vec::with_capacity(num_tree_nodes),
                inv_volumes: Vec::with_capacity(num_tree_nodes),
                deltas: Vec::with_capacity(num_tree_nodes),
                smoothing_term_deltas: Vec::with_capacity(num_tree_nodes),
                generational_flags: Vec::with_capacity(num_tree_nodes),
            },
            shift: pid_constant_output.unwrap_or(Float::from(1.0)),
            pid: shift_pid,
            filtered_average_distance_fraction: Float::neg_infinity(),
            filtering_constant: filtering_constant,
            constant_output: pid_constant_output,
        }

    }

    pub fn insert_edge(&mut self, from: &str, to: &str, weight: EdgeWeight) -> Result<(), DynamicCoresetError>{

        // Ensure this is not a self-loop:
        if from == to {
            return Err(DynamicCoresetError::NoSelfLoopsAllowed(from.to_string()));
        }

        // We add them to the string map if they don't exist and get references to them:
        self.string_map
            .entry(from.to_string())
            .or_insert_with(|| {
                let id = self.node_name_counter;
                self.node_name_counter.0 += 1;
                self.string_map_reverse.insert(id, from.to_string());
                id
            });
        self.string_map
            .entry(to.to_string())
            .or_insert_with(|| {
                let id = self.node_name_counter;
                self.node_name_counter.0 += 1;
                self.string_map_reverse.insert(id, to.to_string());
                id
            });

        let from_id = self.string_map.get(from).unwrap().clone();
        let to_id = self.string_map.get(to).unwrap().clone();


        // Insert them into the tree if they don't exist or modify and propagate changes if they do:
        for node in [from_id, to_id].iter(){
            if !self.node_location_map.contains_key(node) {
                // If the node does not exist in the tree, we insert it:
                self.insert_node_into_tree(*node, weight);
            }else{
                // If the node exists, we just update the weight:
                let leaf_index = self.node_location_map.get(node).unwrap().clone();
                self.tree_data.volumes[leaf_index.0] += convert(weight);
                self.tree_data.inv_volumes[leaf_index.0] = convert(self.tree_data.volumes[leaf_index.0].inv());
                Self::propagate_up_with_closure(&mut self.tree_data, leaf_index, |this, parent| {
                    this.volumes[parent.0] += convert(weight);
                    this.inv_volumes[parent.0] += convert(weight.inv());
                });
            }
        }

        // Finally, we insert the edge into the adjacency list and update degrees:
        self.insert_into_adjacency_and_degrees(from_id, to_id, weight);
        Ok(())
    }

    pub fn delete_entire_edge(&mut self, from: &str, to: &str) -> Result<(), DynamicCoresetError>{
        // Ensure this is not a self-loop:
        if from == to {
            return Err(DynamicCoresetError::NoSelfLoopsAllowed(from.to_string()));
        }
        // ensure both nodes exist in the string map:
        if !self.string_map.contains_key(from){
            return Err(DynamicCoresetError::NodeNotFound(from.to_string()));
        }
        if !self.string_map.contains_key(to){
            return Err(DynamicCoresetError::NodeNotFound(to.to_string()));
        }

        let from_id = self.string_map.get(from).unwrap().clone();
        let to_id = self.string_map.get(to).unwrap().clone();

        // get the edge weight of from -> to:
        let edge_weight = self.adjacency.get(&from_id).unwrap()
            .get(&to_id);

        if edge_weight.is_none() {
            return Err(DynamicCoresetError::InvalidEdge(
                self.string_map_reverse.get(&from_id).unwrap().clone(),
                self.string_map_reverse.get(&to_id).unwrap().clone(),
            ));
        }
        let edge_weight = edge_weight.unwrap().clone();
        // Now we can delete the edge:
        self.delete_edge(from, to, edge_weight)
    }

    pub fn delete_edge(&mut self, from: &str, to: &str, weight: EdgeWeight) -> Result<(), DynamicCoresetError>{
        // Ensure this is not a self-loop:
        if from == to {
            return Err(DynamicCoresetError::NoSelfLoopsAllowed(from.to_string()));
        }

        // ensure both nodes exist in the string map:
        if !self.string_map.contains_key(from){
            return Err(DynamicCoresetError::NodeNotFound(from.to_string()));
        }
        if !self.string_map.contains_key(to){
            return Err(DynamicCoresetError::NodeNotFound(to.to_string()));
        }

        let from_id = self.string_map.get(from).unwrap().clone();
        let to_id = self.string_map.get(to).unwrap().clone();

        let deletion_result = self.delete_from_adjacency_and_degrees(from_id, to_id, weight)?;
        match deletion_result {
            EdgeDeletionResult::BothNodesDisconnected(_,_) => {
                // If both nodes are disconnected, we can delete them from the tree:
                self.delete_node_from_tree(from_id);
                self.delete_node_from_tree(to_id);

                // clear the string maps for these nodes:
                self.string_map.remove(from);
                self.string_map.remove(to);
                self.string_map_reverse.remove(&from_id);
                self.string_map_reverse.remove(&to_id);
            },
            EdgeDeletionResult::OneNodeDisconnected(node) => {
                // If one node is disconnected, we delete it from the tree:
                self.delete_node_from_tree(if node == from.to_string() { from_id } else { to_id });

                // clear the string map for this node:
                let node_id = self.string_map.remove(&node).unwrap();
                self.string_map_reverse.remove(&node_id);

                // We still need to modify the tree node for the node left in the graph (subtract the change in volume )
                let othernode = { if node.as_str() == from { to_id } else { from_id } };
                // Get the leaf index of the other node:
                let leaf_index = self.node_location_map.get(&othernode).unwrap().clone();
                
                // save the old volume_inv to compute the difference later
                let old_volume_inv = self.tree_data.inv_volumes[leaf_index.0];

                // update the volumes and inv_volumes of the leaf node:
                // note that for leaf nodes, volume equals 1/(degree)
                self.tree_data.volumes[leaf_index.0] -= convert(weight);
                self.tree_data.inv_volumes[leaf_index.0] = convert(self.tree_data.volumes[leaf_index.0].inv());

                let inv_vol_diff = self.tree_data.inv_volumes[leaf_index.0] - old_volume_inv;

                // Now we propagate the changes up the tree:
                Self::propagate_up_with_closure(&mut self.tree_data, leaf_index, |this, parent_index|{
                    this.volumes[parent_index.0] -= convert(weight);
                    this.inv_volumes[parent_index.0] += inv_vol_diff;
                });
                
            },
            EdgeDeletionResult::BothNodesStillConnected => {
                // If both nodes are still connected, we do nothing except update the volumes and inv_volumes of the leaf nodes and 
                // propagate the changes up the tree:

                for node in [from_id, to_id].iter(){
                    // Get the leaf index of the node:
                    let leaf_index = self.node_location_map.get(node).unwrap().clone();
                    
                    // save the old volume_inv to compute the difference later
                    let old_volume_inv = self.tree_data.inv_volumes[leaf_index.0];

                    // update the volumes and inv_volumes of the leaf node:
                    // note that for leaf nodes, volume equals 1/(degree)
                    self.tree_data.volumes[leaf_index.0] -= convert(weight);
                    self.tree_data.inv_volumes[leaf_index.0] = convert(self.tree_data.volumes[leaf_index.0].inv());

                    let inv_vol_diff = self.tree_data.inv_volumes[leaf_index.0] - old_volume_inv;

                    // Now we propagate the changes up the tree:
                    Self::propagate_up_with_closure(&mut self.tree_data, leaf_index, |this, parent_index|{
                        this.volumes[parent_index.0] -= convert(weight);
                        this.inv_volumes[parent_index.0] += inv_vol_diff;
                    });
                }
            }
        }
        Ok(())
    }

    pub fn get_capacities(&self) -> HashMap<String, (usize,usize)>{

        // return a hashmap of the names of each datastructure and their sizes/capacities:
        let mut capacities = HashMap::new();
        
        capacities.insert("string_map".to_string(), (self.string_map.len(), self.string_map.capacity()));
        capacities.insert("string_map_reverse".to_string(), (self.string_map_reverse.len(), self.string_map_reverse.capacity()));
        capacities.insert("node_location_map".to_string(), (self.node_location_map.len(), self.node_location_map.capacity()));
        capacities.insert("node_location_map_reverse".to_string(), (self.node_location_map_reverse.len(), self.node_location_map_reverse.capacity()));
        capacities.insert("adjacency".to_string(), (self.adjacency.len(), self.adjacency.capacity()));
        capacities.insert("degrees".to_string(), (self.degrees.len(), self.degrees.capacity()));
        capacities.insert("sizes".to_string(), (self.tree_data.sizes.len(), self.tree_data.sizes.capacity()));
        capacities.insert("volumes".to_string(), (self.tree_data.volumes.len(), self.tree_data.volumes.capacity()));
        capacities.insert("inv_volumes".to_string(), (self.tree_data.inv_volumes.len(), self.tree_data.inv_volumes.capacity()));
        capacities.insert("deltas".to_string(), (self.tree_data.deltas.len(), self.tree_data.deltas.capacity()));
        capacities.insert("generational_flags".to_string(), (self.tree_data.generational_flags.len(), self.tree_data.generational_flags.capacity()));
        capacities
    }

}
