// Copyright (C) 2024 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 "common.h" #include "utilities.h" #include "launchpad.h" #include "archive.h" #include "distribution.h" #include "distro_series.h" #include "person.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace fs = std::filesystem; // Global variables for logging std::mutex logMutex; std::ofstream globalLogFile; // Function to log informational messages void log_info_custom(const std::string &msg) { std::lock_guard lock(logMutex); if (globalLogFile.is_open()) { auto now = std::chrono::system_clock::now(); std::time_t now_c = std::chrono::system_clock::to_time_t(now); char timebuf[20]; std::strftime(timebuf, sizeof(timebuf), "%Y-%m-%d %H:%M:%S", std::gmtime(&now_c)); globalLogFile << timebuf << " - INFO - " << msg << "\n"; globalLogFile.flush(); } } // Function to log error messages void log_error_custom(const std::string &msg) { std::lock_guard lock(logMutex); if (globalLogFile.is_open()) { auto now = std::chrono::system_clock::now(); std::time_t now_c = std::chrono::system_clock::to_time_t(now); char timebuf[20]; std::strftime(timebuf, sizeof(timebuf), "%Y-%m-%d %H:%M:%S", std::gmtime(&now_c)); globalLogFile << timebuf << " - ERROR - " << msg << "\n"; globalLogFile.flush(); } } // Function to parse command-line arguments struct Arguments { std::string user; std::string ppa; std::optional ppa2; std::optional override_output; }; Arguments parseArguments(int argc, char* argv[]) { Arguments args; int opt; bool showHelp = false; static struct option long_options[] = { {"user", required_argument, 0, 'u'}, {"ppa", required_argument, 0, 'p'}, {"ppa2", required_argument, 0, '2'}, {"override-output", required_argument, 0, 'o'}, {"help", no_argument, 0, 'h'}, {0, 0, 0, 0} }; while ((opt = getopt_long(argc, argv, "u:p:2:o:h", long_options, nullptr)) != -1) { switch (opt) { case 'u': args.user = optarg; break; case 'p': args.ppa = optarg; break; case '2': args.ppa2 = optarg; break; case 'o': args.override_output = optarg; break; case 'h': default: std::cout << "Usage: " << argv[0] << " --user --ppa [--ppa2 ] [--override-output ]\n"; exit(0); } } if (args.user.empty() || args.ppa.empty()) { std::cerr << "Error: --user and --ppa are required arguments.\n"; std::cout << "Usage: " << argv[0] << " --user --ppa [--ppa2 ] [--override-output ]\n"; exit(1); } return args; } // Function to parse the Changes file and extract Source and Architecture struct ChangesInfo { std::string source; std::string architecture; }; std::optional parse_changes_file(const fs::path& changesPath) { if (!fs::exists(changesPath)) { log_error_custom("Changelog not found: " + changesPath.string()); return std::nullopt; } std::ifstream infile(changesPath); if (!infile.is_open()) { log_error_custom("Unable to open changelog: " + changesPath.string()); return std::nullopt; } ChangesInfo info; std::string line; while (std::getline(infile, line)) { if (line.empty()) break; // End of headers if (line.find("Source:") == 0) { info.source = line.substr(7); // Trim whitespace info.source.erase(0, info.source.find_first_not_of(" \t")); } if (line.find("Architecture:") == 0) { info.architecture = line.substr(13); // Trim whitespace info.architecture.erase(0, info.architecture.find_first_not_of(" \t")); } } infile.close(); if (info.source.empty() || info.architecture.empty()) { log_error_custom("Invalid changelog format in: " + changesPath.string()); return std::nullopt; } return info; } // Function to run lintian and capture its output std::optional run_lintian(const fs::path& changesPath) { std::vector lintianCmd = {"lintian", "-EvIL", "+pedantic", changesPath.filename().string()}; try { // Redirect stdout and stderr to capture output std::string command = "lintian -EvIL +pedantic \"" + changesPath.string() + "\""; std::array buffer; std::string result; std::unique_ptr pipe(popen(command.c_str(), "r"), pclose); if (!pipe) { log_error_custom("Failed to run lintian command."); return std::nullopt; } while (fgets(buffer.data(), buffer.size(), pipe.get()) != nullptr) { result += buffer.data(); } return result; } catch (...) { log_error_custom("Exception occurred while running lintian."); return std::nullopt; } } // Function to process a single changes file URL void process_sources(const std::string& url, const fs::path& baseOutputDir, const fs::path& lintianTmpDir) { // Generate a unique temporary directory uuid_t uuid_bytes; uuid_generate(uuid_bytes); // Correctly call with one argument char uuid_cstr[37]; // UUIDs are 36 characters plus null terminator uuid_unparse_lower(uuid_bytes, uuid_cstr); // Convert to string std::string uuid_str = std::string(uuid_cstr).substr(0, 8); // Extract first 8 characters std::string tmpdir = (baseOutputDir / ("lintian_tmp_" + uuid_str)).string(); // Create temporary directory fs::create_directories(tmpdir); // Extract the changes file name from URL std::string changes_file = url.substr(url.find_last_of('/') + 1); log_info_custom("Downloading " + changes_file + " via dget."); // Run dget -u in the temporary directory std::vector dgetCmd = {"dget", "-u", url}; try { run_command(dgetCmd, tmpdir); } catch (const std::exception& e) { log_error_custom("dget command failed for URL: " + url); fs::remove_all(tmpdir); return; } // Parse the Changes file fs::path changesPath = fs::path(tmpdir) / changes_file; auto changesInfoOpt = parse_changes_file(changesPath); if (!changesInfoOpt.has_value()) { fs::remove_all(tmpdir); return; } ChangesInfo changesInfo = changesInfoOpt.value(); // Handle Architecture field std::string arch = changesInfo.architecture; arch = std::regex_replace(arch, std::regex("all"), ""); arch = std::regex_replace(arch, std::regex("_translations"), ""); std::istringstream iss(arch); std::string arch_clean; iss >> arch_clean; if (arch_clean.empty()) { fs::remove_all(tmpdir); return; } log_info_custom("Running Lintian for " + changesInfo.source + " on " + arch_clean); // Run lintian and capture output auto lintianOutputOpt = run_lintian(changesPath); if (!lintianOutputOpt.has_value()) { fs::remove_all(tmpdir); return; } std::string lintianOutput = lintianOutputOpt.value(); // Write lintian output to lintian_tmp/source/.txt fs::path outputPath = lintianTmpDir / changesInfo.source; fs::create_directories(outputPath); fs::path archOutputFile = outputPath / (arch_clean + ".txt"); try { writeFile(archOutputFile, lintianOutput); } catch (const std::exception& e) { log_error_custom("Failed to write lintian output for " + changesInfo.source + " on " + arch_clean); } // Remove temporary directory fs::remove_all(tmpdir); } // Function to perform rsync-like copy void rsync_copy(const fs::path& source, const fs::path& destination) { try { if (!fs::exists(destination)) { fs::create_directories(destination); } for (const auto& entry : fs::recursive_directory_iterator(source)) { const auto& path = entry.path(); auto relativePath = fs::relative(path, source); fs::path destPath = destination / relativePath; if (fs::is_symlink(path)) { if (fs::exists(destPath) || fs::is_symlink(destPath)) { fs::remove(destPath); } auto target = fs::read_symlink(path); fs::create_symlink(target, destPath); } else if (fs::is_directory(path)) { fs::create_directories(destPath); } else if (fs::is_regular_file(path)) { fs::copy_file(path, destPath, fs::copy_options::overwrite_existing); } } } catch (const std::exception& e) { log_error_custom("rsync_copy failed from " + source.string() + " to " + destination.string() + ": " + e.what()); } } int main(int argc, char* argv[]) { // Parse command-line arguments Arguments args = parseArguments(argc, argv); // Set BASE_OUTPUT_DIR std::string BASE_OUTPUT_DIR = "/srv/lubuntu-ci/output/"; if (args.override_output.has_value()) { BASE_OUTPUT_DIR = args.override_output.value(); } // Set LOG_DIR fs::path LOG_DIR = fs::path(BASE_OUTPUT_DIR) / "logs" / "lintian"; fs::create_directories(LOG_DIR); // Create log file with current UTC timestamp auto now = std::chrono::system_clock::now(); std::time_t now_c = std::chrono::system_clock::to_time_t(now); char timestamp[20]; std::strftime(timestamp, sizeof(timestamp), "%Y%m%dT%H%M%S", std::gmtime(&now_c)); fs::path logFilePath = LOG_DIR / (std::string(timestamp) + ".log"); // Open global log file globalLogFile.open(logFilePath, std::ios::app); if (!globalLogFile.is_open()) { std::cerr << "Error: Unable to open log file: " << logFilePath << std::endl; return 1; } log_info_custom("Starting lintian-ppa."); // Authenticate with Launchpad log_info_custom("Logging into Launchpad..."); auto lp_opt = launchpad::login(); if (!lp_opt.has_value()) { std::cerr << "Failed to authenticate with Launchpad.\n"; return 1; } auto lp = lp_opt.value().get(); auto ubuntu_opt = lp->distributions["ubuntu"]; distribution ubuntu = ubuntu_opt.value(); auto ds_opt = ubuntu.current_series; if (!ds_opt) { std::cerr << "Failed to get current_series.\n"; return 1; } auto current_series = ds_opt; // Retrieve user and PPA auto user_opt = lp->people[args.user]; person user = user_opt.value(); auto ppa_opt = user.getPPAByName(ubuntu, args.ppa); if (!ppa_opt.has_value()) { log_error_custom("Failed to retrieve PPA: " + args.ppa); return 1; } archive ppa = ppa_opt.value(); log_info_custom("Retrieved PPA: " + args.ppa); std::optional ppa2_opt; if (args.ppa2.has_value()) { auto ppa2_found = user.getPPAByName(ubuntu, args.ppa2.value()); if (!ppa2_found.has_value()) { log_error_custom("Failed to retrieve PPA2: " + args.ppa2.value()); return 1; } ppa2_opt = ppa2_found.value(); log_info_custom("Retrieved PPA2: " + args.ppa2.value()); } // Set up lintian directories fs::path lintianDir = fs::path(BASE_OUTPUT_DIR) / "lintian"; fs::path lintianTmpDir; { std::string uuid_str; uuid_t uuid_bytes; uuid_generate(uuid_bytes); char uuid_cstr[37]; uuid_unparse(uuid_bytes, uuid_cstr); uuid_str = std::string(uuid_cstr); // Truncate UUID to first 8 characters uuid_str = uuid_str.substr(0, 8); lintianTmpDir = fs::path(BASE_OUTPUT_DIR) / ("lintian_tmp_" + uuid_str); } fs::create_directories(lintianDir); fs::create_directories(lintianTmpDir); // Initialize a vector to hold all threads std::vector threads; // Mutex for managing the published sources iterator std::mutex sourcesMutex; // Function to iterate over published sources and enqueue tasks auto main_source_iter = [&](std::vector& threadsRef) { // Path to .LAST_RUN file fs::path lastRunFile = lintianDir / ".LAST_RUN"; std::chrono::system_clock::time_point lastRunTime = std::chrono::system_clock::now() - std::chrono::hours(24*365); if (fs::exists(lastRunFile)) { std::ifstream infile(lastRunFile); if (infile.is_open()) { std::string lastRunStr; std::getline(infile, lastRunStr); infile.close(); std::tm tm = {}; std::istringstream ss(lastRunStr); ss >> std::get_time(&tm, "%Y-%m-%dT%H:%M:%S"); if (!ss.fail()) { lastRunTime = std::chrono::system_clock::from_time_t(timegm(&tm)); log_info_custom("Last run time: " + lastRunStr); } else { log_error_custom("Invalid format in .LAST_RUN file."); } } } else { log_info_custom(".LAST_RUN file does not exist. Using default last run time."); } // Update .LAST_RUN with current time { std::ofstream outfile(lastRunFile, std::ios::trunc); if (outfile.is_open()) { auto currentTime = std::chrono::system_clock::now(); std::time_t currentTime_c = std::chrono::system_clock::to_time_t(currentTime); char timebuf[20]; std::strftime(timebuf, sizeof(timebuf), "%Y-%m-%dT%H:%M:%S", std::gmtime(¤tTime_c)); outfile << timebuf; outfile.close(); log_info_custom("Updated .LAST_RUN with current time: " + std::string(timebuf)); } else { log_error_custom("Failed to update .LAST_RUN file."); } } // Iterate over published sources for (const auto& source : ppa.getPublishedSources("", "", current_series, false, true, "", "", "Published", "")) { for (const auto& build : source.getBuilds()) { if (build.buildstate == "Successfully built") { // Assuming build.datebuilt is a std::chrono::system_clock::time_point if (build.datebuilt >= lastRunTime) { // Enqueue the process_sources task using semaphore and threads threadsRef.emplace_back([=]() { semaphore_guard guard(semaphore); process_sources(build.changesfile_url, fs::path(BASE_OUTPUT_DIR), lintianTmpDir); }); } } } } }; // Start the main_source_iter and enqueue tasks main_source_iter(threads); // Wait for all threads to complete for(auto &t : threads) { if(t.joinable()) { t.join(); } } log_info_custom("All lintian tasks completed. Syncing temporary lintian data to final directory."); rsync_copy(lintianTmpDir, lintianDir); // Remove temporary lintian directory fs::remove_all(lintianTmpDir); // Clean old logs clean_old_logs(LOG_DIR, 86400); // 1 day in seconds, adjust as needed log_info_custom("Lintian-ppa processing completed successfully."); // Close the global log file if (globalLogFile.is_open()) { globalLogFile.close(); } return 0; }