use crate::faster::{DynamicCorest as RDynamicCorest, TreeData};
use crate::faster::common::{Float, Float_Dtype};



use pyo3::{ffi::{PyTupleObject, CO_FUTURE_UNICODE_LITERALS}, prelude::*, types::{PyList, PyTuple}, IntoPyObject, IntoPyObjectExt};
use numpy::IntoPyArray;
use numpy::ToPyArray;
use numpy::PyReadonlyArray1;


const ARITY: usize = 32;
// const ARITY: usize = 64;






#[pyclass]
pub struct FasterDynamicCoreset{
    coreset: RDynamicCorest<ARITY>,
}





#[pymethods]
impl FasterDynamicCoreset{
    #[new]
    #[pyo3(signature = (pid_target, filtering_constant, const_pid_output=None))]
    pub fn py_new(
        pid_target: Float_Dtype,
        filtering_constant: Float_Dtype,
        const_pid_output: Option<Float_Dtype>,
    ) -> Self{
        let coreset = RDynamicCorest::new_with_capacities(
            Float::from(pid_target),
            Float::from(filtering_constant),
            const_pid_output.map(|x| Float::from(x)),
            200_000, // initial capacity
            250 // arity
        );
        FasterDynamicCoreset { coreset }
    }

    pub fn insert_edge(&mut self, u: &str, v: &str, w: Float_Dtype){
        self.coreset.insert_edge(u, v, Float::from(w).into()).unwrap();
    }

    pub fn delete_edge_weighted(
        &mut self, u: &str, v: &str, w: Float_Dtype
    )  {
        self.coreset.delete_edge(u, v, Float::from(w).into()).unwrap();
            
    }

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

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

        let (
            coreset_node_names,
            coreset_weights,
            coreset_graph,
            stats
        ) = self.coreset.extract_coreset_graph(
            coreset_size,
            sampling_seeds).unwrap();

        let (symbolic, data) = coreset_graph.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.iter().map(|x|x.0).collect::<Vec<_>>().to_vec().into_pyarray(py).into_py_any(py)?;
        let nnz = nnz.unwrap().to_vec().into_pyarray(py).into_py_any(py)?;
        let coreset_node_names = PyList::new(py,coreset_node_names)?.into_py_any(py)?;
        let coreset_weights = coreset_weights.iter().map(|x| x.0).collect::<Vec<_>>().into_pyarray(py).into_py_any(py)?;

        let tuple = PyTuple::new(py,
            &[n, indptr,indices, data, nnz, coreset_node_names, coreset_weights])?;
        PyResult::Ok(tuple)
    }
    
    pub fn label_entire_graph<'py>(
        &self, 
        labels: PyReadonlyArray1<usize>,
        coreset_names: Bound<'py, PyList>,
        coreset_weights: PyReadonlyArray1<Float_Dtype>,
        num_clusters: usize,
        py: Python<'py>
    ) -> PyResult<Bound<'py, PyTuple>>{

        let names = coreset_names.iter().map(|x| x.extract::<String>().unwrap()).collect::<Vec<_>>();

        let (names, labels,distances) = self.coreset.rust_label_full_graph(
            names.as_slice(),
            coreset_weights.as_slice().unwrap(),
            labels.as_slice().unwrap(),
            num_clusters);

        let names = PyList::new(
            py,names.iter().map(|x|self.coreset.string_map_reverse.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)
    }
    

}