Skip to content

Commit

Permalink
solver: cache only packages for dependencies without extras and use t…
Browse files Browse the repository at this point in the history
…hem also for dependencies with extras

This way, we can ensure that explicit sources are propagated correctly - no matter if there is a transitive dependency with the same/no/other extras.
  • Loading branch information
radoering committed Jan 21, 2024
1 parent a199981 commit 2a83b5e
Show file tree
Hide file tree
Showing 2 changed files with 166 additions and 2 deletions.
15 changes: 13 additions & 2 deletions src/poetry/mixology/version_solver.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
from poetry.mixology.result import SolverResult
from poetry.mixology.set_relation import SetRelation
from poetry.mixology.term import Term
from poetry.packages import PackageCollection


if TYPE_CHECKING:
Expand Down Expand Up @@ -99,18 +100,28 @@ def search_for(
decision_level: int,
) -> list[DependencyPackage]:
key = (
dependency.complete_name,
dependency.name,
dependency.source_type,
dependency.source_url,
dependency.source_reference,
dependency.source_subdirectory,
)

packages = self._search_for_cached(dependency, key)
# We could always use dependency.without_features() here,
# but for performance reasons we only do it if necessary.
packages = self._search_for_cached(
dependency.without_features() if dependency.features else dependency, key
)
if not self._cache[key] or self._cache[key][-1] is not packages:
self._cache[key].append(packages)
self._cached_dependencies_by_level[decision_level].append(key)

if dependency.features and packages:
# Use the cached dependency so that a possible explicit source is set.
return PackageCollection(
packages[0].dependency.with_features(dependency.features), packages
)

return packages

def clear_level(self, level: int) -> None:
Expand Down
153 changes: 153 additions & 0 deletions tests/puzzle/test_solver.py
Original file line number Diff line number Diff line change
Expand Up @@ -3263,6 +3263,159 @@ def test_solver_ignores_explicit_repo_for_transient_dependencies(
solver.solve()


@pytest.mark.parametrize(
("lib_versions", "other_versions"),
[
# number of versions influences which dependency is resolved first
(["1.0", "2.0"], ["1.0", "1.1", "2.0"]), # more other than lib
(["1.0", "1.1", "2.0"], ["1.0", "2.0"]), # more lib than other
],
)
def test_direct_dependency_with_extras_from_explicit_and_transitive_dependency(
package: ProjectPackage,
repo: Repository,
pool: RepositoryPool,
io: NullIO,
lib_versions: list[str],
other_versions: list[str],
) -> None:
"""
The root package depends on "lib[extra]" and "other", both with an explicit source.
"other" depends on "lib" (without an extra and of course without an explicit source
because explicit sources can only be defined in the root package).
If "other" is resolved before "lib[extra]", the solver must not try to fetch "lib"
from the default source but from the explicit source defined for "lib[extra]".
"""
package.add_dependency(
Factory.create_dependency(
"lib", {"version": ">=1.0", "extras": ["extra"], "source": "explicit"}
)
)
package.add_dependency(
Factory.create_dependency("other", {"version": ">=1.0", "source": "explicit"})
)

explicit_repo = Repository("explicit")
pool.add_repository(explicit_repo, priority=Priority.EXPLICIT)

package_extra = get_package("extra", "1.0")
repo.add_package(package_extra) # extra only in default repo

for version in lib_versions:
package_lib = get_package("lib", version)

dep_extra = get_dependency("extra", ">=1.0")
package_lib.add_dependency(
Factory.create_dependency("extra", {"version": ">=1.0", "optional": True})
)
package_lib.extras[canonicalize_name("extra")] = [dep_extra]

explicit_repo.add_package(package_lib) # lib only in explicit repo

for version in other_versions:
package_other = get_package("other", version)
package_other.add_dependency(Factory.create_dependency("lib", ">=1.0"))
explicit_repo.add_package(package_other) # other only in explicit repo

solver = Solver(package, pool, [], [], io)

transaction = solver.solve()

check_solver_result(
transaction,
[
{"job": "install", "package": get_package("extra", "1.0")},
{"job": "install", "package": get_package("lib", "2.0")},
{"job": "install", "package": get_package("other", "2.0")},
],
)


@pytest.mark.parametrize(
("lib_versions", "other_versions"),
[
# number of versions influences which dependency is resolved first
(["1.0", "2.0"], ["1.0", "1.1", "2.0"]), # more other than lib
(["1.0", "1.1", "2.0"], ["1.0", "2.0"]), # more lib than other
],
)
def test_direct_dependency_with_extras_from_explicit_and_transitive_dependency2(
package: ProjectPackage,
repo: Repository,
pool: RepositoryPool,
io: NullIO,
lib_versions: list[str],
other_versions: list[str],
) -> None:
"""
The root package depends on "lib[extra]" and "other", both with an explicit source.
"other" depends on "lib[other-extra]" (with another extra and of course without an
explicit source because explicit sources can only be defined in the root package).
The solver must not try to fetch "lib[other-extra]" from the default source
but from the explicit source defined for "lib[extra]".
"""
package.add_dependency(
Factory.create_dependency(
"lib", {"version": ">=1.0", "extras": ["extra"], "source": "explicit"}
)
)
package.add_dependency(
Factory.create_dependency("other", {"version": ">=1.0", "source": "explicit"})
)

explicit_repo = Repository("explicit")
pool.add_repository(explicit_repo, priority=Priority.EXPLICIT)

package_extra = get_package("extra", "1.0")
repo.add_package(package_extra) # extra only in default repo
package_other_extra = get_package("other-extra", "1.0")
repo.add_package(package_other_extra) # extra only in default repo

for version in lib_versions:
package_lib = get_package("lib", version)

dep_extra = get_dependency("extra", ">=1.0")
package_lib.add_dependency(
Factory.create_dependency("extra", {"version": ">=1.0", "optional": True})
)
package_lib.extras[canonicalize_name("extra")] = [dep_extra]

dep_other_extra = get_dependency("other-extra", ">=1.0")
package_lib.add_dependency(
Factory.create_dependency(
"other-extra", {"version": ">=1.0", "optional": True}
)
)
package_lib.extras[canonicalize_name("other-extra")] = [dep_other_extra]

explicit_repo.add_package(package_lib) # lib only in explicit repo

for version in other_versions:
package_other = get_package("other", version)
package_other.add_dependency(
Factory.create_dependency(
"lib", {"version": ">=1.0", "extras": ["other-extra"]}
)
)
explicit_repo.add_package(package_other) # other only in explicit repo

solver = Solver(package, pool, [], [], io)

transaction = solver.solve()

check_solver_result(
transaction,
[
{"job": "install", "package": get_package("other-extra", "1.0")},
{"job": "install", "package": get_package("extra", "1.0")},
{"job": "install", "package": get_package("lib", "2.0")},
{"job": "install", "package": get_package("other", "2.0")},
],
)


def test_solver_discards_packages_with_empty_markers(
package: ProjectPackage,
repo: Repository,
Expand Down

0 comments on commit 2a83b5e

Please sign in to comment.