More comments + stable & defined op sorting order
Add more comments to multipip explaining how the ranking/scoring function works, and rename the logger to denote that it's really a constant global value and sort the operators (when versions are found to be the same) in a explictly defined order (instead of the more implicit sort order previously). Change-Id: I948ee78367facbfcd1c593b1b0844ca5230a8bd5
This commit is contained in:
parent
d3e2377269
commit
efcd5845fb
@ -16,9 +16,14 @@ import pip.req
|
|||||||
import pkg_resources
|
import pkg_resources
|
||||||
|
|
||||||
|
|
||||||
|
# Use this for the sorting order of operators
|
||||||
|
OP_ORDER = ('!=', '==', '<', '<=', '>', '>=')
|
||||||
|
|
||||||
|
# Exit codes that are returned on various issues.
|
||||||
BAD_REQUIREMENTS = 2
|
BAD_REQUIREMENTS = 2
|
||||||
INCOMPATIBLE_REQUIREMENTS = 3
|
INCOMPATIBLE_REQUIREMENTS = 3
|
||||||
logger = logging.getLogger()
|
|
||||||
|
LOGGER = logging.getLogger()
|
||||||
|
|
||||||
|
|
||||||
class RequirementException(Exception):
|
class RequirementException(Exception):
|
||||||
@ -69,8 +74,8 @@ def create_parser():
|
|||||||
def setup_logging(options):
|
def setup_logging(options):
|
||||||
level = logging.DEBUG if options.debug else logging.WARNING
|
level = logging.DEBUG if options.debug else logging.WARNING
|
||||||
handler = logging.StreamHandler(sys.stderr)
|
handler = logging.StreamHandler(sys.stderr)
|
||||||
logger.addHandler(handler)
|
LOGGER.addHandler(handler)
|
||||||
logger.setLevel(level)
|
LOGGER.setLevel(level)
|
||||||
|
|
||||||
|
|
||||||
def install_requirement_ensure_req_field(req):
|
def install_requirement_ensure_req_field(req):
|
||||||
@ -114,35 +119,54 @@ def iter_combinations(elements, include_empty=False):
|
|||||||
|
|
||||||
|
|
||||||
def conflict_scorer(versioned):
|
def conflict_scorer(versioned):
|
||||||
"""Scores a list of (op, version) tuples, a higher score means more
|
"""Scores a list of (op, version) tuples, a higher score means more likely
|
||||||
conflicts while a lower score means less conflicts.
|
to cause conflicts while a lower score means less likely to cause
|
||||||
|
conflicts. A zero score means that no conflicts have been detected (aka
|
||||||
|
when installing no version issues will be encountered).
|
||||||
"""
|
"""
|
||||||
if len(versioned) == 1:
|
if len(versioned) == 1:
|
||||||
|
# A single list has no capability to conflict with anything.
|
||||||
return 0
|
return 0
|
||||||
|
# Group by operator (and the versions that those operators are compatible
|
||||||
|
# with).
|
||||||
op_versions = collections.defaultdict(list)
|
op_versions = collections.defaultdict(list)
|
||||||
for (op, version) in versioned:
|
for (op, version) in versioned:
|
||||||
op_versions[op].append(version)
|
op_versions[op].append(version)
|
||||||
score = 0
|
score = 0
|
||||||
for version in sorted(op_versions.get("==", [])):
|
for version in sorted(op_versions.get("==", [])):
|
||||||
for (op, version2) in versioned:
|
for (op, version2) in versioned:
|
||||||
|
# Any request for something not this version is a conflict.
|
||||||
if version != version2:
|
if version != version2:
|
||||||
score += 1
|
score += 1
|
||||||
|
# Any request for something is this version, but isn't '=='
|
||||||
|
# is also a conflict.
|
||||||
if version == version2 and op != "==":
|
if version == version2 and op != "==":
|
||||||
score += 1
|
score += 1
|
||||||
for version in sorted(op_versions.get("!=", [])):
|
for version in sorted(op_versions.get("!=", [])):
|
||||||
for (op, version2) in versioned:
|
for (op, version2) in versioned:
|
||||||
if op in ["!=", ">", "<"]:
|
if op in ["!=", ">", "<"]:
|
||||||
continue
|
continue
|
||||||
|
# Anything that is includes this version would be a conflict,
|
||||||
|
# thats why we exclude !=, <, and > from the above since
|
||||||
|
# those exclude versions.
|
||||||
if version2 == version:
|
if version2 == version:
|
||||||
score += 1
|
score += 1
|
||||||
for version in sorted(op_versions.get(">", [])):
|
for version in sorted(op_versions.get(">", [])):
|
||||||
for (op, version2) in versioned:
|
for (op, version2) in versioned:
|
||||||
if (op, version2) == (">", version):
|
if (op, version2) == (">", version):
|
||||||
continue
|
continue
|
||||||
|
# A request for a lower version than the desired greater than
|
||||||
|
# version is a conflict.
|
||||||
if op in ["<", "<="] and version2 <= version:
|
if op in ["<", "<="] and version2 <= version:
|
||||||
score += 1
|
score += 1
|
||||||
|
# A request for an inclusive version and matching with this
|
||||||
|
# version is also a conflict (since both can not be satisfied).
|
||||||
elif op in ["==", ">="] and version2 == version:
|
elif op in ["==", ">="] and version2 == version:
|
||||||
score += 1
|
score += 1
|
||||||
|
# If another request asks for a version less than this version but
|
||||||
|
# also is asking for a greater than operator, that version spans
|
||||||
|
# a wider range of compatible versions and therefore we are in
|
||||||
|
# more of a conflict than that version.
|
||||||
elif op == ">" and version2 < version:
|
elif op == ">" and version2 < version:
|
||||||
score += 1
|
score += 1
|
||||||
elif op == ">=" and version2 <= version:
|
elif op == ">=" and version2 <= version:
|
||||||
@ -244,11 +268,13 @@ def best_match(req_key, req_list):
|
|||||||
return (req_list, [])
|
return (req_list, [])
|
||||||
|
|
||||||
def spec_sort(spec1, spec2):
|
def spec_sort(spec1, spec2):
|
||||||
|
# Ensure there is always a well defined spec ordering so that the
|
||||||
|
# selection of matched specs is also well defined.
|
||||||
(op1, version1) = spec1
|
(op1, version1) = spec1
|
||||||
(op2, version2) = spec2
|
(op2, version2) = spec2
|
||||||
c = cmp(version1, version2)
|
c = cmp(version1, version2)
|
||||||
if c == 0:
|
if c == 0:
|
||||||
c = cmp(op1, op2)
|
c = cmp(OP_ORDER.index(op1), OP_ORDER.index(op2))
|
||||||
return c
|
return c
|
||||||
|
|
||||||
def reform(specs, versions):
|
def reform(specs, versions):
|
||||||
@ -405,7 +431,7 @@ def main():
|
|||||||
try:
|
try:
|
||||||
requirements, ignored_requirements = parse_requirements(options)
|
requirements, ignored_requirements = parse_requirements(options)
|
||||||
except RequirementException as ex:
|
except RequirementException as ex:
|
||||||
logger.error("Requirement failure: %s", ex)
|
LOGGER.error("Requirement failure: %s", ex)
|
||||||
sys.exit(BAD_REQUIREMENTS)
|
sys.exit(BAD_REQUIREMENTS)
|
||||||
else:
|
else:
|
||||||
joined_requirements, incompatibles = join_requirements(requirements,
|
joined_requirements, incompatibles = join_requirements(requirements,
|
||||||
|
Loading…
Reference in New Issue
Block a user