/* * Copyright (C) 2021-2022 Riyyi * * SPDX-License-Identifier: MIT */ #include // replace #include #include // fprintf, printf, stderr #include // system #include // exists #include #include // istringstream #include // getline #include #include "machine.h" #include "package.h" #include "ruc/file.h" #include "ruc/shell.h" #include "ruc/system.h" Package::Package(s) { } Package::~Package() { } // ----------------------------------------- void Package::aurInstall() { installOrAurInstall(InstallType::AurInstall); } void Package::install() { installOrAurInstall(InstallType::Install); } void Package::list(const std::vector& targets) { auto packagesOrEmpty = getPackageList(); if (!packagesOrEmpty.has_value()) { return; } std::string packages = packagesOrEmpty.value(); if (targets.empty()) { printf("%s", packages.c_str()); return; } auto stream = std::istringstream(packages); packages.clear(); // FIXME: Decide on the type of match, currently 'or, any part of the string'. std::string line; while (std::getline(stream, line)) { for (const auto& target : targets) { if (line.find(target) != std::string::npos) { packages.append(line + '\n'); break; } } } printf("%s", packages.c_str()); } void Package::store() { auto packagesOrEmpty = getPackageList(); if (!packagesOrEmpty.has_value()) { return; } auto packageFile = ruc::File::create("./packages"); packageFile.clear(); packageFile.append(packagesOrEmpty.value()); packageFile.flush(); } // ----------------------------------------- std::optional Package::fetchAurHelper() { const std::string helpers[] = { "yay", "paru", "trizen", }; for (const auto& helper : helpers) { if (findDependency(helper)) { return { helper }; } } return {}; } void Package::installOrAurInstall(InstallType type) { distroDetect(); distroDependencies(); std::optional aurHelper; if (type == InstallType::AurInstall) { if (m_distro != Distro::Arch) { fprintf(stderr, "\033[31;1mPackage:\033[0m AUR is not supported on this distribution\n"); return; } aurHelper = fetchAurHelper(); if (!aurHelper.has_value()) { fprintf(stderr, "\033[31;1mPackage:\033[0m no supported AUR helper found\n"); return; } } std::string command = ""; ruc::System $; if (m_distro == Distro::Arch) { // Grab everything off enabled official repositories that is in the list auto repoList = $("pacman -Ssq") | $("grep -xf ./packages"); if (type == InstallType::AurInstall) { // Determine which packages in the list are from the AUR // NOTE: ruc::System does not support commands with newlines auto aurList = ruc::Shell()("grep -vx '" + repoList.output() + "' ./packages"); command = aurHelper.value() + " -Sy --devel --needed --noconfirm " + aurList.output(); } else { command = "pacman -Sy --needed " + repoList.output(); } } else if (m_distro == Distro::Debian) { // Grab everything off enabled official repositories that is in the list auto repoList = $("apt-cache search .").cut(1, ' ') | $("grep -xf ./packages"); command = "apt install " + repoList.output(); } std::replace(command.begin(), command.end(), '\n', ' '); #ifndef NDEBUG printf("running: $ %s\n", command.c_str()); #endif system(command.c_str()); } bool Package::findDependency(const std::string& search) { return std::filesystem::exists("/bin/" + search) || std::filesystem::exists("/usr/bin/" + search) || std::filesystem::exists("/usr/local/bin/" + search); } bool Package::distroDetect() { std::string id = Machine::the().distroId(); std::string idLike = Machine::the().distroIdLike(); if (id == "arch") { m_distro = Distro::Arch; } else if (id == "debian") { m_distro = Distro::Debian; } else if (id == "ubuntu") { m_distro = Distro::Debian; } else if (idLike.find("arch") != std::string::npos) { m_distro = Distro::Arch; } else if (idLike.find("debian") != std::string::npos) { m_distro = Distro::Debian; } else if (idLike.find("ubuntu") != std::string::npos) { m_distro = Distro::Debian; } else { fprintf(stderr, "\033[31;1mPackage:\033[0m unsupported distribution\n"); return false; } return true; } bool Package::distroDependencies() { std::vector> dependencies; if (m_distro == Distro::Arch) { dependencies.push_back({ "pacman", "pacman" }); dependencies.push_back({ "pactree", "pacman-contrib" }); } else if (m_distro == Distro::Debian) { dependencies.push_back({ "apt-cache", "apt" }); dependencies.push_back({ "apt-mark", "apt" }); dependencies.push_back({ "dpkg-query", "dpkg" }); } for (const auto& dependency : dependencies) { if (!findDependency(dependency.at(0))) { fprintf(stderr, "\033[31;1mPackage:\033[0m required dependency '%s' is missing\n", dependency.at(1).c_str()); return false; } } return true; } std::optional Package::getPackageList() { if (!distroDetect() || !distroDependencies()) { return {}; } std::string packages; ruc::System $; if (m_distro == Distro::Arch) { auto basePackages = $("pactree -u base").tail(2, true); auto develPackages = $("pacman -Qqg base-devel"); auto filterList = (basePackages + develPackages).sort(true); auto packageList = ($("pacman -Qqe") | $("grep -xv " + filterList.output())).sort(); packages = packageList.output(); } else if (m_distro == Distro::Debian) { auto installedList = $("dpkg-query --show --showformat=${Package}\\t${Priority}\\n"); auto filterList = (installedList | $("grep -E required|important|standard")).cut(1); installedList = installedList.cut(1); auto installedManuallyList = $("awk '/Commandline:.* install / && !/APT::/ { print $NF }' /var/log/apt/history.log"); installedManuallyList = (installedManuallyList + $("apt-mark showmanual")).sort(true); auto packageList = installedManuallyList | $("grep -x " + installedList.output()) | $("grep -xv " + filterList.output()); packages = packageList.output(); } return packages; }