Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix reject of valid state in criteria compatibility check #111

Merged
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions setup.cfg
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ lint =
mypy
isort
types-requests
types-setuptools
test =
commentjson
packaging
Expand Down
42 changes: 42 additions & 0 deletions src/resolvelib/resolvers.py
Original file line number Diff line number Diff line change
Expand Up @@ -173,6 +173,32 @@ def _add_to_criteria(self, criteria, requirement, parent):
raise RequirementsConflicted(criterion)
criteria[identifier] = criterion

def _remove_information_from_criteria(self, criteria, parents):
"""Remove information from parents of criteria.

Concretely, removes all values from each criterion's ``information``
field that have one of ``parents`` as provider of the requirement.

:param criteria: The criteria to update.
:param parents: Identifiers for which to remove information from all criteria.
"""
if not parents:
return
for key, criterion in criteria.items():
criteria[key] = Criterion(
criterion.candidates,
# TODO: is empty information allowed?
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Empty information is technically allowed but likely unexpected, since it records where a criterion came from, and if a criterion comes from nowhere, should it not be removed?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My reasoning was that we're dropping the constraint provided by a parent because we're dropping the pin on that parent. But if the new pin on the parent happens to have the same constraint (the common case), we might want to keep incompatibilities, in order to prevent backtracking over the same thing again.
But as I type this I realize that I'm really not yet confident in my understanding of how incompatibilities are gathered. So if you are confident that we can safely drop the criterion when no information remains I'll apply that change. Otherwise I'll have to sink my teeth in the incompatibilities some more.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the intent is to keep incompatibilities in the graph, I think it’s OK to keep a criterion with empty information (it’s basically an orphan node). We should describe this in the release note (by adding a release note fragment file in news) so provider/reporter implementers are aware of the possibility that information could be empty (and what that means), it’s likely no-one would be particular troubled by this since an orphan node can simply be ignored in the final graph anyway.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

All right, I'll do that. I think pip will need a minor change, but there seems to already be a breaking change in master compared to the last release anyway so I guess that is acceptable.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the orphan nodes will be excluded automatically by build_result()

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Indeed they will for the final result, but not for intermediate calls to the provider, see here: #91 (comment).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've made an attempt at the news fragment but I'm not entirely sure about your change entry conventions.

[
information
for information in criterion.information
if (
information[1] is None
or self._p.identify(information[1]) not in parents
)
],
criterion.incompatibilities,
)

def _get_preference(self, name):
return self._p.get_preference(
identifier=name,
Expand Down Expand Up @@ -367,6 +393,11 @@ def resolve(self, requirements, max_rounds):
self._r.ending(state=self.state)
return self.state

# keep track of satisfied names to calculate diff after pinning
satisfied_names = set(self.state.criteria.keys()) - set(
unsatisfied_names
)

# Choose the most preferred unpinned criterion to try.
name = min(unsatisfied_names, key=self._get_preference)
failure_causes = self._attempt_to_pin_criterion(name)
Expand All @@ -383,6 +414,17 @@ def resolve(self, requirements, max_rounds):
if not success:
raise ResolutionImpossible(self.state.backtrack_causes)
else:
# discard as information sources any invalidated names
# (unsatisfied names that were previously satisfied)
newly_unsatisfied_names = {
key
for key, criterion in self.state.criteria.items()
if key in satisfied_names
if not self._is_current_pin_satisfying(key, criterion)
sanderr marked this conversation as resolved.
Show resolved Hide resolved
}
self._remove_information_from_criteria(
self.state.criteria, newly_unsatisfied_names
)
# Pinning was successful. Push a new state to do another pin.
self._push_new_state()

Expand Down
124 changes: 124 additions & 0 deletions tests/test_resolvers.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,18 @@
from typing import (
Any,
Iterable,
Iterator,
List,
Mapping,
Sequence,
Set,
Tuple,
Union,
)

import pytest
from packaging.version import Version
from pkg_resources import Requirement
uranusjr marked this conversation as resolved.
Show resolved Hide resolved

from resolvelib import (
AbstractProvider,
Expand All @@ -7,6 +21,12 @@
ResolutionImpossible,
Resolver,
)
from resolvelib.resolvers import Resolution # type: ignore
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What errors does this ignore?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Resolution is not part of the public interface declared in src/resolvelib/resolvers.pyi. My reasoning was that for testing that does not matter. I meant to ask about this but I forgot about it. The error is as follows:

tests/test_resolvers.py:24: error: Module "resolvelib.resolvers" has no attribute "Resolution"; maybe "ResolutionError"?
Found 1 error in 1 file (checked 1 source file)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can add Resolution to the type stubs; I think someone would need it at some point anyway.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

All right, I'll do that, thanks.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. I did have to add another more specific type ignore because the test case monkeypatches a private method. I think that should be acceptable?

from resolvelib.resolvers import (
Criterion,
RequirementInformation,
RequirementsConflicted,
)


def test_candidate_inconsistent_error():
Expand Down Expand Up @@ -143,3 +163,107 @@ def run_resolver(*args):
backtracking_causes = run_resolver([("a", {1, 2}), ("b", {1})])
exception_causes = run_resolver([("a", {2}), ("b", {1})])
assert exception_causes == backtracking_causes


def test_pin_conflict_with_self(monkeypatch, reporter):
# type: (Any, BaseReporter) -> None
"""
Verify correct behavior of attempting to pin a candidate version that conflicts
with a previously pinned (now invalidated) version for that same candidate (#91).
"""
Candidate = Tuple[
str, Version, Sequence[str]
] # name, version, requirements
all_candidates = {
"parent": [("parent", Version("1"), ["child<2"])],
"child": [
("child", Version("2"), ["grandchild>=2"]),
("child", Version("1"), ["grandchild<2"]),
("child", Version("0.1"), ["grandchild"]),
],
"grandchild": [
("grandchild", Version("2"), []),
("grandchild", Version("1"), []),
],
} # type: Mapping[str, Sequence[Candidate]]

class Provider(AbstractProvider): # AbstractProvider[int, Candidate, str]
def identify(self, requirement_or_candidate):
# type: (Union[str, Candidate]) -> str
result = (
Requirement.parse(requirement_or_candidate).key
if isinstance(requirement_or_candidate, str)
else requirement_or_candidate[0]
)
assert result in all_candidates, "unknown requirement_or_candidate"
return result

def get_preference(self, identifier, *args, **kwargs):
# type: (str, *object, **object) -> str
# prefer child over parent (alphabetically)
return identifier

def get_dependencies(self, candidate):
# type: (Candidate) -> Sequence[str]
return candidate[2]

def find_matches(
self,
identifier, # type: str
requirements, # type: Mapping[str, Iterator[str]]
incompatibilities, # type: Mapping[str, Iterator[Candidate]]
):
# type: (...) -> Iterator[Candidate]
return (
candidate
for candidate in all_candidates[identifier]
if all(
self.is_satisfied_by(req, candidate)
for req in requirements[identifier]
)
if candidate not in incompatibilities[identifier]
)

def is_satisfied_by(self, requirement, candidate):
# type: (str, Candidate) -> bool
return str(candidate[1]) in Requirement.parse(requirement)

# patch Resolution._get_updated_criteria to collect rejected states
rejected_criteria = [] # type: List[Criterion]
get_updated_criteria_orig = Resolution._get_updated_criteria

def get_updated_criteria_patch(self, candidate):
try:
return get_updated_criteria_orig(self, candidate)
except RequirementsConflicted as e:
rejected_criteria.append(e.criterion)
raise

monkeypatch.setattr(
Resolution, "_get_updated_criteria", get_updated_criteria_patch
)

resolver = Resolver(
Provider(), reporter
) # type: Resolver[str, Candidate, str]
result = resolver.resolve(["child", "parent"])

def get_child_versions(information):
# type: (Iterable[RequirementInformation[str, Candidate]]) -> Set[str]
return {
str(inf.parent[1])
for inf in information
if inf.parent is not None and inf.parent[0] == "child"
}

# verify that none of the rejected criteria are based on more than one candidate for
# child
assert not any(
len(get_child_versions(criterion.information)) > 1
for criterion in rejected_criteria
)

assert set(result.mapping) == {"parent", "child", "grandchild"}
assert result.mapping["parent"][1] == Version("1")
assert result.mapping["child"][1] == Version("1")
assert result.mapping["grandchild"][1] == Version("1")