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(pep440): do not ignore post for inclusive ordering #379

Open
wants to merge 3 commits into
base: main
Choose a base branch
from

Conversation

abn
Copy link
Member

@abn abn commented May 26, 2022

Summary by Sourcery

Bug Fixes:

  • Fixed an issue where inclusive ordering with post releases were inconsistent with PEP 440.

@abn abn requested a review from a team May 26, 2022 15:54
@abn abn force-pushed the fix-local-version-comparisons branch from cfdccfc to c4bb9bd Compare May 26, 2022 16:26
@sonarqubecloud
Copy link

Kudos, SonarCloud Quality Gate passed!    Quality Gate passed

Bug A 0 Bugs
Vulnerability A 0 Vulnerabilities
Security Hotspot A 0 Security Hotspots
Code Smell A 0 Code Smells

No Coverage information No Coverage information
No Duplication information No Duplication information

Copy link
Member

@radoering radoering left a comment

Choose a reason for hiding this comment

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

I don't think that's completely correct yet. PEP440 says:

"Comparison and ordering of local versions considers each segment of the local version (divided by a .) separately. If a segment consists entirely of ASCII digits then that section should be considered an integer for comparison purposes and if a segment contains any ASCII letters then that segment is compared lexicographically with case insensitivity. When comparing a numeric and lexicographic segment, the numeric section always compares as greater than the lexicographic segment. Additionally a local version with a great number of segments will always compare as greater than a local version with fewer segments, as long as the shorter local version’s segments match the beginning of the longer local version’s segments exactly."

It seems that comparing local version segments is already prepared here:

_local: tuple[tuple[int, int | str], ...]
if self.local is None:
# Versions without a local segment should sort before those with one.
_local = ((NegativeInfinity(), ""),)
else:
# Versions with a local segment need that segment parsed to implement
# the sorting rules in PEP440.
# - Alpha numeric segments sort before numeric segments
# - Alpha numeric segments sort lexicographically
# - Numeric segments sort numerically
# - Shorter versions sort before longer versions when the prefixes
# match exactly
assert isinstance(self.local, tuple)
_local = tuple(
# We typecast strings that are integers so that they can be compared
(int(i), "") if str(i).isnumeric() else (NegativeInfinity(), i)
for i in self.local
)

tests/semver/test_version.py Outdated Show resolved Hide resolved
src/poetry/core/semver/version_range.py Outdated Show resolved Hide resolved
src/poetry/core/semver/version_range.py Outdated Show resolved Hide resolved
@dimbleby
Copy link
Contributor

dimbleby commented Jun 7, 2022

@abn
Copy link
Member Author

abn commented Jan 13, 2025

@sourcery-ai review

Copy link

sourcery-ai bot commented Jan 13, 2025

Reviewer's Guide by Sourcery

This pull request fixes an issue where local label comparison was inconsistent with PEP 440. The fix ensures that post releases are not ignored when performing inclusive ordering comparisons.

Class diagram showing changes to VersionRange class

classDiagram
    class VersionRange {
        -_min: Version
        -_max: Version
        -_include_min: bool
        -_include_max: bool
        +is_any(): bool
        +is_simple(): bool
        +is_inclusive(): bool
        +is_exclusive(): bool
        +allows(other: Version): bool
    }
    note for VersionRange "Added new properties is_inclusive and is_exclusive
Modified allows() method to handle post-releases correctly"
Loading

Flow diagram for version comparison logic

flowchart TD
    A[Start Version Comparison] --> B{Is minimum version set?}
    B -->|No| E[Allow version]
    B -->|Yes| C{Is exclusive comparison?}
    C -->|No| E
    C -->|Yes| D{Is other version a post-release<br>and min version not?}
    D -->|Yes| F[Reject version]
    D -->|No| E
Loading

File-Level Changes

Change Details Files
Fixed an issue where local label comparison was inconsistent with PEP 440.
  • Added parameterized tests to verify inclusive ordering with post and local releases.
  • Removed old tests that were not parameterized.
tests/constraints/version/test_version_range.py
Ensure that post releases are not ignored when performing inclusive ordering comparisons.
  • Added is_inclusive and is_exclusive properties to the VersionRange class.
  • Modified the allows method to correctly handle post releases when performing exclusive ordered comparisons.
src/poetry/core/constraints/version/version_range.py

Tips and commands

Interacting with Sourcery

  • Trigger a new review: Comment @sourcery-ai review on the pull request.
  • Continue discussions: Reply directly to Sourcery's review comments.
  • Generate a GitHub issue from a review comment: Ask Sourcery to create an
    issue from a review comment by replying to it.
  • Generate a pull request title: Write @sourcery-ai anywhere in the pull
    request title to generate a title at any time.
  • Generate a pull request summary: Write @sourcery-ai summary anywhere in
    the pull request body to generate a PR summary at any time. You can also use
    this command to specify where the summary should be inserted.

Customizing Your Experience

Access your dashboard to:

  • Enable or disable review features such as the Sourcery-generated pull request
    summary, the reviewer's guide, and others.
  • Change the review language.
  • Add, remove or edit custom review instructions.
  • Adjust other review settings.

Getting Help

Copy link

@sourcery-ai sourcery-ai bot left a comment

Choose a reason for hiding this comment

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

Hey @abn - I've reviewed your changes and they look great!

Here's what I looked at during the review
  • 🟢 General issues: all looks good
  • 🟢 Security: all looks good
  • 🟢 Testing: all looks good
  • 🟢 Complexity: all looks good
  • 🟢 Documentation: all looks good

Sourcery is free for open source - if you like our reviews please consider sharing them ✨
Help me be more useful! Please click 👍 or 👎 on each comment and I'll use the feedback to improve your reviews.

@abn abn force-pushed the fix-local-version-comparisons branch from 0416da4 to 1a3a7e9 Compare January 18, 2025 16:05
@radoering
Copy link
Member

radoering commented Jan 18, 2025

Actually, I am not sure if this fixes anything regarding compliance with PEP 440. I even think this will be less compliant with PEP 440 than before.

It feels like switching from one use case to another. According to PEP 440, local version labels are not allowed in constraints with >, <, >=, <=. However, since we are building version ranges, we have to deal with it anyway. Although they are not allowed in constraints, it is defined how they are ordered and compared, see #379 (review)

If I understand correctly, this PR tries to make something like this work:
>= 1.0+cpu does not allow 1.0+cuda or 1.1+cuda (even though they are both greater!) but 1.1+cpu (which is also greater)

However, to make this work, it breaks the use case where >=1.0+internal.1 allows 1.0+internal.2.

Honestly, the second use case feels more correct (i.e. more compliant) than the first one. On the other side, I understand that the first one might be the more common one (at least when neglecting private packages inside of companies).

I think this change will be breaking for some users and make us less compliant with PEP 440. I feel like we should carefully weigh if it is worth it. Is there an open issue about this? The issue that is linked has already been resolved.

@abn abn force-pushed the fix-local-version-comparisons branch 3 times, most recently from 0877107 to 2bff41a Compare January 19, 2025 03:39
@abn abn force-pushed the fix-local-version-comparisons branch from 2bff41a to 1a61f52 Compare January 19, 2025 21:22
@abn
Copy link
Member Author

abn commented Jan 19, 2025

When through the spec again, been a while. And I agree that the original changes I had is swapping one "interpretation" for another. I also agree that >=1.0+internal.1 allows 1.0+internal.2 is better than >= 1.0+cpu does not allow 1.0+cuda as the latter does not comply with if a segment contains any ASCII letters then that segment is compared lexicographically with case insensitivity.

Taking all that into considering, I have gone ahead and repurposed this PR to only include the following:

  1. Test cleanups and new test cases.
  2. Fixing inclusive ordering.

As for what to do with cpu vs cuda, I am more convinced that solving this at the version range level is not the appropriate fix if we want to ensure that VersionRange.allows adheres to PEP440 inclusive and exclusive ordering. We do bend the rules (see quote below) a bit already because we have to consider local versions.

Except where specifically noted below, local version identifiers MUST NOT be permitted in version specifiers, and local version labels MUST be ignored entirely when checking if candidate versions match a given version specifier.

@abn abn changed the title version: fix local label comparison fix(pep440): do not ignore post for inclusive ordering Jan 19, 2025
@abn
Copy link
Member Author

abn commented Jan 19, 2025

@sourcery-ai review

@abn abn dismissed radoering’s stale review January 19, 2025 21:35

Out of date.

Copy link

@sourcery-ai sourcery-ai bot left a comment

Choose a reason for hiding this comment

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

Hey @abn - I've reviewed your changes and they look great!

Here's what I looked at during the review
  • 🟢 General issues: all looks good
  • 🟢 Security: all looks good
  • 🟢 Testing: all looks good
  • 🟢 Complexity: all looks good
  • 🟢 Documentation: all looks good

Sourcery is free for open source - if you like our reviews please consider sharing them ✨
Help me be more useful! Please click 👍 or 👎 on each comment and I'll use the feedback to improve your reviews.

Copy link
Member

@radoering radoering left a comment

Choose a reason for hiding this comment

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

Looks like a valid fix now. 😄 Just some nitpicks.

Comment on lines +62 to +63
def is_inclusive(self) -> bool:
return self._include_min or self.include_max
Copy link
Member

Choose a reason for hiding this comment

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

Isn't the naming a little ambiguous? Why or and not and? Further, the or self.include_max part seems not to be required by any test.

def allows(self, other: Version) -> bool:
if self._min is not None:
_this, _other = self.allowed_min, other

assert _this is not None

if not _this.is_postrelease() and _other.is_postrelease():
if (
self.is_exclusive
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
self.is_exclusive
not self._include_min

That should be sufficient concerning the comment below, shouldn't it? (<V will not allow a post-release anyway.)

pytest.param(
Version.parse("3.0.0"), Version.parse("3.0.0+local.1"), id="local"
),
# Inclusive ordering
Copy link
Member

Choose a reason for hiding this comment

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

This is really an improvement. 👍 One minor thing: Can we add all cases where check_version is the version in the constraint, e.g. <=3.0.0+local1 with 3.0.0+local1, >=3.0.0-1 with 3.0.0-1 and so on?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants