#include "config.h"

#include <algorithm>
#include <cctype>
#include <fstream>
#include <stdexcept>
#include <string>
#include <unordered_map>

static std::string trim(const std::string &s) {
    std::size_t start = s.find_first_not_of(" \t\r\n");
    if (start == std::string::npos) {
        return "";
    }
    std::size_t end = s.find_last_not_of(" \t\r\n");
    return s.substr(start, end - start + 1);
}

static std::string to_lower(std::string s) {
    for (char &c : s) {
        c = static_cast<char>(std::tolower(static_cast<unsigned char>(c)));
    }
    return s;
}

static std::unordered_map<std::string, std::string> load_kv(const std::string &path) {
    std::ifstream in(path);
    if (!in) {
        throw std::runtime_error("Failed to open config: " + path);
    }
    std::unordered_map<std::string, std::string> kv;
    std::string line;
    while (std::getline(in, line)) {
        auto hash = line.find('#');
        if (hash != std::string::npos) {
            line = line.substr(0, hash);
        }
        line = trim(line);
        if (line.empty()) {
            continue;
        }
        auto eq = line.find('=');
        if (eq == std::string::npos) {
            continue;
        }
        std::string key = trim(line.substr(0, eq));
        std::string value = trim(line.substr(eq + 1));
        if (!key.empty()) {
            kv[key] = value;
        }
    }
    return kv;
}

static int to_int(const std::unordered_map<std::string, std::string> &kv,
                  const std::string &key, int def) {
    auto it = kv.find(key);
    if (it == kv.end()) {
        return def;
    }
    return std::stoi(it->second);
}

static double to_double(const std::unordered_map<std::string, std::string> &kv,
                        const std::string &key, double def) {
    auto it = kv.find(key);
    if (it == kv.end()) {
        return def;
    }
    return std::stod(it->second);
}

static std::string to_string(const std::unordered_map<std::string, std::string> &kv,
                             const std::string &key, const std::string &def) {
    auto it = kv.find(key);
    if (it == kv.end()) {
        return def;
    }
    return it->second;
}

Config parse_config(const std::string &path) {
    Config cfg;
    auto kv = load_kv(path);
    cfg.data_file = to_string(kv, "data_file", "");
    cfg.instance_index = to_int(kv, "instance_index", cfg.instance_index);
    cfg.run_all = to_int(kv, "run_all", cfg.run_all ? 1 : 0) != 0;
    cfg.num_nodes = to_int(kv, "num_nodes", cfg.num_nodes);

    cfg.ants = to_int(kv, "ants", cfg.ants);
    cfg.iterations = to_int(kv, "iterations", cfg.iterations);
    cfg.candidate_list_size = to_int(kv, "candidate_list_size", cfg.candidate_list_size);
    cfg.nn_ants = to_int(kv, "nn_ants", 0);
    cfg.nn_ls = to_int(kv, "nn_ls", 0);
    cfg.threads = to_int(kv, "threads", cfg.threads);
    cfg.alpha = to_double(kv, "alpha", cfg.alpha);
    cfg.beta = to_double(kv, "beta", cfg.beta);
    cfg.rho = to_double(kv, "rho", cfg.rho);
    cfg.q = to_double(kv, "q", cfg.q);
    cfg.q0 = to_double(kv, "q0", cfg.q0);
    cfg.p_best = to_double(kv, "p_best", cfg.p_best);
    cfg.tau_min = to_double(kv, "tau_min", cfg.tau_min);
    cfg.tau_max = to_double(kv, "tau_max", cfg.tau_max);
    cfg.lambda = to_double(kv, "lambda", cfg.lambda);
    cfg.branch_fac = to_double(kv, "branch_fac", cfg.branch_fac);
    cfg.u_gb = to_int(kv, "u_gb", cfg.u_gb);
    cfg.dlb_flag = to_int(kv, "dlb_flag", cfg.dlb_flag);
    cfg.max_time_seconds = to_int(kv, "max_time_seconds", cfg.max_time_seconds);
    cfg.seed = to_int(kv, "seed", cfg.seed);

    cfg.local_search = to_string(kv, "local_search", cfg.local_search);
    cfg.max_2opt_passes = to_int(kv, "max_2opt_passes", cfg.max_2opt_passes);
    cfg.max_3opt_passes = to_int(kv, "max_3opt_passes", cfg.max_3opt_passes);
    cfg.lk_passes = to_int(kv, "lk_passes", cfg.lk_passes);
    cfg.lk_depth = to_int(kv, "lk_depth", cfg.lk_depth);

    cfg.heatmap_mode = to_string(kv, "heatmap_mode", cfg.heatmap_mode);
    cfg.heatmap_file = to_string(kv, "heatmap_file", cfg.heatmap_file);
    cfg.heatmap_root = to_string(kv, "heatmap_root", cfg.heatmap_root);
    cfg.heatmap_dir = to_string(kv, "heatmap_dir", cfg.heatmap_dir);
    cfg.heatmap_type = to_string(kv, "heatmap_type", cfg.heatmap_type);
    cfg.heatmap_threshold = to_double(kv, "heatmap_threshold", cfg.heatmap_threshold);
    cfg.confidence_gamma = to_double(kv, "confidence_gamma", cfg.confidence_gamma);

    cfg.distance_mode = to_string(kv, "distance_mode", cfg.distance_mode);
    cfg.distance_precompute_limit = to_int(kv, "distance_precompute_limit", cfg.distance_precompute_limit);

    cfg.output_dir = to_string(kv, "output_dir", cfg.output_dir);

    if (cfg.data_file.empty()) {
        throw std::runtime_error("data_file is required in config");
    }
    if (cfg.nn_ants <= 0) {
        cfg.nn_ants = cfg.candidate_list_size;
    }
    if (cfg.nn_ls <= 0) {
        cfg.nn_ls = cfg.candidate_list_size;
    }
    cfg.q0 = std::clamp(cfg.q0, 0.0, 1.0);
    cfg.local_search = to_lower(cfg.local_search);
    cfg.heatmap_mode = to_lower(cfg.heatmap_mode);
    cfg.distance_mode = to_lower(cfg.distance_mode);
    if (cfg.heatmap_dir.empty() && !cfg.heatmap_type.empty()) {
        cfg.heatmap_dir = cfg.heatmap_type;
    }
    return cfg;
}
