// Copyright (C) 2024-2025 Simon Quigley // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . #include "utilities.h" #include "common.h" #include #include #include #include #include #include #include #include #include #include #include #include // for std::format in C++20/23 namespace fs = std::filesystem; // Define a semaphore with a maximum of 10 concurrent jobs static std::counting_semaphore<10> sem(10); // Job queue and synchronization primitives static std::queue> job_queue; static std::mutex queue_mutex; static std::atomic daemon_running{false}; // Function to read the entire content of a file into a string std::string read_file(const fs::path& file_path) { std::ifstream in_file(file_path, std::ios::binary); if (in_file) { return std::string((std::istreambuf_iterator(in_file)), std::istreambuf_iterator()); } return ""; } // Function to write a string into a file void write_file(const fs::path& file_path, const std::string& content) { std::ofstream out_file(file_path, std::ios::binary); if (out_file) { out_file << content; } } // Function to perform in-place regex replace on a file void regex_replace_in_file(const fs::path& file_path, const std::string& pattern, const std::string& replacement) { std::string content = read_file(file_path); content = std::regex_replace(content, std::regex(pattern), replacement); write_file(file_path, content); } // Function to decompress gzipped files std::string decompress_gzip(const fs::path& file_path) { gzFile infile = gzopen(file_path.c_str(), "rb"); if (!infile) return ""; std::string decompressed_data; char buffer[8192]; int num_read = 0; while ((num_read = gzread(infile, buffer, sizeof(buffer))) > 0) { decompressed_data.append(buffer, num_read); } gzclose(infile); return decompressed_data; } // Helper function for libcurl write callback size_t write_data(void* ptr, size_t size, size_t nmemb, void* stream) { FILE* out = static_cast(stream); return fwrite(ptr, size, nmemb, out); } // Function to download a file with timestamping using libcurl void download_file_with_timestamping(const std::string& url, const fs::path& output_path, const fs::path& log_file_path, std::mutex& log_mutex) { CURL* curl; CURLcode res; FILE* fp; curl = curl_easy_init(); if (curl) { fs::path temp_file_path = output_path.string() + ".tmp"; fp = fopen(temp_file_path.c_str(), "wb"); if (!fp) { std::cerr << "Failed to open file: " << temp_file_path << std::endl; curl_easy_cleanup(curl); return; } // Set curl options for downloading the file curl_easy_setopt(curl, CURLOPT_URL, url.c_str()); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_data); curl_easy_setopt(curl, CURLOPT_WRITEDATA, fp); // Timestamping: set If-Modified-Since header struct stat file_info; if (stat(output_path.c_str(), &file_info) == 0) { // Set the time condition to If-Modified-Since curl_easy_setopt(curl, CURLOPT_TIMECONDITION, CURL_TIMECOND_IFMODSINCE); curl_easy_setopt(curl, CURLOPT_TIMEVALUE, file_info.st_mtime); } // Perform the file download res = curl_easy_perform(curl); // Get the HTTP response code long response_code = 0; curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &response_code); fclose(fp); curl_easy_cleanup(curl); // Log the result and handle the downloaded file { std::lock_guard lock(log_mutex); std::ofstream log_file(log_file_path, std::ios::app); if (res == CURLE_OK && (response_code == 200 || response_code == 201)) { fs::rename(temp_file_path, output_path); log_file << "Downloaded: " << url << std::endl; } else if (response_code == 304) { fs::remove(temp_file_path); log_file << "Not Modified: " << url << std::endl; } else { fs::remove(temp_file_path); log_file << "Failed to download: " << url << std::endl; } } } else { std::cerr << "Failed to initialize CURL." << std::endl; } } std::filesystem::path create_temp_directory() { auto temp_dir = std::filesystem::temp_directory_path() / generate_random_string(32); std::filesystem::create_directory(temp_dir); return temp_dir; } // Function to copy a directory recursively void copy_directory(const fs::path& source, const fs::path& destination) { if (!std::filesystem::exists(source) || !std::filesystem::is_directory(source)) { throw std::runtime_error("Source directory does not exist or is not a directory: " + source.string()); } // Create the destination directory std::filesystem::create_directories(destination); // Copy files and directories recursively for (const auto& entry : std::filesystem::recursive_directory_iterator(source)) { auto relative_path = std::filesystem::relative(entry.path(), source); auto target_path = destination / relative_path; try { if (std::filesystem::is_directory(entry)) { std::filesystem::create_directory(target_path); } else if (std::filesystem::is_regular_file(entry)) { std::filesystem::copy(entry, target_path, std::filesystem::copy_options::overwrite_existing); } } catch (...) { continue; } } } // Function to generate a random string of given length std::string generate_random_string(size_t length) { const std::string chars = "abcdefghijklmnopqrstuvwxyz" "ABCDEFGHIJKLMNOPQRSTUVWXYZ" "0123456789"; thread_local std::mt19937 rg{std::random_device{}()}; thread_local std::uniform_int_distribution<> pick(0, chars.size() - 1); std::string s; s.reserve(length); while (length--) s += chars[pick(rg)]; return s; } // Function to get current UTC time formatted as per the given format string std::string get_current_utc_time(const std::string& format) { auto now = std::chrono::system_clock::now(); std::time_t now_time = std::chrono::system_clock::to_time_t(now); std::tm tm_utc; gmtime_r(&now_time, &tm_utc); char buf[64]; // Ensure sufficient buffer size for different formats std::strftime(buf, sizeof(buf), format.c_str(), &tm_utc); return std::string(buf); } // Function to convert filesystem time to time_t std::time_t to_time_t(const fs::file_time_type& ftime) { using namespace std::chrono; // Convert to system_clock time_point auto sctp = time_point_cast( ftime - fs::file_time_type::clock::now() + system_clock::now() ); return system_clock::to_time_t(sctp); } std::vector split_string(const std::string& input, const std::string& delimiter) { std::vector result; size_t start = 0; size_t end = 0; while ((end = input.find(delimiter, start)) != std::string::npos) { result.emplace_back(input.substr(start, end - start)); start = end + delimiter.length(); } // Add the remaining part of the string result.emplace_back(input.substr(start)); return result; } std::string remove_suffix(const std::string& input, const std::string& suffix) { if (input.size() >= suffix.size() && input.compare(input.size() - suffix.size(), suffix.size(), suffix) == 0) { return input.substr(0, input.size() - suffix.size()); } return input; // Return the original string if the suffix doesn't exist } // Utility which basically does the following: // "noble" (std::string) -> 2504 (int) // The bool represents whether this codename is the development release std::pair get_version_from_codename(const std::string& codename) { std::ifstream file("/usr/share/distro-info/ubuntu.csv"); if (!file.is_open()) { throw std::runtime_error("Failed to open file."); } std::string line; // Skip the header line std::getline(file, line); std::string last_codename; int version = 0; while (std::getline(file, line)) { std::istringstream iss(line); std::string version_str, name, series; std::getline(iss, version_str, ','); std::getline(iss, name, ','); std::getline(iss, series, ','); if (series == codename) { version_str.erase(std::remove(version_str.begin(), version_str.end(), '.'), version_str.end()); version = std::stoi(version_str); } last_codename = series; } bool is_last = (codename == last_codename); if (version == 0) { throw std::runtime_error("Codename not found."); } return {version, is_last}; } void ensure_git_inited() { static std::once_flag git_init_flag; std::call_once(git_init_flag, []() { git_libgit2_init(); }); } void run_task_every(std::stop_token _stop_token, int interval_minutes, std::function task) { if (interval_minutes < 2) interval_minutes = 2; std::this_thread::sleep_for(std::chrono::minutes(interval_minutes / 2)); while (!_stop_token.stop_requested()) { task(); std::this_thread::sleep_for(std::chrono::minutes(interval_minutes)); } }