#ifndef Z3_UTIL_H
#define Z3_UTIL_H

#include <utility>

#include "z3++.h"

z3::sort make_enum_sort(const std::vector<std::string> &states, z3::context &ctx, z3::expr_vector &consts,
                        const std::string &name) {
    const char *enum_names[states.size()];
    for (size_t i = 0; i < states.size(); i++) {
        enum_names[i] = states[i].c_str();
    }
    z3::func_decl_vector enum_consts(ctx);
    z3::func_decl_vector enum_testers(ctx);
    z3::sort s = ctx.enumeration_sort(name.c_str(), states.size(), enum_names, enum_consts, enum_testers);

    for (int i = 0; i < enum_consts.size(); ++i) {
        consts.push_back(enum_consts[i]());
    }

    return s;
}

void dump_vector(z3::expr_vector vs) {
    std::cout << "[";
    for (int i = 0; i < vs.size(); i++) {
        std::cout << vs[i];
        if (i < vs.size() - 1) {
            std::cout << ", ";
        }
    }
    std::cout << "]";
    std::cout << std::endl;
}

void dump_variables(std::unordered_map<std::string, int> ks, z3::expr_vector vs, int num_nodes, int num_round,
                    z3::context &context) {
    z3::expr_vector starting(context);
    for (int i = 0; i < num_nodes; i++) {
        std::stringstream xName;
        xName << "s_n" << i;
        int index = ks[xName.str()];
        starting.push_back(vs[index]);
    }

    // std::cout << "" << std::endl;
    // std::cout << "Starting states:" << std::endl;
    dump_vector(starting);
    // std::cout << "" << std::endl;

    std::vector<z3::expr_vector> decisions;
    for (int p = 0; p < num_round; p++) {
        z3::expr_vector d(context);
        for (int n = 0; n < num_nodes; n++) {
            std::stringstream xName;
            xName << "d" << p << "n" << n;
            int index = ks[xName.str()];
            d.push_back(vs[index]);
            // std::cout << xName.str() << " = " << vs[index] << std::endl;
        }
        decisions.push_back(d);
    }

    std::vector<std::vector<z3::expr_vector>> messages;
    
    for (int p = 0; p < num_round; ++p) {
        std::vector<z3::expr_vector> msg;
        for (int n = 0; n < num_nodes; ++n) {
            z3::expr_vector m(context);
            for (int s = 0; s < num_nodes; ++s) {
                std::stringstream xName;
                xName << "m" << p << "n" << n << "s" << s;
                int index = ks[xName.str()];
                m.push_back(vs[index]);
                // std::cout << xName.str() << " = " << vs[index] << " ";
            }
            // std::cout << std::endl;
            msg.push_back(m);
        }
        messages.push_back(msg);
    }

    for (int i = 0; i < num_round; i++) {
        std::cout << "===============================" << std::endl;
        std::cout << "        phase " << i + 1 << std::endl;
        std::cout << "===============================" << std::endl;
        std::cout << "message exchange:" << std::endl;
        for (int j = 0; j < num_nodes; ++j) {
            std::cout << "node " << j << ": ";
            for (int k = 0; k < num_nodes; ++k) {
                std::cout << messages[i][j][k] << " ";
            }
            std::cout << std::endl;
        }

        std::cout << "decisions:" << std::endl;
        for (int j = 0; j < num_nodes; ++j) {
            std::cout << decisions[i][j] << " ";
        }
        std::cout << std::endl;
        // for (int j = 0; j < num_nodes; j++)
        // {
        //     std::cout << messages[i][j] << " ";
        // }
        // std::cout << std::endl;
        // std::cout << "Decisions:" << std::endl;
        // std::cout << decisions[i] << std::endl;
    }
    std::cout << "";
}

void print_model(const z3::model &m, int num_nodes, int num_round, z3::context &context) {
    z3::expr_vector vs(context);
    std::unordered_map<std::string, int> ks;

    for (unsigned i = 0; i < m.size(); i++) {
        z3::func_decl v = m[i];
        std::string name = v.name().str();
        z3::expr value = m.get_const_interp(v);

        vs.push_back(value);
        ks[name] = vs.size() - 1;
    }
    dump_variables(ks, vs, num_nodes, num_round, context);
}

#endif