- rename check_provision() to split_revision()

- fix indentation to conform to the Python style guide
- add more tests and documentation
This commit is contained in:
Fred Drake 2005-03-21 06:36:32 +00:00
parent dd6f04630f
commit 227e8ffa20
2 changed files with 137 additions and 80 deletions

View file

@ -1181,11 +1181,7 @@ class DistributionMetadata:
value = [v.strip() for v in value] value = [v.strip() for v in value]
for v in value: for v in value:
import distutils.versionpredicate import distutils.versionpredicate
ver = distutils.versionpredicate.check_provision(v) distutils.versionpredicate.split_provision(v)
if ver:
import distutils.version
sv = distutils.version.StrictVersion()
sv.parse(ver.strip()[1:-1])
self.provides = value self.provides = value
def get_obsoletes(self): def get_obsoletes(self):

View file

@ -1,9 +1,10 @@
"""Module for parsing and testing package version predicate strings. """Module for parsing and testing package version predicate strings.
""" """
import re import re
import version import distutils.version
import operator import operator
re_validPackage = re.compile(r"(?i)^\s*([a-z_]\w*(?:\.[a-z_]\w*)*)(.*)") re_validPackage = re.compile(r"(?i)^\s*([a-z_]\w*(?:\.[a-z_]\w*)*)(.*)")
# (package) (rest) # (package) (rest)
@ -11,93 +12,153 @@ re_paren = re.compile(r"^\s*\((.*)\)\s*$") # (list) inside of parentheses
re_splitComparison = re.compile(r"^\s*(<=|>=|<|>|!=|==)\s*([^\s,]+)\s*$") re_splitComparison = re.compile(r"^\s*(<=|>=|<|>|!=|==)\s*([^\s,]+)\s*$")
# (comp) (version) # (comp) (version)
def splitUp(pred): def splitUp(pred):
"""Parse a single version comparison. """Parse a single version comparison.
Return (comparison string, StrictVersion)
""" Return (comparison string, StrictVersion)
res = re_splitComparison.match(pred) """
if not res: res = re_splitComparison.match(pred)
raise ValueError, "Bad package restriction syntax: " + pred if not res:
comp, verStr = res.groups() raise ValueError("bad package restriction syntax: %r" % pred)
return (comp, version.StrictVersion(verStr)) comp, verStr = res.groups()
return (comp, distutils.version.StrictVersion(verStr))
compmap = {"<": operator.lt, "<=": operator.le, "==": operator.eq, compmap = {"<": operator.lt, "<=": operator.le, "==": operator.eq,
">": operator.gt, ">=": operator.ge, "!=": operator.ne} ">": operator.gt, ">=": operator.ge, "!=": operator.ne}
class VersionPredicate: class VersionPredicate:
"""Parse and test package version predicates. """Parse and test package version predicates.
>>> v = VersionPredicate("pyepat.abc (>1.0, <3333.3a1, !=1555.1b3)") >>> v = VersionPredicate('pyepat.abc (>1.0, <3333.3a1, !=1555.1b3)')
>>> print v
pyepat.abc (> 1.0, < 3333.3a1, != 1555.1b3)
>>> v.satisfied_by("1.1")
True
>>> v.satisfied_by("1.4")
True
>>> v.satisfied_by("1.0")
False
>>> v.satisfied_by("4444.4")
False
>>> v.satisfied_by("1555.1b3")
False
>>> v = VersionPredicate("pat( == 0.1 ) ")
>>> v.satisfied_by("0.1")
True
>>> v.satisfied_by("0.2")
False
>>> v = VersionPredicate("p1.p2.p3.p4(>=1.0, <=1.3a1, !=1.2zb3)")
Traceback (most recent call last):
...
ValueError: invalid version number '1.2zb3'
""" The `name` attribute provides the full dotted name that is given::
def __init__(self, versionPredicateStr): >>> v.name
"""Parse a version predicate string. 'pyepat.abc'
"""
# Fields:
# name: package name
# pred: list of (comparison string, StrictVersion)
versionPredicateStr = versionPredicateStr.strip() The str() of a `VersionPredicate` provides a normalized
if not versionPredicateStr: human-readable version of the expression::
raise ValueError, "Empty package restriction"
match = re_validPackage.match(versionPredicateStr)
if not match:
raise ValueError, "Bad package name in " + versionPredicateStr
self.name, paren = match.groups()
paren = paren.strip()
if paren:
match = re_paren.match(paren)
if not match:
raise ValueError, "Expected parenthesized list: " + paren
str = match.groups()[0]
self.pred = [splitUp(aPred) for aPred in str.split(",")]
if not self.pred:
raise ValueError("Empty Parenthesized list in %r"
% versionPredicateStr )
else:
self.pred=[]
def __str__(self): >>> print v
if self.pred: pyepat.abc (> 1.0, < 3333.3a1, != 1555.1b3)
seq = [cond + " " + str(ver) for cond, ver in self.pred]
return self.name + " (" + ", ".join(seq) + ")"
else:
return self.name
def satisfied_by(self, version): The `satisfied_by()` method can be used to determine with a given
"""True if version is compatible with all the predicates in self. version number is included in the set described by the version
The parameter version must be acceptable to the StrictVersion restrictions::
constructor. It may be either a string or StrictVersion.
""" >>> v.satisfied_by('1.1')
for cond, ver in self.pred: True
if not compmap[cond](version, ver): >>> v.satisfied_by('1.4')
return False True
return True >>> v.satisfied_by('1.0')
False
>>> v.satisfied_by('4444.4')
False
>>> v.satisfied_by('1555.1b3')
False
`VersionPredicate` is flexible in accepting extra whitespace::
>>> v = VersionPredicate(' pat( == 0.1 ) ')
>>> v.name
'pat'
>>> v.satisfied_by('0.1')
True
>>> v.satisfied_by('0.2')
False
If any version numbers passed in do not conform to the
restrictions of `StrictVersion`, a `ValueError` is raised::
>>> v = VersionPredicate('p1.p2.p3.p4(>=1.0, <=1.3a1, !=1.2zb3)')
Traceback (most recent call last):
...
ValueError: invalid version number '1.2zb3'
It the module or package name given does not conform to what's
allowed as a legal module or package name, `ValueError` is
raised::
>>> v = VersionPredicate('foo-bar')
Traceback (most recent call last):
...
ValueError: expected parenthesized list: '-bar'
>>> v = VersionPredicate('foo bar (12.21)')
Traceback (most recent call last):
...
ValueError: expected parenthesized list: 'bar (12.21)'
"""
def __init__(self, versionPredicateStr):
"""Parse a version predicate string.
"""
# Fields:
# name: package name
# pred: list of (comparison string, StrictVersion)
versionPredicateStr = versionPredicateStr.strip()
if not versionPredicateStr:
raise ValueError("empty package restriction")
match = re_validPackage.match(versionPredicateStr)
if not match:
raise ValueError("bad package name in %r" % versionPredicateStr)
self.name, paren = match.groups()
paren = paren.strip()
if paren:
match = re_paren.match(paren)
if not match:
raise ValueError("expected parenthesized list: %r" % paren)
str = match.groups()[0]
self.pred = [splitUp(aPred) for aPred in str.split(",")]
if not self.pred:
raise ValueError("empty parenthesized list in %r"
% versionPredicateStr)
else:
self.pred=[]
def __str__(self):
if self.pred:
seq = [cond + " " + str(ver) for cond, ver in self.pred]
return self.name + " (" + ", ".join(seq) + ")"
else:
return self.name
def satisfied_by(self, version):
"""True if version is compatible with all the predicates in self.
The parameter version must be acceptable to the StrictVersion
constructor. It may be either a string or StrictVersion.
"""
for cond, ver in self.pred:
if not compmap[cond](version, ver):
return False
return True
def check_provision(value): _provision_rx = None
m = re.match("[a-zA-Z_]\w*(\.[a-zA-Z_]\w*)*(\s*\([^)]+\))?$", value)
def split_provision(value):
"""Return the name and optional version number of a provision.
The version number, if given, will be returned as a `StrictVersion`
instance, otherwise it will be `None`.
>>> split_provision('mypkg')
('mypkg', None)
>>> split_provision(' mypkg( 1.2 ) ')
('mypkg', StrictVersion ('1.2'))
"""
global _provision_rx
if _provision_rx is None:
_provision_rx = re.compile(
"([a-zA-Z_]\w*(?:\.[a-zA-Z_]\w*)*)(?:\s*\(\s*([^)\s]+)\s*\))?$")
value = value.strip()
m = _provision_rx.match(value)
if not m: if not m:
raise ValueError("illegal provides specification: %r" % value) raise ValueError("illegal provides specification: %r" % value)
ver = m.group(2) or None
if ver:
ver = distutils.version.StrictVersion(ver)
return m.group(1), ver