You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
299 lines
10 KiB
299 lines
10 KiB
// Copyright (C) 2024-2025 Simon Quigley <tsimonq2@ubuntu.com>
|
|
//
|
|
// 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 <https://www.gnu.org/licenses/>.
|
|
|
|
#include "utilities.h"
|
|
#include "common.h"
|
|
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <filesystem>
|
|
#include <regex>
|
|
#include <zlib.h>
|
|
#include <curl/curl.h>
|
|
#include <sys/stat.h>
|
|
#include <sstream>
|
|
#include <random>
|
|
#include <queue>
|
|
#include <ranges>
|
|
#include <format> // 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<std::function<void()>> job_queue;
|
|
static std::mutex queue_mutex;
|
|
static std::atomic<bool> 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<char>(in_file)),
|
|
std::istreambuf_iterator<char>());
|
|
}
|
|
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<FILE*>(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<std::mutex> 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<system_clock::duration>(
|
|
ftime - fs::file_time_type::clock::now() + system_clock::now()
|
|
);
|
|
return system_clock::to_time_t(sctp);
|
|
}
|
|
|
|
std::vector<std::string> split_string(const std::string& input, const std::string& delimiter) {
|
|
std::vector<std::string> 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<int, bool> 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<void()> 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));
|
|
}
|
|
}
|