#include "AgentOps.h"
#include <thread>
#include <mutex>
#include <omp.h> // OpenMP for parallelism

void agents_sort_abs_cust(std::vector<int>& A, const Instance& instance, char order) {
    static std::random_device rd;
    static std::mt19937 gen(rd());

    if (order == '\0') {
        static std::vector<char> options = { 'R', 'D', 'F', 'C' };
        static std::discrete_distribution<> dist({ 4, 4, 2, 1 });
        order = options[dist(gen)];
    }

    if (order == 'R') {
        std::shuffle(A.begin(), A.end(), gen);
    }
    else if (order == 'D') {
        std::sort(A.begin(), A.end(), [&](int a, int b) {
            return instance.demand[a] > instance.demand[b];
            });
    }
    else if (order == 'F') {
        std::sort(A.begin(), A.end(), [&](int a, int b) {
            return instance.distanceMatrix[a][0] > instance.distanceMatrix[b][0];
            });
    }
    else {
        std::sort(A.begin(), A.end(), [&](int a, int b) {
            return instance.distanceMatrix[a][0] < instance.distanceMatrix[b][0];
            });
    }
}

std::vector<std::vector<int>> agents_heuristic_deconstruction_selection(Solution & solution, float c_bar, int m) {
    std::vector<std::vector<int>> A;
    for (int i = 0; i < m; i++) {
        ModifiedSolution msol(solution);
        std::unordered_set<int> removedCustomers_set = msol.destroy(c_bar, 10, 0.01);
        std::vector<int>removedCustomers(removedCustomers_set.begin(), removedCustomers_set.end());
        A.push_back(removedCustomers);
    }
    return A;
}

std::tuple<Solution, std::vector<float>> agents_remove_recreate_allImp(Solution solution, std::vector<std::vector<int>>& A, float beta, int n, float T, bool insertInNewToursOnly) {
    ModifiedSolution bestSol(solution);
    std::vector<float> costs(A.size());
    for (int i = 0; i < A.size(); i++)
    {
        ModifiedSolution msol(solution);
        msol.removeCustomers(A[i]);
        std::vector<int>& removedCustomers = A[i];
        float bestCosts = std::numeric_limits<float>::infinity();
        for (int j = 0; j < n; j++)
        {
            if (j > 0) {
                agents_sort_abs_cust(removedCustomers, solution.instance, 'R');
            }
            ModifiedSolution msol_copy(msol);
            msol_copy.repair(removedCustomers, beta, insertInNewToursOnly);
            if (msol_copy.totalCosts < bestCosts)
            {
                bestSol = msol_copy;
                bestCosts = msol_copy.totalCosts;
            }
        }
        costs[i] = bestCosts;
        float thresh = solution.totalCosts - T * std::log(getRandomFraction(0,1));
        if (thresh > bestSol.totalCosts) {
           solution.acceptModifiedSolution(bestSol);
        }
    }
    return std::make_tuple(solution, costs); // TODO use a struct to return refs to objects
}