#include <clupig.h>

#include <pybind11/pybind11.h>
#include <pybind11/stl.h>
#include <pybind11/eigen.h>
#include <pybind11/numpy.h>

#include <sstream>

namespace python {

namespace py = pybind11;

PYBIND11_MODULE(clupig, m) { // Must be the same name with class Dbscan
    py::class_<clupig>(m, "clupig")
        .def(py::init<const int&, const int&>(),  py::arg("n_points"), py::arg("n_features"))
        
        .def_readonly("labels_", &clupig::labels) // must be def_readonly
//        .def_readonly("indices_", &clupig::indices_) // must be def_readonly
//        .def_readonly("distances_", &clupig::distances_) // must be def_readonly

//        .def_readonly("flat_indices_", &clupig::flat_indices_) // must be def_readonly
//        .def_readonly("flat_distances_", &clupig::flat_distances_) // must be def_readonly
//        .def_readonly("flat_offset_", &clupig::flat_offset_) // must be def_readonly

//    .def_readonly("matrix_flat_indices_", &clupig::matrix_flat_indices_,
//         py::return_value_policy::reference_internal) // must be def_readonly
//    .def_readonly("matrix_flat_distances_", &clupig::matrix_flat_distances_,
//       py::return_value_policy::reference_internal) // Risk of copy!
//        .def("matrix_flat_indices_", &clupig::get_indices,
//           py::return_value_policy::reference_internal)  // ← NO COPY
//        .def("matrix_flat_distances_", &clupig::get_distances,
//           py::return_value_policy::reference_internal)  // ← NO COPY


        .def_readonly("n_clusters_", &clupig::n_clusters) // must be def_readonly
        .def_readwrite("min_cluster_size", &clupig::min_cluster_size, "Change minimum of initialized cluster size.")
        .def_readwrite("propagation_cutoff", &clupig::propagation_cutoff, "Change neighbor_cutoff flag to reduce the time/space complexity.")

        .def("set_top_s", &clupig::set_top_s, py::arg("top_s"))
        .def("set_top_m", &clupig::set_top_m, py::arg("top_m"))
        .def("set_top_p", &clupig::set_top_p, py::arg("top_p"))
        .def("set_min_cluster_size", &clupig::set_min_cluster_size, py::arg("min_cluster_size"))
        .def("set_propagation_cutoff", &clupig::set_propagation_cutoff, py::arg("propagation_cutoff")=true)

        .def("set_threads", &clupig::set_threads, py::arg("n_threads"), "Change number of threads.")
        .def("clear", &clupig::clear)

        // clupig
        .def("brute_knn_dnp", &clupig::brute_knn_dnp, py::arg("X"), py::arg("k"), py::arg("c") = 1.0)

        // DNP from pre-computed kNN graph
        .def("dnp_from_knn", &clupig::dnp_from_knn, py::arg("indices"), py::arg("distances"), py::arg("k"), py::arg("c") = 1.0)


        ;

} // namespace clupig
} // namespace python
