#include "common.h" #include <iostream> #include <fstream> #include <sstream> #include <stdexcept> #include <cstdio> #include <cstdlib> #include <filesystem> #include <chrono> #include <regex> namespace fs = std::filesystem; static void log_info(const std::string &msg) { std::cout << "[INFO] " << msg << "\n"; } static void log_error(const std::string &msg) { std::cerr << "[ERROR] " << msg << "\n"; } std::string parse_version(const fs::path &changelog_path) { if (!fs::exists(changelog_path)) { throw std::runtime_error("Changelog not found: " + changelog_path.string()); } std::ifstream f(changelog_path); if (!f) throw std::runtime_error("Unable to open changelog"); std::string first_line; std::getline(f, first_line); f.close(); size_t start = first_line.find('('); size_t end = first_line.find(')'); if (start == std::string::npos || end == std::string::npos) { throw std::runtime_error("Invalid changelog format"); } std::string version_match = first_line.substr(start+1, end - (start+1)); std::string epoch; std::string upstream_version = version_match; if (auto pos = version_match.find(':'); pos != std::string::npos) { epoch = version_match.substr(0, pos); upstream_version = version_match.substr(pos+1); } if (auto pos = upstream_version.find('-'); pos != std::string::npos) { upstream_version = upstream_version.substr(0, pos); } std::regex git_regex("(\\+git[0-9]+)?(~[a-z]+)?$"); upstream_version = std::regex_replace(upstream_version, git_regex, ""); auto t = std::time(nullptr); std::tm tm = *std::gmtime(&t); char buf[32]; std::strftime(buf, sizeof(buf), "%Y%m%d%H%M", &tm); std::string current_date = buf; std::string version; if (!epoch.empty()) { version = epoch + ":" + upstream_version + "+git" + current_date; } else { version = upstream_version + "+git" + current_date; } return version; } void run_command(const std::vector<std::string> &cmd, const std::optional<fs::path> &cwd, bool show_output) { std::string full_cmd; for (const auto &c : cmd) { full_cmd += c + " "; } if (cwd) { full_cmd = "cd " + cwd->string() + " && " + full_cmd; } log_info("Executing: " + full_cmd); int ret = std::system(full_cmd.c_str()); if (ret != 0) { log_error("Command failed: " + full_cmd); throw std::runtime_error("Command failed"); } if (show_output) { std::cout << "[INFO] Command succeeded: " + full_cmd << "\n"; } } void clean_old_logs(const fs::path &log_dir, int max_age_seconds) { auto now = std::chrono::system_clock::now(); for (auto &entry : fs::directory_iterator(log_dir)) { if (fs::is_regular_file(entry)) { auto ftime = fs::last_write_time(entry); auto sctp = decltype(ftime)::clock::to_sys(ftime); auto age = std::chrono::duration_cast<std::chrono::seconds>(now - sctp).count(); if (age > max_age_seconds) { fs::remove(entry); } } } } void create_tarball(const std::string &name, const fs::path &source_dir, const std::vector<std::string> &exclusions) { std::string tar_filename = name + "_MAIN.orig.tar.gz"; std::cout << "[INFO] Creating tarball: " << tar_filename << "\n"; std::vector<std::string> cmd; cmd.push_back("tar"); for (auto &ex : exclusions) { cmd.push_back("--exclude=" + ex); } cmd.push_back("--exclude=.git/"); cmd.push_back("-czf"); cmd.push_back(tar_filename); cmd.push_back(fs::path(source_dir).filename().string()); run_command(cmd, source_dir.parent_path()); std::cout << "[INFO] Tarball created and compressed: " << tar_filename << "\n"; }