#include "paths.h"

#include <algorithm>
#include <filesystem>
#include <iomanip>
#include <regex>
#include <sstream>
#include <stdexcept>

std::string default_output_dir(const Config &cfg, int n, const std::string &root_override) {
    std::string sparsify = "none";
    if (cfg.heatmap_mode == "threshold") {
        if (!cfg.heatmap_dir.empty()) {
            sparsify = cfg.heatmap_dir;
        } else if (!cfg.heatmap_type.empty()) {
            sparsify = cfg.heatmap_type;
        } else if (!cfg.heatmap_file.empty()) {
            sparsify = "heatmap";
        } else {
            sparsify = "heatmap";
        }
    }
    std::string ls = cfg.local_search.empty() ? "none" : cfg.local_search;
    std::ostringstream gamma;
    gamma.setf(std::ios::fixed);
    gamma << std::setprecision(1) << cfg.confidence_gamma;
    std::string gamma_str = gamma.str();
    std::ostringstream thr;
    thr.setf(std::ios::fixed);
    thr << std::setprecision(4) << cfg.heatmap_threshold;
    std::string thr_str = thr.str();
    if (!thr_str.empty()) {
        for (char &c : thr_str) {
            if (c == '.') {
                c = 'p';
            }
        }
    }
    std::filesystem::path out(root_override.empty() ? "output" : root_override);
    out /= sparsify;
    out /= ("TSP" + std::to_string(n));
    out /= ("gemma_" + gamma_str);
    out /= ("thr" + thr_str);
    out /= ls;
    out /= ("seed" + std::to_string(cfg.seed));
    return out.string();
}

std::vector<std::string> list_heatmap_types(const std::string &root_dir) {
    std::filesystem::path root(root_dir.empty() ? "all_heatmap" : root_dir);
    if (!std::filesystem::exists(root)) {
        throw std::runtime_error("Heatmap root not found: " + root.string());
    }
    std::vector<std::string> types;
    for (const auto &entry : std::filesystem::directory_iterator(root)) {
        if (!entry.is_directory()) {
            continue;
        }
        auto heatmap_dir = entry.path() / "heatmap";
        if (!std::filesystem::is_directory(heatmap_dir)) {
            continue;
        }
        types.push_back(entry.path().filename().string());
    }
    std::sort(types.begin(), types.end());
    return types;
}

int infer_n_from_filename(const std::string &path) {
    std::regex re("tsp(\\d+)");
    std::smatch match;
    if (std::regex_search(path, match, re) && match.size() > 1) {
        return std::stoi(match[1].str());
    }
    return 0;
}

static std::string resolve_heatmap_path(const std::string &templ, int n, int idx) {
    std::string path = templ;
    bool has_placeholder = false;
    auto replace_all = [&](const std::string &from, const std::string &to) {
        std::size_t pos = 0;
        while ((pos = path.find(from, pos)) != std::string::npos) {
            path.replace(pos, from.size(), to);
            pos += to.size();
        }
    };
    if (path.find("{n}") != std::string::npos) {
        replace_all("{n}", std::to_string(n));
        has_placeholder = true;
    }
    if (path.find("{idx}") != std::string::npos) {
        replace_all("{idx}", std::to_string(idx));
        has_placeholder = true;
    }
    if (has_placeholder) {
        return path;
    }
    std::filesystem::path fs_path(path);
    if (std::filesystem::is_directory(fs_path)) {
        std::ostringstream name;
        name << "heatmaptsp" << n << "_" << idx << ".txt";
        return (fs_path / name.str()).string();
    }
    return path;
}

std::string build_heatmap_path(const Config &cfg, int n, int idx) {
    // Prefer heatmap_file (supports template or directory)
    if (!cfg.heatmap_file.empty()) {
        return resolve_heatmap_path(cfg.heatmap_file, n, idx);
    }
    // Otherwise use heatmap_root + heatmap_dir (folder name only)
    if (!cfg.heatmap_dir.empty()) {
        std::filesystem::path root(cfg.heatmap_root.empty() ? "all_heatmap" : cfg.heatmap_root);
        std::filesystem::path dir(cfg.heatmap_dir);
        std::ostringstream name;
        name << "heatmaptsp" << n << "_" << idx << ".txt";
        return (root / dir / "heatmap" / ("tsp" + std::to_string(n)) / name.str()).string();
    }
    return "";
}
