#include <queue>
#include <vector>
#include <utility>
#include <limits>
#include <cassert>

#ifdef TIME
    #include <time.hxx>
#endif

/*
Greedy arc fixation algorithm proposed by 

Böcker, S., Briesemeister, S. & Klau, G.W. 
On optimal comparability editing with applications to molecular diagnostics. 
BMC Bioinformatics 10 (Suppl 1), S61 (2009). https://doi.org/10.1186/1471-2105-10-S1-S61
*/


namespace preordering 
{


template<typename ARC_COSTS, typename ADJACENCY>
std::pair<typename ARC_COSTS::VALUE_TYPE, ADJACENCY> greedy_arc_fixation(
    const ARC_COSTS& arc_costs
){
    #ifdef TIME
        Clock clock;
    #endif
    typedef typename ARC_COSTS::VALUE_TYPE COST_TYPE;

    struct Fix {
        size_t u;
        size_t v;
        COST_TYPE loss;
        size_t fix; // 0 or 1

        bool operator<(const Fix& other) const {
            return loss < other.loss;
        }
    };

    size_t n = arc_costs.size();
    COST_TYPE inf = std::numeric_limits<COST_TYPE>::infinity();

    ARC_COSTS pos(n);
    ARC_COSTS neg(n);
    ARC_COSTS loss0(n);  // loss0(i, j) is the implied disagreement when setting arc i->j to 0
    ARC_COSTS loss1(n);  // loss1(i, j) is the implied disagreement when setting arc i->j to 1

    for (size_t i = 0; i < n; ++i)
    for (size_t j = 0; j < n; ++j)
    {
        if (arc_costs(i, j) > 0)
        {
            loss0(i, j) = arc_costs(i, j);
            pos(i, j) = arc_costs(i, j);
        }
        else 
        {
            loss1(i, j) = -arc_costs(i, j);
            neg(i, j) = -arc_costs(i, j);
        }
    }

    for (size_t i = 0; i < n; ++i)
    {
        loss0(i, i) = -inf;
        loss1(i, i) = -inf;
        for (size_t j = 0; j < n; ++j)
        {
            if (i == j) 
                continue;
            for (size_t k = 0; k < n; ++k)
            {
                if (k == i || k == j) 
                    continue;
                loss0(i, j) += std::min(pos(i, k), pos(k, j));
                loss1(i, j) += std::min(neg(i, k), pos(j, k));
                loss1(i, j) += std::min(neg(k, j), pos(k, i));
            }
        }
    }

    ADJACENCY adjacency(n, -1);
    for (size_t i = 0; i < n; ++i)
        adjacency(i, i) = 1;

    auto get_queue_threshold = [&] () 
    {
        return 0.0;
        COST_TYPE max_loss = 0;
        for (size_t i = 0; i < n; ++i)
        for (size_t j = 0; j < n; ++j)
        {
            if (adjacency(i, j) != -1)
                continue;
            if (loss0(i, j) > max_loss)
                max_loss = loss0(i, j);
            if (loss1(i, j) > max_loss)             
                max_loss = loss1(i, j);
        }
        return max_loss * 0.9;
    };

    COST_TYPE queue_threshold = get_queue_threshold();

    std::priority_queue<Fix> queue;

    auto refill_queue = [&] ()
    {
        for (size_t i = 0; i < n; ++i)
        for (size_t j = 0; j < n; ++j)
        {
            if (adjacency(i, j) != -1)
                continue;
            if (loss0(i, j) >= queue_threshold)
                queue.push({i, j, loss0(i, j), 1});
            if (loss1(i, j) >= queue_threshold)
                queue.push({i, j, loss1(i, j), 0});
        }
    };

    refill_queue();

    COST_TYPE total_cost = 0;

    #ifdef TIME
        std::cout << std::setprecision(5) << "t_setup:  " << clock.elapsed() << "\n";   
        double t_queue;
        double t_00 = 0;
        double t_01_neg = 0;
        double t_01_pos = 0;
        double t_10 = 0;
        double t_11 = 0;
    #endif


    auto change_loss = [&] (size_t u, size_t v, COST_TYPE change, size_t fix) 
    {
        if (change != 0)
        {
            if (fix == 0)
            {
                loss1(u, v) += change;
                if (loss1(u, v) >= queue_threshold)
                    queue.push({u, v, loss1(u, v), 0});
            }
            else
            {
                loss0(u, v) += change;
                if (loss0(u, v) >= queue_threshold)
                    queue.push({u, v, loss0(u, v), 1});
            }
        }
    };


    for (size_t num_free_arcs = n * (n-1); num_free_arcs > 0; --num_free_arcs)
    {
        #ifdef TIME
            clock.reset();
        #endif

        Fix best_fix;
        while (true)
        {
            if (queue.empty())
            {
                queue_threshold = get_queue_threshold();
                refill_queue();
            }

            best_fix = queue.top();
            queue.pop();
            if ((best_fix.fix == 0 && loss1(best_fix.u, best_fix.v) == best_fix.loss) || 
                (best_fix.fix == 1 && loss0(best_fix.u, best_fix.v) == best_fix.loss))
                break;
        }

        size_t u = best_fix.u;
        size_t v = best_fix.v;
        size_t fix = best_fix.fix;

        #ifdef TIME
            t_queue += clock.elapsed();
            clock.reset();
        #endif

        assert (adjacency(u, v) == -1);

        adjacency(u, v) = fix;
        total_cost += arc_costs(u, v) * fix;
        
        COST_TYPE change;
        if (fix == 1)
        {
            // update loss0
            for (size_t i = 0; i < n; ++i)
            {
                if (adjacency(u, i) == -1)
                {   
                    change_loss(u, i, -std::min(pos(u, v), pos(v, i)) + pos(v, i), 1);
                }
                if (adjacency(i, v) == -1)
                {
                    change_loss(i, v, -std::min(pos(u, v), pos(i, u)) + pos(i, u), 1);
                }
            }
            loss0(u, v) = -inf;

            #ifdef TIME
                t_10 += clock.elapsed();
                clock.reset();
            #endif
                
            // update loss1
            if (neg(u, v) > 0)
            {
                for (size_t i = 0; i < n; ++i)
                {
                    if (adjacency(u, i) == -1)
                    {
                        change_loss(u, i, -std::min(neg(u, v), pos(i, v)), 0);
                    }
                    if (adjacency(i, v) == -1)
                    {
                        change_loss(i, v, -std::min(neg(u, v), pos(u, i)), 0);
                    }
                }
            }
            if (pos(u, v) < inf)
            {
                for (size_t i = 0; i < n; ++i)
                {
                    if (adjacency(v, i) == -1)
                    {
                        change_loss(v, i, -std::min(pos(u, v), neg(u, i)) + neg(u, i), 0);
                    }
                    if (adjacency(i, u) == -1)
                    {
                        change_loss(i, u, -std::min(pos(u, v), neg(i, v)) + neg(i, v), 0);
                    }
                }
            }
            loss1(u, v) = -inf;

            #ifdef TIME
                t_11 += clock.elapsed();
                clock.reset();
            #endif

            // update costs
            pos(u, v) = inf;
            neg(u, v) = 0;
        }
        else // fix == 0
        {
            // update loss0
            if (pos(u, v) > 0)
            {
                for (size_t i = 0; i < n; ++i)
                {
                    if (adjacency(u, i) == -1)
                    {
                        change_loss(u, i, -std::min(pos(u, v), pos(v, i)), 1);
                    }
                    if (adjacency(i, v) == -1)
                    {
                        change_loss(i, v, -std::min(pos(u, v), pos(i, u)), 1);
                    }
                }
            }
            loss0(u, v) = -inf;

            #ifdef TIME
                t_00 += clock.elapsed();
                clock.reset();
            #endif

            // update loss1
            if (pos(u, v) > 0)
            {
                for (size_t i = 0; i < n; ++i)
                {
                    if (adjacency(v, i) == -1)
                    {
                        change_loss(v, i, -std::min(pos(u, v), neg(u, i)), 0);
                    }
                    if (adjacency(i, u) == -1)
                    {
                        change_loss(i, u, -std::min(pos(u, v), neg(i, v)), 0);
                    }
                }
            }

            #ifdef TIME
                t_01_pos += clock.elapsed();
                clock.reset();
            #endif

            if (neg(u, v) < inf)
            {
                for (size_t i = 0; i < n; ++i)
                {
                    if (adjacency(u, i) == -1)
                    {
                        change_loss(u, i, -std::min(neg(u, v), pos(i, v)) + pos(i, v), 0);
                    }
                    if (adjacency(i, v) == -1)
                    {
                        change_loss(i, v, -std::min(neg(u, v), pos(u, i)) + pos(u, i), 0);
                    }
                }
            }
            loss1(u, v) = -inf;

            #ifdef TIME
                t_01_neg += clock.elapsed();
                clock.reset();
            #endif

            // update costs
            pos(u, v) = 0;
            neg(u, v) = inf;
        }
    }

    #ifdef TIME
        std::cout << "t_queue: " << t_queue << "\n";
        std::cout << "t_00: " << t_00 << "\n";
        std::cout << "t_01_neg: " << t_01_neg << "\n";
        std::cout << "t_01_pos: " << t_01_pos << "\n";
        std::cout << "t_10: " << t_10 << "\n";
        std::cout << "t_11: " << t_11 << "\n";
    #endif

    return {total_cost, adjacency};
}

}  // namespace preordering

