From 45aa4db0dbf80afa740fad71299a68eeb157a58d Mon Sep 17 00:00:00 2001 From: Evan Goode Date: Tue, 30 Apr 2024 19:16:22 +0000 Subject: [PATCH] Install oneof module items on `module enable` Like in DNF 5, we should require oneof the module items when a module is enabled. This way, a clearer error message is printed when trying to enable a module for an incompatible architecture. For https://issues.redhat.com/browse/RHEL-16580. --- libdnf/module/ModulePackageContainer.cpp | 64 +++++++++++++++++++++--- libdnf/module/ModulePackageContainer.hpp | 19 ++++++- 2 files changed, 74 insertions(+), 9 deletions(-) diff --git a/libdnf/module/ModulePackageContainer.cpp b/libdnf/module/ModulePackageContainer.cpp index 5727a96b68..2cc30e741d 100644 --- a/libdnf/module/ModulePackageContainer.cpp +++ b/libdnf/module/ModulePackageContainer.cpp @@ -214,6 +214,8 @@ class ModulePackageContainer::Impl { std::map moduleDefaults; std::vector> modulesV2; + std::vector> modules_to_enable; + bool isEnabled(const std::string &name, const std::string &stream); }; @@ -575,6 +577,23 @@ ModulePackageContainer::enable(const ModulePackage * module, const bool count) return enable(module->getName(), module->getStream(), count); } +bool +ModulePackageContainer::enable(const std::string & name, const std::string & stream, const std::vector & module_packages, const bool count) +{ + bool changed{false}; + // Like in DNF 5, create a PackageSet of IDs for the stream to be enabled, + // because it better matches user requirements than just + // "module(name:stream)" provides. (E.g. user might have requested specific + // context or version.) + libdnf::PackageSet package_set{pImpl->moduleSack}; + for (const auto & module_package : module_packages) { + changed |= enable(module_package, count); + package_set.set(module_package->getId()); + } + pImpl->modules_to_enable.push_back(std::make_tuple(name, stream, package_set)); + return changed; +} + /** * @brief Mark module as not part of an enabled stream. */ @@ -691,16 +710,46 @@ ModulePackageContainer::Impl::moduleSolve(const std::vector & m dnf_sack_make_provides_ready(moduleSack); Goal goal(moduleSack); Goal goalWeak(moduleSack); - for (const auto &module : modules) { - std::ostringstream ss; - auto name = module->getName(); - ss << "module(" << name << ":" << module->getStream() << ")"; - Selector selector(moduleSack); - bool optional = persistor->getState(name) == ModuleState::DEFAULT; - selector.set(HY_PKG_PROVIDES, HY_EQ, ss.str().c_str()); + + std::set module_packages_already_requested; + for (auto & module_to_enable : modules_to_enable) { + const auto & name = std::get<0>(module_to_enable); + auto & stream = std::get<1>(module_to_enable); + auto & package_set = std::get<2>(module_to_enable); + + module_packages_already_requested.insert(tfm::format("{}:{}", name, stream)); + + const bool optional = persistor->getState(name) == ModuleState::DEFAULT; + + libdnf::Selector selector{moduleSack}; + selector.set(&package_set); + + // Request to install one of the module packages goal.install(&selector, optional); goalWeak.install(&selector, true); } + + for (const auto &module : modules) { + std::ostringstream ss; + const auto & name = module->getName(); + const auto & stream = module->getStream(); + + const auto & nameAndStream = tfm::format("{}:{}", name, stream); + + // Only request to install "Provides: module(name:stream)" if we + // haven't already requested to install one of the packages belonging + // to that stream + if (module_packages_already_requested.find(nameAndStream) == + module_packages_already_requested.end()) { + ss << "module(" << nameAndStream << ")"; + Selector selector(moduleSack); + bool optional = persistor->getState(name) == ModuleState::DEFAULT; + selector.set(HY_PKG_PROVIDES, HY_EQ, ss.str().c_str()); + goal.install(&selector, optional); + goalWeak.install(&selector, true); + } + } + auto ret = goal.run(static_cast(DNF_IGNORE_WEAK | DNF_FORCE_BEST)); if (debugSolver) { goal.writeDebugdata("debugdata/modules"); @@ -1208,6 +1257,7 @@ void ModulePackageContainer::save() void ModulePackageContainer::rollback() { + pImpl->modules_to_enable.clear(); pImpl->persistor->rollback(); } diff --git a/libdnf/module/ModulePackageContainer.hpp b/libdnf/module/ModulePackageContainer.hpp index a949578c6e..c66a66b2f5 100644 --- a/libdnf/module/ModulePackageContainer.hpp +++ b/libdnf/module/ModulePackageContainer.hpp @@ -149,7 +149,7 @@ struct ModulePackageContainer std::vector requiresModuleEnablement(const libdnf::PackageSet & packages); /** - * @brief Enable module stream. Return true if requested change realy triggers a change in + * @brief Enable module stream. Return true if requested change really triggers a change in * the persistor. * When the count parameter is set to false the change will not count towards the limit of * module state modifications. @@ -161,7 +161,22 @@ struct ModulePackageContainer bool enable(const std::string &name, const std::string &stream, const bool count = true); /** - * @brief Enable module stream. Return true if requested changes realy triggers a change in + * @brief Enable module stream with an explicit set of module packages. + * Return true only if requested change really reiggers a change in the + * persistor. + * When the count parameter is set to false the change will not count towards the limit of + * module state modifications. + * It can throw ModulePackageContainer::EnableMultipleStreamsException or + * ModulePackageContainer::NoModuleException exceprion if module do not exist + * + * @return bool + */ + bool enable(const std::string &name, const std::string &stream, + const std::vector & module_packages, + const bool count = true); + + /** + * @brief Enable module stream. Return true if requested changes really triggers a change in * the persistor. * When the count parameter is set to false the change will not count towards the limit of * module state modifications.