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

Feature/better version ranges #14912

Merged
6 changes: 5 additions & 1 deletion conans/client/graph/graph_builder.py
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,11 @@ def _conflicting_version(require, node,
if version_range:
# TODO: Check user/channel conflicts first
if prev_version_range is not None:
pass # Do nothing, evaluate current as it were a fixed one
# It it is not conflicting, but range can be incompatible, restrict range
restricted_version_range = version_range.intersection(prev_version_range)
if restricted_version_range is None:
raise GraphConflictError(node, require, prev_node, prev_require, base_previous)
require.ref.version = restricted_version_range.version()
else:
if version_range.contains(prev_ref.version, resolve_prereleases):
require.ref = prev_ref
Expand Down
64 changes: 62 additions & 2 deletions conans/model/version_range.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,42 @@
from collections import namedtuple
from functools import total_ordering
from typing import Optional

from conans.errors import ConanException
from conans.model.recipe_ref import Version


_Condition = namedtuple("_Condition", ["operator", "version"])
@total_ordering
class _Condition:
def __init__(self, operator, version):
self.operator = operator
self.version = version

def __str__(self):
return f"{self.operator}{self.version}"

def __repr__(self):
return self.__str__()

def __hash__(self):
return hash((self.operator, self.version))

def __lt__(self, other):
if self.version < other.version:
return True
elif self.version == other.version:
if "<" in self.operator:
assert "<" in other.operator # Only ordering possible
return other.operator == "<="
else:
if ">" in other.operator:
return other.operator == ">"
else: # valid range check lower against upper
return self.operator == ">=" and other.operator == "<="
return False

def __eq__(self, other):
return (self.version == other.version and
self.operator == other.operator)


class _ConditionSet:
Expand Down Expand Up @@ -135,3 +166,32 @@ def contains(self, version: Version, resolve_prerelease: Optional[bool]):
return True
return False

def intersection(self, other):
conditions = []

def _calculate_limits(operator, lhs, rhs):
limits = ([c for c in lhs.conditions if operator in c.operator]
+ [c for c in rhs.conditions if operator in c.operator])
if limits:
return sorted(limits, reverse=operator == ">")[0]

for lhs_conditions in self.condition_sets:
for rhs_conditions in other.condition_sets:
internal_conditions = []
lower_limit = _calculate_limits(">", lhs_conditions, rhs_conditions)
upper_limit = _calculate_limits("<", lhs_conditions, rhs_conditions)
internal_conditions.append(lower_limit)
internal_conditions.append(upper_limit)
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
if internal_conditions and (not lower_limit or not upper_limit or lower_limit <= upper_limit):
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
conditions.append(internal_conditions)

if not conditions:
return None
expression = ' || '.join(' '.join(str(c) for c in cs if c is not None) for cs in conditions)
result = VersionRange(expression)
# TODO: Direct definition of conditions not reparsing
# result.condition_sets = self.condition_sets + other.condition_sets
return result

def version(self):
return Version(f"[{self._expression}]")
9 changes: 5 additions & 4 deletions conans/test/integration/graph/core/test_version_ranges.py
Original file line number Diff line number Diff line change
Expand Up @@ -322,9 +322,10 @@ def test_two_ranges_overriden(self):
self._check_node(libb, "libb/0.1#123", dependents=[app], deps=[liba])
self._check_node(app, "app/0.1", deps=[libb, liba])

def test_two_ranges_overriden_conflict(self):
def test_two_ranges_overriden_no_conflict(self):
# app -> libb/0.1 -(range >0)-> liba/0.1
# \ ---------liba/[<0.3>]-------------/
# Conan learned to solve this conflict in 2.0.14
self.recipe_cache("liba/0.1")
self.recipe_cache("liba/0.2")
self.recipe_cache("liba/0.3")
Expand All @@ -333,16 +334,16 @@ def test_two_ranges_overriden_conflict(self):
.with_requirement("liba/[<0.3]"))
deps_graph = self.build_consumer(consumer, install=False)

assert type(deps_graph.error) == GraphConflictError
# This is no longer a conflict, and Conan knows that liba/2.0 is a valid joint solution

self.assertEqual(3, len(deps_graph.nodes))
app = deps_graph.root
libb = app.dependencies[0].dst
liba = libb.dependencies[0].dst

self._check_node(liba, "liba/0.3#123", dependents=[libb], deps=[])
self._check_node(liba, "liba/0.2#123", dependents=[libb, app], deps=[])
self._check_node(libb, "libb/0.1#123", dependents=[app], deps=[liba])
self._check_node(app, "app/0.1", deps=[libb])
self._check_node(app, "app/0.1", deps=[libb, liba])


def test_mixed_user_channel():
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
import pytest

from conans.model.version_range import VersionRange

values = [
# single lower limits bounds
['>1.0', ">1.0", ">1.0"],
['>=1.0', ">1.0", ">1.0"],
['>1.0', ">1.1", ">1.1"],
['>1.0', ">=1.1", ">=1.1-"],
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
['>=1.0', ">=1.1", ">=1.1-"],
# single upper limits bounds
['<2.0', "<2.0", "<2.0-"],
['<=1.0', "<1.0", "<1.0-"],
['<2.0', "<2.1", "<2.0-"],
['<2.0', "<=1.1", "<=1.1"],
['<=1.0', "<=1.1", "<=1.0"],
# One lower limit, one upper
['>=1.0', "<2.0", ">=1.0- <2.0-"],
# Two lower, one upper
['>=1.0', ">1.0 <2.0", ">1.0 <2.0-"],
['>=1.0', ">1.1 <2.0", ">1.1 <2.0-"],
['>1.0', ">1.1 <=2.0", ">1.1 <=2.0"],
['>1.0', ">=1.1 <=2.0", ">=1.1- <=2.0"],
# one lower, two upper
['<3.0', ">1.0 <2.0", ">1.0 <2.0-"],
['<=2.0', ">1.1 <2.0", ">1.1 <2.0-"],
['<1.9', ">1.1 <=2.0", ">1.1 <1.9-"],
['<=1.9', ">=1.1 <=2.0", ">=1.1- <=1.9"],
# two lower, two upper
['>0.1 <3.0', ">1.0 <2.0", ">1.0 <2.0-"],
['>1.2 <=2.0', ">1.1 <2.0", ">1.2 <2.0-"],
['>0.1 <1.9', ">1.1 <=2.0", ">1.1 <1.9-"],
['>=1.3 <=1.9', ">=1.1 <=2.0", ">=1.3- <=1.9"],
['>=1.0 <=5.0', ">2 <2.5", ">2 <2.5-"],
# equal limits
['>=1.0 <3.0', ">0.0 <=1.0", ">=1.0- <=1.0"],
# prereleases
['>=1.0- <3.0', ">=1.0 <3.0-", ">=1.0- <3.0-"],
# OR
['>=1.0 <2.0 || >=2.1 <3', ">=2.3", ">=2.3- <3-"],
['>=1.3 <=1.9 || >2.1', ">=1.1 <=2.0 || >=2.1 <2.6", ">=1.3- <=1.9 || >2.1 <2.6-"],
['>=1.3 <=1.9 || >=2.2', ">=1.8 <2.3 || >=2.1 <2.6", ">=1.8- <=1.9 || >=2.2- <2.3- || >=2.2- <2.6-"],
]


@pytest.mark.parametrize("range1, range2, result", values)
def test_range_intersection(range1, range2, result):
r1 = VersionRange(range1)
r2 = VersionRange(range2)
inter = r1.intersection(r2)
result = f"[{result}]"
assert inter.version() == result
inter = r2.intersection(r1) # Test reverse order, result should be the same
assert inter.version() == result


incompatible_values = [
['>1.0', "<1.0"],
['>=1.0', "<1.0"],
['>1.0', "<=1.0"],
['>1.0 <2.0', ">2.0"],
['>1.0 <2.0', "<1.0"],
['>1.0 <2.0', ">3.0 <4.0"],
['<1.0', ">3.0 <4.0"],
]


@pytest.mark.parametrize("range1, range2", incompatible_values)
def test_range_intersection_incompatible(range1, range2):
r1 = VersionRange(range1)
r2 = VersionRange(range2)
inter = r1.intersection(r2)
assert inter is None
inter = r2.intersection(r1) # Test reverse order, result should be the same
assert inter is None