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

// 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));
}
}