use super::common::*;

use super::DynamicCorest;

use rand::{Rng, SeedableRng};
use rand::rngs::StdRng;
use indicatif::{ProgressBar};
use std::collections::HashMap;
use rand::prelude::IteratorRandom;

use rustc_hash::{FxHashMap, FxBuildHasher};

#[derive(Debug, Clone)]
pub enum Instruction{
    Insert(String, String, EdgeWeight),
    Delete(String, String),
    DeleteHalf(String, String, EdgeWeight),
}

pub fn generate_commands(seed: u64, num_nodes: usize, num_updates: usize, insert_prob: Float, full_delete_prob: f64) -> Vec<Instruction>{

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

    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>::with_capacity_and_hasher((num_updates as f32*1.5) as usize, FxBuildHasher::default());

    // Store the operations we are going to perform:
    let mut operations: Vec<Instruction> = Vec::with_capacity(num_updates);

    let pb = ProgressBar::new(num_updates as u64);
    pb.set_style(indicatif::ProgressStyle::default_bar()
        .template("{spinner:.green} {bar:.green/yellow} {decimal_bytes_per_sec} {eta} [{elapsed_precise}] ").unwrap());
        // .progress_chars("##-"));

    for _ in 0..num_updates {
        pb.inc(1);
        match rng.random_range(0.0..1.0) < insert_prob.0{
            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: Float = rng.random_range(1.0..10.0).into();
                operations.push(Instruction::Insert(u.clone(), v.clone(), w.into()));
                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 entry = known_edges.iter().choose(&mut rng).unwrap();
                    let u = entry.0.0.clone();
                    let v = entry.0.1.clone();
                    let w = entry.1.clone();
                    // with probability 0.5 delete the entire edge
                    if rng.random_range(0.0..1.0) < full_delete_prob{
                        operations.push(Instruction::Delete(u.clone(), v.clone()));
                        known_edges.remove(&(u.clone(),v.clone()));
                        known_edges.remove(&(v.clone(),u.clone()));
                    }else{
                        // delete half the weight
                        let new_w: Float = w/2.0;
                        operations.push(Instruction::DeleteHalf(u.clone(), v.clone(), new_w.into()));
                        // 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);
                    }
                }
            }
        }
    }
    pb.finish_with_message("Done generating instructions");
    operations
}


pub fn process_commands<const ARITY:usize>(coreset: &mut DynamicCorest<ARITY>, commands: &[Instruction])-> Result<(), DynamicCoresetError> 
where ConstPow2<ARITY>: PowerOfTwo
{

    let num_updates = commands.len();
    let pb = ProgressBar::new(num_updates as u64);
    pb.set_style(indicatif::ProgressStyle::default_bar()
        .template("{spinner:.green} {bar:.green/yellow} {decimal_bytes_per_sec} {eta} [{elapsed_precise}] ").unwrap());


    for command in commands.iter() {
        match command {
            Instruction::Insert(u, v, w) => {
                coreset.insert_edge(&u, &v, *w)?;
            },
            Instruction::Delete(u, v) => {
                coreset.delete_entire_edge(&u, &v)?;
            },
            Instruction::DeleteHalf(u, v, w) => {
                coreset.delete_edge(&u, &v, *w)?;
            }
        }
        pb.inc(1);
    }
    pb.finish_with_message("Done processing commands");
    Ok(())
}

pub fn get_node_identities<const ARITY: usize>(
    coreset: &DynamicCorest<ARITY>,
    node_names: &[&str],
) -> Result<Vec<NodeIdentity>, DynamicCoresetError> {
    let mut identities = Vec::with_capacity(node_names.len());
    for name in node_names {
        match coreset.string_map.get(*name) {
            Some(id) => identities.push(*id),
            None => return Err(DynamicCoresetError::NodeNotFound(name.to_string())),
        }
    }
    Ok(identities)
}

pub fn get_node_degrees<const ARITY: usize>(
    coreset: &DynamicCorest<ARITY>,
    node_names: &[&str],
) -> Result<Vec<(String, NodeIdentity, NodeDegree)>, DynamicCoresetError> {
    let mut degrees = Vec::with_capacity(node_names.len());

    for name in node_names {
        match coreset.string_map.get(*name) {
            Some(id) => {
                if let Some(degree) = coreset.degrees.get(id) {
                    degrees.push((name.to_string(), *id, degree.1.clone()));
                } else {
                    return Err(DynamicCoresetError::NodeNotFound(name.to_string()));
                }
            },
            None => return Err(DynamicCoresetError::NodeNotFound(name.to_string())),
        }
    }
    Ok(degrees)
}