
#include <cassert>
#include <chrono>
#include <iostream>
#include <Rcpp.h>
#include "auxiliary.h"
#include "OrderScoring.h"
#include "RightOrder.h"
#include "LeftOrder.h"
#include "opruner_right.h"
#include "path_pruning.h"

double EPSILON = 0.0000001;

using namespace std;

/**
 * @brief Swaps two nodes in the order ro.
 * @param lower the lower index
 * @param upper the upper index
 * @param ro the RightOrder
 * @param scoring the OrderScoring
 */
void swap_nodes(const int lower, const int upper, RightOrder &ro, OrderScoring &scoring)
{
    int node1 = ro.order[lower];
    int node2 = ro.order[upper];
    const auto &[node1_scoretmp, node2_scoretmp] = scoring.swap_nodes(lower, upper, ro.order, ro.node_scores);
    ro.order_score = ro.order_score - (ro.node_scores[node1] + ro.node_scores[node2]) + (node1_scoretmp + node2_scoretmp);
    ro.node_scores[node1] = node1_scoretmp;
    ro.node_scores[node2] = node2_scoretmp;
}

/**
 * @brief Checks if the new node is optimal at the front of the order.
 * @param ro the RightOrder
 * @param new_node the new node
 * @param scoring the OrderScoring
 * @return
 */
bool optimal_front(const RightOrder &ro,
                   size_t new_node,
                   OrderScoring &scoring)
{
    if (definitelyGreaterThan(ro.inserted_max_order_scores[new_node], ro.new_top_scores[new_node], EPSILON)) // a > b
    {
        return (false);
    }
    return (true);
}

/**
 * @brief Checks if the front node is actually optimal at the front of the order.
 * @param ro the RightOrder
 * @param scoring the OrderScoring
 * @return
 */
bool optimal_front(const RightOrder &ro,
                   OrderScoring &scoring)
{
    RightOrder ro_tmp(ro);
    int p = ro.order.size();

    for (int i = ro.front_ind(); i < p - 1; ++i)
    {
        swap_nodes(i, i + 1, ro_tmp, scoring);
        if (definitelyGreaterThan(ro_tmp.order_score, ro.order_score, EPSILON))
        {
            return (false);
        }
    }
    return (true);
}

/**
 * @brief Checks if the new node is independent at the front of the order.
 * @param ro the RightOrder
 * @param top_scores the top scores
 * @param scoring the OrderScoring
 * @return true if the new node is independent at the front of the order.
 */
bool independent_front(const RightOrder &ro,
                       const vector<double> &top_scores,
                       OrderScoring &scoring)
{
    size_t x = ro.front();
    return (approximatelyEqual(top_scores[x], ro.node_scores[x], EPSILON));
}

/**
 * @brief Moves an unsscored node to the right of the sub order and rescores the order (ie makes it visible).
 *        Eg if x is the node to score: ([0,x,2,3],5,6,4) -> ([0,2,3],x,5,6,4)
 * @param from_index the index of the node to move
 * @param to_index the index to move the node to
 * @param ro the RightOrder
 * @param scoring the OrderScoring
 */
void make_visible(int from_index,
                  int to_index,
                  RightOrder &ro,
                  OrderScoring &scoring)
{
    int p = ro.order.size();
    int node = ro.order[from_index];
    int n = ro.n_nodes;

    if (to_index < p - n)
    {
        // If the new node is to be put somewhere in the order.
        // [0,(1),2,3|5,6,4] -> [(1),0,2,3|5,6,4]
        move_element(ro.order, from_index, to_index);                 // put it in the back [0,(1),2,3,|5,6,4] -> [(1),0,2,3,|,5,6,4]
        ro.node_scores[node] = scoring.score_pos(ro.order, to_index); // O(p)
        ro.order_score += ro.node_scores[node];
    }

    if (to_index >= p - n)
    {
        // If the new node is to be put on the very left of the sub order
        // ([0,x,2,3],5,6,4) -s> ([0,2,3],x,5,6,4)
        move_element(ro.order, from_index, p - n - 1);                 // put it in the back [0,(1),2,3,|5,6,4] -> [0,2,3,|(1),5,6,4]
        ro.node_scores[node] = scoring.score_pos(ro.order, p - n - 1); // O(p)?
        ro.order_score += ro.node_scores[node];

        // If the new node is put somewhere in the sub order we have to shift it in from the left.
        // [0,(1),2,3,|5,6,4] -> [0,2,3,|(1),5,6,4] -> [0,2,3,|5,(1),6,4]
        for (int i = p - n - 1; i < to_index; ++i)
        {
            swap_nodes(i, i + 1, ro, scoring);
        }
    }
    ro.n_nodes++;
}
/// @brief The maximum score when the node at index node_ind is inserted among the visible ones.
/// @param ro RightOrder
/// @param node_ind
/// @param scoring
/// @return
double max_inserted_score(const RightOrder &ro,
                          size_t node_ind,
                          OrderScoring &scoring)
{
    RightOrder ro_tmp(ro);
    make_visible(node_ind, ro.front_ind() - 1, ro_tmp, scoring);
    double maxsc = ro_tmp.order_score;
    for (size_t i = ro_tmp.front_ind(); i < ro.order.size() - 1; i++)
    {
        swap_nodes(i, i + 1, ro_tmp, scoring);

        if (definitelyGreaterThan(ro_tmp.order_score, maxsc, EPSILON))
        {
            maxsc = ro_tmp.order_score;
        }
    }
    return (maxsc);
}

bool equal_and_unordered_top(const RightOrder &ro, int new_front, OrderScoring &scoring)
{
    size_t p = ro.order.size();

    if (ro.best_insert_pos[new_front] == ro.front_ind()) // So we can only compare when its best at right after the top.. sort of..
    {
        // If new_node is best inserted as the second top ([..i..],a,b,c) before, and best pos for i is s([...],a,i,b,c)
        if (approximatelyEqual(ro.new_top_scores[new_front], ro.inserted_max_order_scores[new_front], EPSILON))
        {
            // Compare a > i in ([...], i,a,b,c) and ([...], a,i,b,c)
            if (new_front > ro.front())
            {
                // If new_node can be added as new front and has higher value, prune.
                return (true);
            }
        }
    }
    return (false);
}

bool equal_and_unordered_top(RightOrder &ro, OrderScoring &scoring)
{
    int p = scoring.numparents.size();
    int n = ro.n_nodes;
    int node1 = ro.order[p - n];
    int node2 = ro.order[p - n + 1];
    double node1_score = ro.node_scores[node1];
    double node2_score = ro.node_scores[node2];

    const auto &[node1_score_swap, node2_score_swap] = scoring.swap_nodes(p - n, p - n + 1, ro.order, ro.node_scores); // This should update the orderscore directly maybe.
    myswap(p - n, p - n + 1, ro.order);                                                                                // swap back

    if (approximatelyEqual(node1_score + node2_score, node1_score_swap + node2_score_swap, EPSILON))
    {
        if (node1 > node2)
        {
            return (true);
        }
    }
    return (false);
}

bool has_hidden_gap(const RightOrder &ro,
                    const vector<double> &top_scores,
                    OrderScoring &scoring)
{
    size_t p = ro.order.size();
    size_t n = ro.n_nodes;

    // Checking insertion of each of the hidden nodes.
    for (size_t node_index = 0; node_index < p - n; node_index++)
    {
        int inserted_node = ro.order[node_index];
        double score_at_very_top = ro.order_score + top_scores[inserted_node];
        if (definitelyGreaterThan(ro.inserted_max_order_scores[inserted_node], score_at_very_top, EPSILON))
        {
            // Here s(([...],a,b,i,c)) = "best inserted score"
            // cout << inserted_node <<" better at knd place "<< endl;
            return (true);
        }
    }

    return (false);
}

bool unord_hidden_gap(const RightOrder &ro,
                      const vector<double> &top_scores,
                      OrderScoring &scoring)
{
    size_t p = ro.order.size();
    size_t n = ro.n_nodes;

    // Checking insertion of each of the remaining nodes
    for (size_t node_index = 0; node_index < p - n; node_index++)
    {
        int inserted_node = ro.order[node_index];
        // cout <<  "Insert "<< inserted_node<< endl;
        //  Total score when the new node is at the top
        double score_at_very_top = ro.order_score + top_scores[inserted_node];
        // Special case if the best position is the second top (i.e ro's top), since then we have to check if the
        // top score is invariant under swapping the top 2 nodes and if so, prune if they are not ordered.
        if (ro.best_insert_pos[inserted_node] == ro.front_ind())
        {
            // Here, best insert score is at the second top ([...],a,i,b,c)
            // Need to compare S(i,[...],a,b,c) == S([...],a,i,b,c)
            if (approximatelyEqual(ro.inserted_max_order_scores[inserted_node], score_at_very_top, EPSILON))
            {
                // We also have to check if the top is ordered.
                // If not, we prune, i.e. return True.
                // Evaluate S([...],i,a,b,c) == S([...],a,i,b,c) and i > a
                if (inserted_node > ro.front())
                {
                    // cout <<  " Equal unordered top "<< endl;
                    // cout << ro.inserted_max_order_scores[inserted_node]<< " = "<< score_at_very_top<< endl;
                    return (true);
                }
            }
        }
    }

    return (false);
}

RightOrder init_right_order(const vector<double> &top_scores, OrderScoring &scoring)
{
    size_t p = scoring.numparents.size();
    // a range from 0 to p-1
    // The makes it easier to keep track of which nodes are used.
    vector<int> order(p);
    iota(order.begin(), order.end(), 0);
    // score nodes
    vector<double> node_scores(p, 0); // The scores of the nodes in the sub order.
    //  score order
    double order_score = 0;
    // Why is not order and node_scores just set in the constructor? 
    RightOrder ro(order, order_score, node_scores, 0); // copy node_scores and order O(p)
    // unrestrained top score sum, for higher upper bound.
    ro.hidden_top_score_sum = accumulate(top_scores.begin(), top_scores.end(), 0.0); // O(p)
    // cout << "Top scores: ";
    // for (auto &score : top_scores)
    //{
    //    cout << score << " ";
    //}
    // cout << endl;
    // cout << "Hidden top score sum: " << ro.hidden_top_score_sum << endl;

    // This might not be needed.
    for (size_t i = 0; i < p; i++)
    {
        size_t inserted_node = ro.order[i];
        move_element(ro.order, i, p - 1);
        ro.new_top_scores[inserted_node] = scoring.score_pos(ro.order, p - 1);
        ro.inserted_max_order_scores[inserted_node] = ro.new_top_scores[inserted_node];
        move_element(ro.order, p - 1, i); // Move back
        ro.best_insert_pos[inserted_node] = p - 1;
    }

    return (ro);
}

/**
 * @brief Adds a node in the front of the right order without updating the insertion scores.
 */
RightOrder add_node_in_front(const RightOrder &ro_prev, size_t index_of_el_to_insert, const vector<double> &top_scores, OrderScoring &scoring)
{
    RightOrder ro(ro_prev); // O(p)
    ro.n_nodes = ro_prev.n_nodes + 1;
    size_t n = ro.n_nodes;
    size_t p = ro.order.size();
    size_t node = ro_prev.order[index_of_el_to_insert];

    move_element(ro.order, index_of_el_to_insert, p - n);     // put node in the front: ([...],node,a,b,c)
    double orderscore_new = ro_prev.new_top_scores[node];     // O(1) this is for the whole ro with node added in the front/top.
    double node_score = orderscore_new - ro_prev.order_score; // O(1)

    ro.node_scores[node] = node_score;
    ro.order_score = orderscore_new;

    // updated the hidden unrestricted sum
    ro.hidden_top_score_sum = ro_prev.hidden_top_score_sum - top_scores[node];

    // Should also update the inserted node order scores.
    // 1. Remove node from the list, since this is now part of the visible nodes in the order.
    // However, you should run update_insertion_scores before using these.
    ro.inserted_max_order_scores[node] = 0;
    ro.best_insert_pos[node] = -10;
    ro.new_top_scores[node] = 0;
    return (ro);
}

/**
 * Note that the ro should have the insertions scores etc from its parent when entering this function.
 * This focuses on the lastly added node (front) in the order.
 */
void update_insertion_scores(RightOrder &ro, OrderScoring &scoring)
{
    size_t n = ro.n_nodes;
    size_t p = ro.order.size();

    // If the last one
    if (n == p)
    {
        return;
    }

    double order_score_bkp = ro.order_score; // I added this
    size_t node = ro.front();
    double node_score = ro.node_scores[node];

    // The focus is on the ordering, not on the node.
    // We need to update the clculation of the max order scores that would be if each of the
    // hidden nodes where inserted. For this we can just add the score of the new
    // (since this was calculated) top node when it is on top. The hidden nodes has
    // to be pushed in between first.

    size_t top_ind = p - n;           // ro.front_ind();
    size_t new_top_ind = top_ind - 1; // ro.front_ind() - 1;

    if (n == 1)
    {
        for (size_t i = 0; i <= p - 2; i++)
        {
            size_t inserted_node = ro.order[i];
            make_visible(i, new_top_ind, ro, scoring);         // ([..i..],a) -> ([...],i,a). This changes ro.order_score.
            ro.new_top_scores[inserted_node] = ro.order_score; // set the new top score i.e. s([...],i,a)
            swap_nodes(new_top_ind, top_ind, ro, scoring);     // ([...],i,a) -> ([...],a,i). This changes ro.order_score.
            ro.inserted_max_order_scores[inserted_node] = ro.order_score;

            move_element(ro.order, top_ind, i);  // Move back
            ro.order_score = order_score_bkp;    // Restore order score
            ro.node_scores[inserted_node] = 0.0; // restore node score
            ro.node_scores[node] = node_score;
            ro.n_nodes--;
        }
    }

    if (n > 1)
    {

        // 1. Update the insertion scores for node up to position n-2?
        for (size_t i = 0; i < p - n; i++)
        {
            // Note that inserted_node must be put as child of node when calculating score of node.
            // ([...],node,inserted_node,a,b,c)
            // then for the inserted_max_order_scores[inserted_node] we just have to add th score of node.
            // Its like we can adjust the previously calculated values for when no de is added. Then,
            // below (3), we calculate the missing part, where node is on top?
            size_t inserted_node = ro.order[i];
            move_element(ro.order, i, ro.front_ind());                                                      // ([..,inserted_node,..],node,a,b,c) -> ([...],node,inserted_node,a,b,c)
            ro.inserted_max_order_scores[inserted_node] += scoring.score_pos(ro.order, ro.front_ind() - 1); // O(p). This can be O(1) if inserted_node is not a plus1 parent of node.
            move_element(ro.order, ro.front_ind(), i);
        }

        //  3. Go through all other nodes and compare max inserted score to inserted score
        //     between node and a in ([...],node,a,b,c).
        for (size_t i = 0; i <= new_top_ind; i++)
        {
            size_t inserted_node = ro.order[i];
            make_visible(i, new_top_ind, ro, scoring);                                     // ([..i..],a,b,c) -> ([...],i,a,b,c). This changes ro.order_score.
            ro.new_top_scores[inserted_node] = ro.order_score;                             // set the new top score i.e. s([...],i,a,b,c)
            swap_nodes(new_top_ind, top_ind, ro, scoring);                                 // ([...],i,a,b,c) -> ([...],a,i,b,c). This changes ro.order_score.
            double inserted_max_order_score = ro.inserted_max_order_scores[inserted_node]; // this should contain max score of the other positions.

            // Note that new_top_ind is not part of the insertion indices.
            if (approximatelyEqual(ro.order_score, inserted_max_order_score, EPSILON) ||
                definitelyGreaterThan(ro.order_score, inserted_max_order_score, EPSILON))
            {
                ro.inserted_max_order_scores[inserted_node] = ro.order_score;
                ro.best_insert_pos[inserted_node] = top_ind;
            }

            move_element(ro.order, top_ind, i);  // Move back
            ro.order_score = order_score_bkp;    // Restore order score
            ro.node_scores[inserted_node] = 0.0; // restore node score
            ro.node_scores[node] = node_score;
            ro.n_nodes--;
        }
    }
}

/// @brief Binary array version of sub order.
/// @param order
/// @param n
/// @param right_type
/// @return
vector<bool> order_to_boolvec(const vector<int> &order, size_t n, bool right_type)
{
    vector<bool> boolvec(order.size(), false);
    size_t p = order.size();

    if (right_type)
    {
        for (size_t i = p - n; i < p; i++)
        {
            boolvec[order[i]] = true;
        }
    }
    else // left order
    {
        for (size_t i = 0; i < n; i++)
        {
            boolvec[order[i]] = true;
        }
    }
    return (boolvec);
}

vector<RightOrder> prune_equal_sets(vector<RightOrder> &right_orders,
                                    bool right_type)
{
    vector<vector<bool>> boolmat;
    vector<double> order_scores;

    // cout << "Creating boolmatrix" << endl;
    for (const RightOrder &ro : right_orders)
    {
        vector<bool> boolvec = order_to_boolvec(ro.order, ro.n_nodes, right_type);
        boolmat.push_back(move(boolvec));
        order_scores.push_back(ro.order_score);
    }

    // cout << "Get indices of unique maximal scoring sets" << endl;
    vector<int> pruned_inds = unique_sets(boolmat, order_scores, EPSILON);
    vector<double> pruned_scores;
    vector<RightOrder> kept_ros;

    // list<RightOrder>::iterator i = right_orders.begin();
    // int ind = 0;

    // for(int j = 0; j < right_orders.size(); j++){

    //     ind = pruned_inds[j];
    //     advance(i, ind);

    // }

    // while (i != items.end()){
    //     if(ind == 1){
    //         i = items.erase(i);
    //     } else {
    //         ++i;
    //     }

    // }

    for (const auto &ind : pruned_inds)
    {
        kept_ros.push_back(right_orders[ind]);
    }

    return (kept_ros);
}

/**
 * @brief TODO: Check that the visible nodes does not need any hidden node.
 * i.e. that S((h,[H],a,b,c)) < S(([H],a,h,b,c))
 *
 * @param ro RightOrder
 * @param top_scores scores of the nodes when put at the top.
 * @param scoring OrderScoring
 * @return vector<int> indices of the nodes that are not independent of the hidden nodes TODO: .
 */
vector<int> no_right_gaps(RightOrder &ro,
                          vector<double> &top_scores,
                          OrderScoring &scoring)

{
    // get all vectors with indendent front.
    int max_indep_node = 0;
    bool has_indep_node = false;
    size_t max_indep_ind = 0;
    size_t p = ro.order.size();
    size_t n = ro.n_nodes + 1;

    vector<int> indices_to_consider;

    for (size_t node_index = 0; node_index <= p - n; node_index++) // O(p)
    {
        int new_node = ro.order[node_index];

        if (approximatelyEqual(top_scores[new_node] + ro.order_score, ro.new_top_scores[new_node], EPSILON))
        {
            // new node is independent of hidden nodes.
            has_indep_node = true;
            if (new_node > max_indep_node)
            {
                max_indep_node = new_node;
                max_indep_ind = node_index;
            }
        }
    }

    // Add the nodes with higher number that the maximal that is indep of hidden.
    if (has_indep_node)
    {
        for (size_t node_index = 0; node_index <= p - n; node_index++)
        {

            int new_node = ro.order[node_index];
            if (new_node >= max_indep_node)
            {
                indices_to_consider.push_back(node_index);
            }
        }
    }
    else // No node is indep of hidden, so add all.
    {
        for (size_t node_index = 0; node_index <= p - n; node_index++)
        {
            indices_to_consider.push_back(node_index);
        }
    }

    // cout << "has indep of hodden node " << has_indep_node << endl;

    return (indices_to_consider);
}

tuple<vector<int>, double, vector<double>, size_t, size_t> opruner_right(OrderScoring &scoring, vector<RightOrder> initial_right_orders)
{
    size_t p = scoring.numparents.size();
    // cout << "Starting optimization" << endl;
    vector<RightOrder> right_orders;
    // vector<RightOrder> right_orders_prev; // this should contain the initial sub order if that exists.
    vector<RightOrder> right_orders_prev = initial_right_orders;
    //vector<vector<double>> M = get_unrestr_mat(p, scoring);
    //vector<vector<double>> H = get_hard_restr_mat(p, scoring);
    vector<double> top_scores = get_unrestricted_vec(p, scoring);
    //vector<double> bottom_scores = all_restr(p, scoring);

    size_t orders1 = 0;
    size_t orders2 = 0;
    size_t orders3 = 0;
    size_t max_n_particles = 0;
    size_t tot_n_particles = 0;
    /**
     * Start build and prune
     * n_nodes is the number of nodes after one is added to the order.
     */
    size_t nodes_in_initial_order = 0;
    if (right_orders_prev.size() == 1)
    {
        nodes_in_initial_order = right_orders_prev[0].n_nodes;
        // cout << right_orders_prev[0] << endl;
        // PrintVector(right_orders_prev[0].order);
    }
    if (right_orders_prev.size() > 1)
    {
        cout << "too many initial orders" << endl;
        exit(1); // This should not happen. should be 1 or 0 elements in the vector.
    }

    // cout << "nodes_in_initial_order " << nodes_in_initial_order << endl;
    //  print p
    // cout << "p = " << p << endl;

    for (size_t n_nodes = nodes_in_initial_order + 1; n_nodes <= p; n_nodes++) // we should check how long the init order is here.
    {
        // cout << "n = " << n_nodes << endl;
        if (n_nodes == 1)
        {
            for (size_t node_index = 0; node_index <= p - n_nodes; node_index++)
            {
                RightOrder ro = init_right_order(top_scores, scoring);
                ro = add_node_in_front(ro, node_index, top_scores, scoring);
                update_insertion_scores(ro, scoring);
                if (!optimal_front(ro, scoring))
                    continue;
                ++orders1;
                if (has_hidden_gap(ro, top_scores, scoring))
                    continue;
                if (unord_hidden_gap(ro, top_scores, scoring))
                    continue;
                right_orders.push_back(move(ro));
            }
            orders2 = right_orders.size();
            orders3 = orders2;
        }
        else
        {
            // Print the initial Right order
            //cout << endl;
            //cout << "Rightorders_prev before add in front and prune indep front "<< endl;
            //for (auto& rr : right_orders_prev) cout << rr << endl;
            // Loop over all the sub orders from the previous round.            
            for (RightOrder &prev_order : right_orders_prev) // O( (p CR n-1) )
            {
                // cout << "prev_order " << prev_order << endl;
                //  If some new node is independent of the hidden, we prune all with
                //  lower numbers than the maximal one of these, using no_right_gaps.
                vector<int> inds_to_consider = no_right_gaps(prev_order, top_scores, scoring); // O(p)
                // cout << "inds_to_consider.size() " << inds_to_consider.size() << endl;
                for (auto node_index : inds_to_consider) // O(p)
                {
                    int new_node = prev_order.order[node_index];
                    // cout << "new_node " << new_node << endl;
                    //  Check if the new node is optimal at front, is S(([hidden], n_node, visible)) S(([hidden], visible and n)).
                    //  If not, don't consider this node for addition.
                    if (!optimal_front(prev_order, new_node, scoring))
                        continue; // O(1)
                    // if ((nodes_in_initial_order > 0) && (n_nodes == nodes_in_initial_order + 1) )
                    // {
                    //     as
                    // }

                    // the top can be swapped, i.e. S(([hidden],n,m,rest-m)) = S(([hidden],m,n,rest-m))
                    // OBS! equal_and_unordered_top assumes (correctly) that new_node is optimal in the front or equally good somewhere else.
                    // so it just have to check if the best insertion position is the second top and that
                    // the score at that position is equal.
                    // If so we check the ordering of the top two nodes.
                    if (equal_and_unordered_top(prev_order, new_node, scoring))
                        continue; // O(1)

                    RightOrder ro = add_node_in_front(prev_order, node_index, top_scores, scoring); // O(p) for copying the order

                    // double upper_bound = ro.order_score + ro.hidden_top_score_sum;
                    // if (definitelyLessThan(upper_bound, reference_order.order_score, EPSILON))
                    //     continue; // O(1)

                    right_orders.push_back(move(ro));
                }
                // cout << "right_orders.size() " << right_orders.size() << endl;
            }

            // O(|right_orders_prev| * p) particles. space
            orders1 = right_orders.size();
            // cout << "# orders after add in front and prune indep front: " << orders1 << endl;
            // print the orders

            ///cout << "Rightorders after addinng node and pruning optimal_front, equal_and_unordered_top, no_right_gaps  "<< endl;
            //for (auto& rr : right_orders) cout << rr << endl;

        
            // For orders with the same nodes, keep only one having the maximal score.
            right_orders = prune_equal_sets(right_orders, true); // O(#particles * p) = O( (p CR n) * n * p) space and time.
            //cout << "Rightorders after prune equal sets "<< endl;
            //for (auto& rr : right_orders) cout << rr << endl;

            orders2 = right_orders.size();

            // update the insertion scores for all the right orders
            for (RightOrder &ro : right_orders)
                update_insertion_scores(ro, scoring); // O(p)

            // cout << "# orders after prune equal sets: " << orders2 << endl;

            // Remove any order that has gaps.
            // After equal set pruning there are O((p CR n)) particles left so
            // this is O((p CR n))
            vector<RightOrder> right_orders_tmp;
            for (RightOrder &ro : right_orders)
            {
                // cout << "Check for gaps in: " << ro << endl;
                //  This is O(p-n) = O(p)
                if (has_hidden_gap(ro, top_scores, scoring))
                    continue;
                if (unord_hidden_gap(ro, top_scores, scoring))
                    continue;

                right_orders_tmp.push_back(move(ro));
            }
            right_orders = move(right_orders_tmp);
            

            orders3 = right_orders.size();
            // cout << "# orders after has gap prune: " << orders3 << endl;
            //  tree estimate of the left side of the orders

            // cout << "right_orders.size() " << right_orders.size() << endl;
            // cout << "reference order " << reference_order << endl;
            // //prune_path(reference_order, right_orders, M, H, bottom_scores, top_scores, scoring);
            // right_orders = prune_path(reference_order, right_orders, M, H, bottom_scores, top_scores, scoring);

            // if (n_nodes == p - 2)
            // {
            //     cout << "reference_order " << reference_order << endl;
            //     break; // The reference order should be optimal.
            // }

            // cout << "kept_right_orders.size() " << right_orders.size() << endl;
            if (right_orders.size() == 0)
                break; // how would this be 0?
        }

        // cout << "orders of size " << n << endl;

        // Print some statistics //
        /* Add to number of particles sum */
        tot_n_particles += orders3;
        /* Check that scores are correct */
        if (orders3 > max_n_particles)
        {
            max_n_particles = orders3;
        }

        // This is O((p CR n))

        // cout << "# of orders: " << right_orders.size() << endl;
        auto max_ro = max_element(right_orders.begin(),
                                  right_orders.end(),
                                  [](const RightOrder &a, const RightOrder &b)
                                  { return a.order_score < b.order_score; });

        // cout << "max scoring sub order " << endl;
        // cout << *max_ro << endl;
        // cout << "score: " << max_ro->order_score << endl;
        // PrintVector(max_ro->node_scores);

        // if right_oder is not empty, tha max order should be set to the max
        // of the right orders and the reference order.
        // if (right_orders.size() > 0)
        // {
        //     if (definitelyGreaterThan(reference_order.order_score, max_ro->order_score, EPSILON))
        //     {
        //         // reference_order = *max_ro;
        //         cout << "Using refeterence order as max order" << endl;
        //         *max_ro = reference_order;
        //         // reference_order = *max_ro;
        //     }
        // }

        // check correct score
        vector<double> sc = scoring.score(max_ro->order, p - n_nodes, n_nodes); // Take only the last n elements in the vector
        // PrintVector(sc);
        double max_score_check = accumulate(sc.begin(), sc.end(), 0.0);
        // cout << "correct max score: " << max_score_check << endl;
        assert(abs(max_ro->order_score - max_score_check) < EPSILON);

        // cout << n << " & " << orders1 << " & " << orders2 << " & " << orders3 << " & " << max_ro->order_score << " \\\\" << endl;
        right_orders_prev = move(right_orders);
        // cout << "after move" << endl;
    }

    // cout << endl;
    //  assert(right_orders_prev.size() == 1); // we quite 2 steps before...
    //   It should only be one here anyways

    // for (auto& rr : right_orders_prev)
    // {
    //     cout << rr << endl;
    // }

    // cout << "reference order: " << reference_order << endl;
    // right_orders_prev.push_back(reference_order);

    auto max_ro = max_element(right_orders_prev.begin(),
                              right_orders_prev.end(),
                              [](const RightOrder &a, const RightOrder &b)
                              { return a.order_score < b.order_score; });

    // cout << "MAX order " << *max_ro << endl;

    return (make_tuple(max_ro->order, max_ro->order_score, max_ro->node_scores, max_n_particles, tot_n_particles));
    // return (make_tuple(reference_order.order, reference_order.order_score, max_n_particles, tot_n_particles));
}

// Function to print the
// index of an element
size_t getIndex(vector<int> &v, int K)
{
    auto it = find(v.begin(), v.end(), K);
    int index = it - v.begin();
    return index;
}

/**
 * opruner_right taking just initial nodes, which are automatically converted to a list with RighOrder
 * 
*/
tuple<vector<int>, double, vector<int>, double, vector<double>, size_t, size_t> opruner_right(OrderScoring &scoring, vector<int> &initial_right_order){
    vector<RightOrder> initial_right_orders = {};
    if (initial_right_order.size() > 0)
    {
        size_t p = scoring.numparents.size();    // total number of nodes
        size_t n = initial_right_order.size(); // number of nodes in the initial sub order
        vector<double> top_scores = get_unrestricted_vec(p, scoring);

        // Me: To avoid calculation, maybe we shouldnt initialize here
        // and just update in the end
        RightOrder initial_order = init_right_order(top_scores, scoring);
        size_t initial_node_index = getIndex(initial_order.order, initial_right_order[n - 1]);
        initial_order = add_node_in_front(initial_order, initial_node_index, top_scores, scoring); // Adding node p-1, i.e.:  <[...], p-1>
        update_insertion_scores(initial_order, scoring);
        // As a reference order, add all nodes in order a.t.m.: <1,2,...,p-1>

        for (int i = n - 2; i >= 0; i--)
        {
            // reference_order = add_node_in_front(reference_order, r_initial_right_order[i], top_scores, scoring);
            //  Find the initial node amongs the hidden nodes.
            initial_node_index = getIndex(initial_order.order, initial_right_order[i]);
            initial_order = add_node_in_front(initial_order, initial_node_index, top_scores, scoring);
            update_insertion_scores(initial_order, scoring);
        }

        // Manipulating the insertion scores, so that the initial order doesnt get pruned.
        for (size_t i = 0; i < initial_order.size_hidden(); i++)
        {
            size_t hidden_node = initial_order.order[i];

            //  Make the best inserted score worse than adding it to the front
            initial_order.inserted_max_order_scores[hidden_node] = initial_order.new_top_scores[hidden_node] - 100000;
        }

        initial_right_orders.push_back(initial_order);
    }

    // Add the initial right orders

    const auto &[order, log_score, node_scores, max_n_particles, tot_n_particles] = opruner_right(scoring, initial_right_orders);

    // get the suborder
    vector<int> sub_order = order;
    sub_order.resize(order.size() - initial_right_order.size());
    // get the total score for the suborders
    double sub_order_log_score = 0;
    for (size_t i = 0; i < sub_order.size(); i++)
    {
        sub_order_log_score += node_scores[sub_order[i]];
    }

    return (make_tuple(order, log_score, sub_order, sub_order_log_score, node_scores, max_n_particles, tot_n_particles));

}
/**
 * Get DAG associated with order.
 */
vector<vector<bool>> order_to_dag(const vector<int> &order, const OrderScoring &scoring)
{
    size_t p = scoring.numparents.size();
    vector<vector<bool>> dag(p, vector<bool>(p, 0));
    for (size_t i = 0; i < order.size(); i++)
    {
        int node = order[i];
        vector<int> parents = scoring.get_opt_parents(order, i);
        for (size_t j = 0; j < parents.size(); j++)
        {
            dag[parents[j]][node] = true;
        }
    }
    return dag;
}


// [[Rcpp::plugins(cpp17)]]

// [[Rcpp::export]]

Rcpp::NumericMatrix r_order_to_dag(Rcpp::List cpp_friendly_scores, Rcpp::NumericVector order)
{
    OrderScoring scoring = get_score(cpp_friendly_scores);

    // convert order to vector
    vector<int> order_vec = Rcpp::as<vector<int>>(order);
    // decrement order by 1
    for (size_t i = 0; i < order_vec.size(); i++)
    {
        order_vec[i] = order_vec[i] - 1;
    }
    // get DAG from order
    size_t p = order.size();
    vector<vector<bool>> dag = order_to_dag(order_vec, scoring);
    // convert dag to Rcpp matrix
    Rcpp::NumericMatrix dag_mat(p, p);
    for (size_t i = 0; i < p; i++)
    {
        for (size_t j = 0; j < dag[i].size(); j++)
        {
            dag_mat(i, j) = dag[i][j];
        }
    }

    return (dag_mat);
}




// [[Rcpp::plugins(cpp17)]]

// [[Rcpp::export]]

Rcpp::List r_opruner_right(Rcpp::List ret, Rcpp::List r_initial_right_order)
{
    OrderScoring scoring = get_score(ret);

    // cout << "initial right order: " << endl;

    // decrease r_initial_right_order by 1

    // if r_initial_right_orders is not empty and contains one list, then use that as the initial right order.
    // otherwise, use the empty list.

    vector<RightOrder> initial_right_orders = {};

    if (r_initial_right_order.size() > 0)
    {
        for (int i = 0; i < r_initial_right_order.size(); i++)
        {
            r_initial_right_order[i] = r_initial_right_order[i] - 1;
        }

        size_t p = scoring.numparents.size();    // total number of nodes
        size_t n = r_initial_right_order.size(); // number of nodes in the initial sub order
        vector<double> top_scores = get_unrestricted_vec(p, scoring);

        // Me: To avoid calculation, maybe we shouldnt initialize here
        // and just update in the end
        RightOrder initial_order = init_right_order(top_scores, scoring);
        size_t initial_node_index = getIndex(initial_order.order, r_initial_right_order[n - 1]);
        initial_order = add_node_in_front(initial_order, initial_node_index, top_scores, scoring); // Adding node p-1, i.e.:  <[...], p-1>
        update_insertion_scores(initial_order, scoring);
        // As a reference order, add all nodes in order a.t.m.: <1,2,...,p-1>

        for (int i = n - 2; i >= 0; i--)
        {
            // reference_order = add_node_in_front(reference_order, r_initial_right_order[i], top_scores, scoring);
            //  Find the initial node amongs the hidden nodes.
            initial_node_index = getIndex(initial_order.order, r_initial_right_order[i]);
            initial_order = add_node_in_front(initial_order, initial_node_index, top_scores, scoring);
            update_insertion_scores(initial_order, scoring);
        }

        // Manipution the insertion scores, so that the initial order doesnt get pruned.
        for (size_t i = 0; i < initial_order.size_hidden(); i++)
        {
            size_t hidden_node = initial_order.order[i];

            //  Make the best inserted score worse than adding it to the front
            initial_order.inserted_max_order_scores[hidden_node] = initial_order.new_top_scores[hidden_node] - 100000;
        }

        initial_right_orders.push_back(initial_order);
    }

    // Add the initial right orders

    const auto &[order, log_score, node_scores, max_n_particles, tot_n_particles] = opruner_right(scoring, initial_right_orders);

    // Maybe extract the sub order and corresponding score.

    Rcpp::List L = Rcpp::List::create(Rcpp::Named("order") = order,
                                      Rcpp::Named("log_score") = log_score,
                                      Rcpp::Named("node_scores") = node_scores,
                                      Rcpp::Named("max_n_particles") = max_n_particles,
                                      Rcpp::Named("tot_n_particles") = tot_n_particles);

    return (L);
}