diff --git a/libmamba/include/mamba/fs/filesystem.hpp b/libmamba/include/mamba/fs/filesystem.hpp index a2cf66824e..d3598fc1d6 100644 --- a/libmamba/include/mamba/fs/filesystem.hpp +++ b/libmamba/include/mamba/fs/filesystem.hpp @@ -87,6 +87,9 @@ namespace mamba::fs { public: + using value_type = char; + using string_type = std::basic_string; + u8path() = default; // Copy is allowed. @@ -303,6 +306,12 @@ namespace mamba::fs return to_utf8(m_path); } + // Returns a default encoded string. + const std::string& native() const + { + return m_path.native(); + } + // Returns an utf-8 string. operator std::string() const { diff --git a/libmambapy/src/libmambapy/bindings/expected_caster.hpp b/libmambapy/src/libmambapy/bindings/expected_caster.hpp index 58209a9d29..a73293f66c 100644 --- a/libmambapy/src/libmambapy/bindings/expected_caster.hpp +++ b/libmambapy/src/libmambapy/bindings/expected_caster.hpp @@ -4,9 +4,7 @@ // // The full license is in the file LICENSE, distributed with this software. -#include #include -#include #include #include @@ -20,58 +18,22 @@ namespace PYBIND11_NAMESPACE { namespace detail { - namespace - { - template < - typename Expected, - typename T = typename Expected::value_type, - typename E = typename Expected::error_type> - auto expected_to_variant(Expected&& expected) -> std::variant - { - if (expected) - { - return { std::forward(expected).value() }; - } - return { std::forward(expected).error() }; - } - - template < - typename Variant, - typename T = std::decay_t(std::declval()))>, - typename E = std::decay_t(std::declval()))>> - auto expected_to_variant(Variant&& var) -> tl::expected - { - static_assert(std::variant_size_v == 2); - return std::visit( - [](auto&& v) -> tl::expected { return { std::forward(v) }; }, - var - ); - } - } /** - * A caster for tl::expected that converts to a union. - * - * The caster works by converting to a the expected to a variant and then calls the - * variant caster. - * - * A future direction could be considered to wrap the union into a Python "Expected", - * with methods such as ``and_then``, ``or_else``, and thowing method like ``value`` - * and ``error``. + * A caster for tl::expected that throws on unexpected. */ template struct type_caster> { - using value_type = tl::expected; - using variant_type = std::variant; - using caster_type = variant_caster; + using value_type = T; auto load(handle src, bool convert) -> bool { - auto caster = caster_type(); + auto caster = make_caster(); if (caster.load(src, convert)) { - value = variant_to_expected(cast_op(std::move(caster))); + value = cast_op(std::move(caster)); + return true; } return false; } @@ -79,14 +41,17 @@ namespace PYBIND11_NAMESPACE template static auto cast(Expected&& src, return_value_policy policy, handle parent) -> handle { - return caster_type::cast(expected_to_variant(std::forward(src)), policy, parent); + if (src) + { + return make_caster::cast(std::forward(src).value(), policy, parent); + } + else + { + throw std::forward(src).error(); + } } - PYBIND11_TYPE_CASTER( - value_type, - const_name(R"(Union[)") + detail::concat(make_caster::name, make_caster::name) - + const_name(R"(])") - ); + PYBIND11_TYPE_CASTER(value_type, detail::concat(make_caster::name, make_caster::name)); }; } } diff --git a/libmambapy/src/libmambapy/bindings/legacy.cpp b/libmambapy/src/libmambapy/bindings/legacy.cpp index 4f68869596..6746de96ad 100644 --- a/libmambapy/src/libmambapy/bindings/legacy.cpp +++ b/libmambapy/src/libmambapy/bindings/legacy.cpp @@ -31,8 +31,6 @@ #include "mamba/core/transaction.hpp" #include "mamba/core/util_os.hpp" #include "mamba/core/virtual_packages.hpp" -#include "mamba/solver/libsolv/database.hpp" -#include "mamba/solver/libsolv/repo_info.hpp" #include "mamba/solver/problems_graph.hpp" #include "mamba/validation/tools.hpp" #include "mamba/validation/update_framework_v0_6.hpp" @@ -40,6 +38,7 @@ #include "bindings.hpp" #include "expected_caster.hpp" #include "flat_set_caster.hpp" +#include "path_caster.hpp" #include "utils.hpp" namespace py = pybind11; @@ -247,10 +246,28 @@ bind_submodule_impl(pybind11::module_ m) throw std::runtime_error( // "Use Pool.add_repo_from_repodata_json or Pool.add_repo_from_native_serialization" " instead and cache with Pool.native_serialize_repo." - " Also consider load_subdir_in_pool for a high_level function to load" - " subdir index and manage cache, and load_installed_packages_in_pool for loading" - " prefix packages." - "The Repo class itself has been moved to libmambapy.solver.libsolv.RepoInfo." + " Also consider load_subdir_in_database for a high_level function to load" + " subdir index and manage cache, and load_installed_packages_in_database for" + " loading prefix packages." + " The Repo class itself has been moved to libmambapy.solver.libsolv.RepoInfo." + ); + } + )); + + struct PoolV2Migrator + { + }; + + py::class_(m, "Pool").def(py::init( + [](py::args, py::kwargs) -> PoolV2Migrator + { + throw std::runtime_error( // + "libmambapy.Pool has been moved to libmambapy.solver.libsolv.Database." + " The database contains functions to directly load packages, from a list or a" + " repodata.json." + " High level functions such as libmambapy.load_subdir_in_database and" + " libmambapy.load_installed_packages_in_database are also available to work" + " with other Mamba objects and Context parameters." ); } )); @@ -380,76 +397,19 @@ bind_submodule_impl(pybind11::module_ m) py::add_ostream_redirect(m, "ostream_redirect"); - py::class_(m, "Pool") - .def(py::init(), py::arg("channel_params")) - .def( - "set_logger", - &solver::libsolv::Database::set_logger, - py::call_guard() - ) - .def( - "add_repo_from_repodata_json", - &solver::libsolv::Database::add_repo_from_repodata_json, - py::arg("path"), - py::arg("url"), - py::arg("add_pip_as_python_dependency") = solver::libsolv::PipAsPythonDependency::No, - py::arg("use_only_tar_bz2") = solver::libsolv::UseOnlyTarBz2::No, - py::arg("repodata_parsers") = solver::libsolv::RepodataParser::Mamba - ) - .def( - "add_repo_from_native_serialization", - &solver::libsolv::Database::add_repo_from_native_serialization, - py::arg("path"), - py::arg("expected"), - py::arg("add_pip_as_python_dependency") = solver::libsolv::PipAsPythonDependency::No - ) - .def( - "add_repo_from_packages", - [](solver::libsolv::Database& db, - py::iterable packages, - std::string_view name, - solver::libsolv::PipAsPythonDependency add) - { - // TODO(C++20): No need to copy in a vector, simply transform the input range. - auto pkg_infos = std::vector(); - for (py::handle pkg : packages) - { - pkg_infos.push_back(pkg.cast()); - } - return db.add_repo_from_packages(pkg_infos, name, add); - }, - py::arg("packages"), - py::arg("name") = "", - py::arg("add_pip_as_python_dependency") = solver::libsolv::PipAsPythonDependency::No - ) - .def( - "native_serialize_repo", - &solver::libsolv::Database::native_serialize_repo, - py::arg("repo"), - py::arg("path"), - py::arg("metadata") - ) - .def("set_installed_repo", &solver::libsolv::Database::set_installed_repo, py::arg("repo")) - .def( - "set_repo_priority", - &solver::libsolv::Database::set_repo_priority, - py::arg("repo"), - py::arg("priorities") - ); - m.def( - "load_subdir_in_pool", + "load_subdir_in_database", &load_subdir_in_database, py::arg("context"), - py::arg("pool"), + py::arg("database"), py::arg("subdir") ); m.def( - "load_installed_packages_in_pool", + "load_installed_packages_in_database", &load_installed_packages_in_database, py::arg("context"), - py::arg("pool"), + py::arg("database"), py::arg("prefix_data") ); @@ -534,7 +494,7 @@ bind_submodule_impl(pybind11::module_ m) "create_repo", [](SubdirData& subdir, solver::libsolv::Database& db) -> solver::libsolv::RepoInfo { - deprecated("Use `load_subdir_in_pool` instead", "2.0"); + deprecated("Use libmambapy.load_subdir_in_database instead", "2.0"); return extract(load_subdir_in_database(mambapy::singletons.context(), db, subdir)); } ) diff --git a/libmambapy/src/libmambapy/bindings/path_caster.hpp b/libmambapy/src/libmambapy/bindings/path_caster.hpp new file mode 100644 index 0000000000..59ecf2e674 --- /dev/null +++ b/libmambapy/src/libmambapy/bindings/path_caster.hpp @@ -0,0 +1,21 @@ +// Copyright (c) 2024, QuantStack and Mamba Contributors +// +// Distributed under the terms of the BSD 3-Clause License. +// +// The full license is in the file LICENSE, distributed with this software. + +#include +#include + +#include "mamba/fs/filesystem.hpp" + +namespace PYBIND11_NAMESPACE +{ + namespace detail + { + template <> + struct type_caster : path_caster + { + }; + } +} diff --git a/libmambapy/src/libmambapy/bindings/solver_libsolv.cpp b/libmambapy/src/libmambapy/bindings/solver_libsolv.cpp index 4d0b86cbbf..67b034563a 100644 --- a/libmambapy/src/libmambapy/bindings/solver_libsolv.cpp +++ b/libmambapy/src/libmambapy/bindings/solver_libsolv.cpp @@ -16,6 +16,7 @@ #include "bindings.hpp" #include "expected_caster.hpp" +#include "path_caster.hpp" #include "utils.hpp" namespace mambapy @@ -98,15 +99,109 @@ namespace mambapy .def("__deepcopy__", &deepcopy, py::arg("memo")); py::class_(m, "RepoInfo") - .def("id", &RepoInfo::id) - .def("name", &RepoInfo::name) - .def("priority", &RepoInfo::priority) + .def_property_readonly("id", &RepoInfo::id) + .def_property_readonly("name", &RepoInfo::name) + .def_property_readonly("priority", &RepoInfo::priority) .def("package_count", &RepoInfo::package_count) .def(py::self == py::self) .def(py::self != py::self) .def("__copy__", ©) .def("__deepcopy__", &deepcopy, py::arg("memo")); + py::class_(m, "Database") + .def(py::init(), py::arg("channel_params")) + .def("set_logger", &Database::set_logger, py::call_guard()) + .def( + "add_repo_from_repodata_json", + &Database::add_repo_from_repodata_json, + py::arg("path"), + py::arg("url"), + py::arg("add_pip_as_python_dependency") = PipAsPythonDependency::No, + py::arg("use_only_tar_bz2") = UseOnlyTarBz2::No, + py::arg("repodata_parser") = RepodataParser::Mamba + ) + .def( + "add_repo_from_native_serialization", + &Database::add_repo_from_native_serialization, + py::arg("path"), + py::arg("expected"), + py::arg("add_pip_as_python_dependency") = PipAsPythonDependency::No + ) + .def( + "add_repo_from_packages", + [](Database& db, py::iterable packages, std::string_view name, PipAsPythonDependency add) + { + // TODO(C++20): No need to copy in a vector, simply transform the input range. + auto pkg_infos = std::vector(); + for (py::handle pkg : packages) + { + pkg_infos.push_back(pkg.cast()); + } + return db.add_repo_from_packages(pkg_infos, name, add); + }, + py::arg("packages"), + py::arg("name") = "", + py::arg("add_pip_as_python_dependency") = PipAsPythonDependency::No + ) + .def( + "native_serialize_repo", + &Database::native_serialize_repo, + py::arg("repo"), + py::arg("path"), + py::arg("metadata") + ) + .def("set_installed_repo", &Database::set_installed_repo, py::arg("repo")) + .def("installed_repo", &Database::installed_repo) + .def("set_repo_priority", &Database::set_repo_priority, py::arg("repo"), py::arg("priorities")) + .def("remove_repo", &Database::remove_repo, py::arg("repo")) + .def("repo_count", &Database::repo_count) + .def("package_count", &Database::package_count) + .def( + "packages_in_repo", + [](const Database& db, RepoInfo repo) + { + // TODO(C++20): When Database function are refactored to use range, take the + // opportunity here to make a Python iterator to avoid large alloc. + auto out = py::list(); + db.for_each_package_in_repo( + repo, + [&](specs::PackageInfo&& pkg) { out.append(std::move(pkg)); } + ); + return out; + }, + py::arg("repo") + ) + .def( + "packages_matching", + [](Database& db, const specs::MatchSpec& ms) + { + // TODO(C++20): When Database function are refactored to use range, take the + // opportunity here to make a Python iterator to avoid large alloc. + auto out = py::list(); + db.for_each_package_matching( + ms, + [&](specs::PackageInfo&& pkg) { out.append(std::move(pkg)); } + ); + return out; + }, + py::arg("spec") + ) + .def( + "packages_depending_on", + [](Database& db, const specs::MatchSpec& ms) + { + // TODO(C++20): When Database function are refactored to use range, take the + // opportunity here to make a Python iterator to avoid large alloc. + auto out = py::list(); + db.for_each_package_depending_on( + ms, + [&](specs::PackageInfo&& pkg) { out.append(std::move(pkg)); } + ); + return out; + }, + py::arg("spec") + ); + py::class_(m, "UnSolvable") .def("problems", &UnSolvable::problems) .def("problems_to_str", &UnSolvable::problems_to_str) diff --git a/libmambapy/tests/test_solver_libsolv.py b/libmambapy/tests/test_solver_libsolv.py index 714045692e..f882824fb6 100644 --- a/libmambapy/tests/test_solver_libsolv.py +++ b/libmambapy/tests/test_solver_libsolv.py @@ -1,7 +1,10 @@ import copy +import json +import itertools import pytest +import libmambapy import libmambapy.solver.libsolv as libsolv @@ -103,3 +106,126 @@ def test_RepodataOrigin(): other = copy.deepcopy(orig) assert other is not orig assert other == orig + + +@pytest.mark.parametrize("add_pip_as_python_dependency", [True, False]) +def test_Database_RepoInfo_from_packages(add_pip_as_python_dependency): + db = libsolv.Database(libmambapy.specs.ChannelResolveParams()) + assert db.repo_count() == 0 + assert db.installed_repo() is None + assert db.package_count() == 0 + + repo = db.add_repo_from_packages( + [libmambapy.specs.PackageInfo(name="python")], + name="duck", + add_pip_as_python_dependency=add_pip_as_python_dependency, + ) + db.set_installed_repo(repo) + + assert repo.id > 0 + assert repo.name == "duck" + assert repo.priority == libsolv.Priorities() + assert repo.package_count() == 1 + assert db.repo_count() == 1 + assert db.package_count() == 1 + assert db.installed_repo() == repo + + new_priority = libsolv.Priorities(2, 3) + db.set_repo_priority(repo, new_priority) + assert repo.priority == new_priority + + pkgs = db.packages_in_repo(repo) + assert len(pkgs) == 1 + assert pkgs[0].name == "python" + assert pkgs[0].depends == [] if add_pip_as_python_dependency else ["pip"] + + db.remove_repo(repo) + assert db.repo_count() == 0 + assert db.package_count() == 0 + assert db.installed_repo() is None + + +@pytest.fixture +def tmp_repodata_json(tmp_path): + file = tmp_path / "repodata.json" + with open(file, "w+") as f: + json.dump( + { + "packages": { + "python-1.0-bld": { + "name": "python", + "version": "1.0", + "build": "bld", + "build_number": 0, + }, + }, + "packages.conda": { + "foo-1.0-bld": { + "name": "foo", + "version": "1.0", + "build": "bld", + "build_number": 0, + } + }, + }, + f, + ) + return file + + +@pytest.mark.parametrize( + ["add_pip_as_python_dependency", "use_only_tar_bz2", "repodata_parser"], + itertools.product([True, False], [True, False], ["Mamba", "Libsolv"]), +) +def test_Database_RepoInfo_from_repodata( + tmp_path, tmp_repodata_json, add_pip_as_python_dependency, use_only_tar_bz2, repodata_parser +): + db = libsolv.Database(libmambapy.specs.ChannelResolveParams()) + + url = "https://repo.mamba.pm" + + # Json + repo = db.add_repo_from_repodata_json( + path=tmp_repodata_json, + url=url, + add_pip_as_python_dependency=add_pip_as_python_dependency, + use_only_tar_bz2=use_only_tar_bz2, + repodata_parser=repodata_parser, + ) + db.set_installed_repo(repo) + + assert repo.package_count() == 1 if use_only_tar_bz2 else 2 + assert db.package_count() == repo.package_count() + + pkgs = db.packages_in_repo(repo) + assert len(pkgs) == repo.package_count() + assert pkgs[0].name == "python" + assert pkgs[0].depends == [] if add_pip_as_python_dependency else ["pip"] + + # Native serialize repo + solv_file = tmp_path / "repodata.solv" + + origin = libsolv.RepodataOrigin(url=url) + repo_saved = db.native_serialize_repo(repo, path=solv_file, metadata=origin) + assert repo_saved == repo + + # Native deserialize repo + db.remove_repo(repo) + assert db.package_count() == 0 + + repo_loaded = db.add_repo_from_native_serialization( + path=solv_file, expected=origin, add_pip_as_python_dependency=add_pip_as_python_dependency + ) + assert repo_loaded.package_count() == 1 if use_only_tar_bz2 else 2 + + +def test_Database_RepoInfo_from_repodata_error(): + db = libsolv.Database(libmambapy.specs.ChannelResolveParams()) + + with pytest.raises(libmambapy.MambaNativeException, match="/does/not/exists"): + db.add_repo_from_repodata_json(path="/does/not/exists", url="https://repo..mamba.pm") + + with pytest.raises(libmambapy.MambaNativeException, match="/does/not/exists"): + db.add_repo_from_native_serialization( + path="/does/not/exists", expected=libsolv.RepodataOrigin() + )