Kaydet (Commit) 227e8ffa authored tarafından Fred Drake's avatar Fred Drake

- rename check_provision() to split_revision()

- fix indentation to conform to the Python style guide
- add more tests and documentation
üst dd6f0463
...@@ -1181,11 +1181,7 @@ class DistributionMetadata: ...@@ -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):
......
"""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,15 +12,17 @@ re_paren = re.compile(r"^\s*\((.*)\)\s*$") # (list) inside of parentheses ...@@ -11,15 +12,17 @@ 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) res = re_splitComparison.match(pred)
if not res: if not res:
raise ValueError, "Bad package restriction syntax: " + pred raise ValueError("bad package restriction syntax: %r" % pred)
comp, verStr = res.groups() comp, verStr = res.groups()
return (comp, version.StrictVersion(verStr)) 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}
...@@ -27,29 +30,66 @@ compmap = {"<": operator.lt, "<=": operator.le, "==": operator.eq, ...@@ -27,29 +30,66 @@ compmap = {"<": operator.lt, "<=": operator.le, "==": operator.eq,
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)')
The `name` attribute provides the full dotted name that is given::
>>> v.name
'pyepat.abc'
The str() of a `VersionPredicate` provides a normalized
human-readable version of the expression::
>>> print v >>> print v
pyepat.abc (> 1.0, < 3333.3a1, != 1555.1b3) pyepat.abc (> 1.0, < 3333.3a1, != 1555.1b3)
>>> v.satisfied_by("1.1")
The `satisfied_by()` method can be used to determine with a given
version number is included in the set described by the version
restrictions::
>>> v.satisfied_by('1.1')
True True
>>> v.satisfied_by("1.4") >>> v.satisfied_by('1.4')
True True
>>> v.satisfied_by("1.0") >>> v.satisfied_by('1.0')
False False
>>> v.satisfied_by("4444.4") >>> v.satisfied_by('4444.4')
False False
>>> v.satisfied_by("1555.1b3") >>> v.satisfied_by('1555.1b3')
False False
>>> v = VersionPredicate("pat( == 0.1 ) ")
>>> v.satisfied_by("0.1") `VersionPredicate` is flexible in accepting extra whitespace::
>>> v = VersionPredicate(' pat( == 0.1 ) ')
>>> v.name
'pat'
>>> v.satisfied_by('0.1')
True True
>>> v.satisfied_by("0.2") >>> v.satisfied_by('0.2')
False False
>>> v = VersionPredicate("p1.p2.p3.p4(>=1.0, <=1.3a1, !=1.2zb3)")
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): Traceback (most recent call last):
... ...
ValueError: invalid version number '1.2zb3' 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): def __init__(self, versionPredicateStr):
...@@ -61,21 +101,21 @@ class VersionPredicate: ...@@ -61,21 +101,21 @@ class VersionPredicate:
versionPredicateStr = versionPredicateStr.strip() versionPredicateStr = versionPredicateStr.strip()
if not versionPredicateStr: if not versionPredicateStr:
raise ValueError, "Empty package restriction" raise ValueError("empty package restriction")
match = re_validPackage.match(versionPredicateStr) match = re_validPackage.match(versionPredicateStr)
if not match: if not match:
raise ValueError, "Bad package name in " + versionPredicateStr raise ValueError("bad package name in %r" % versionPredicateStr)
self.name, paren = match.groups() self.name, paren = match.groups()
paren = paren.strip() paren = paren.strip()
if paren: if paren:
match = re_paren.match(paren) match = re_paren.match(paren)
if not match: if not match:
raise ValueError, "Expected parenthesized list: " + paren raise ValueError("expected parenthesized list: %r" % paren)
str = match.groups()[0] str = match.groups()[0]
self.pred = [splitUp(aPred) for aPred in str.split(",")] self.pred = [splitUp(aPred) for aPred in str.split(",")]
if not self.pred: if not self.pred:
raise ValueError("Empty Parenthesized list in %r" raise ValueError("empty parenthesized list in %r"
% versionPredicateStr ) % versionPredicateStr)
else: else:
self.pred=[] self.pred=[]
...@@ -97,7 +137,28 @@ class VersionPredicate: ...@@ -97,7 +137,28 @@ class VersionPredicate:
return True 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
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment