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

#include <tuple>

#include <arc-properties.hxx>
#include <greedy-arc-fixation.hxx>
#include <moving.hxx>
#include <greedy-arc-insertion.hxx>

// #define VERBOSE
// #define TIME

namespace py = pybind11;

typedef preordering::ArcProperties<double> ACM;
typedef preordering::ArcProperties<int> AVM;
typedef std::vector<std::vector<int>> VALUE_VECTOR;
typedef std::vector<std::vector<double>> COST_VECTOR;
 

std::pair<double, VALUE_VECTOR>
greedy_arc_fixation(
    const COST_VECTOR& cost_matrix
 ) {
    auto result = preordering::greedy_arc_fixation<ACM, AVM>(cost_matrix);
    return {std::get<0>(result), std::get<1>(result).data()};
}

std::pair<double, VALUE_VECTOR>
greedy_moving(
    const COST_VECTOR& cost_matrix, const VALUE_VECTOR& adjacency
 ) {
    ACM acm(cost_matrix);
    AVM avm(adjacency);
    double gain = preordering::greedy_moving(acm, avm);
    return {gain, avm.data()};
}

std::pair<double, VALUE_VECTOR>
greedy_arc_insertion(
    const COST_VECTOR& cost_matrix, const VALUE_VECTOR& adjacency
 ) {
    ACM acm(cost_matrix);
    AVM avm(adjacency);
    double objective = preordering::greedy_arc_insertion(acm, avm);
    return {objective, avm.data()};
}


// wrap as Python module
PYBIND11_MODULE(preordering_local_search, m)
{
    m.def("greedy_arc_fixation", &greedy_arc_fixation);
    m.def("greedy_moving", &greedy_moving);
    m.def("greedy_arc_insertion", &greedy_arc_insertion);
}
