| | |
| | |
| | |
| | |
| | |
| | """ |
| | Implementation of a flexible versioning scheme providing support for PEP-440, |
| | setuptools-compatible and semantic versioning. |
| | """ |
| |
|
| | import logging |
| | import re |
| |
|
| | from .compat import string_types |
| | from .util import parse_requirement |
| |
|
| | __all__ = ['NormalizedVersion', 'NormalizedMatcher', |
| | 'LegacyVersion', 'LegacyMatcher', |
| | 'SemanticVersion', 'SemanticMatcher', |
| | 'UnsupportedVersionError', 'get_scheme'] |
| |
|
| | logger = logging.getLogger(__name__) |
| |
|
| |
|
| | class UnsupportedVersionError(ValueError): |
| | """This is an unsupported version.""" |
| | pass |
| |
|
| |
|
| | class Version(object): |
| | def __init__(self, s): |
| | self._string = s = s.strip() |
| | self._parts = parts = self.parse(s) |
| | assert isinstance(parts, tuple) |
| | assert len(parts) > 0 |
| |
|
| | def parse(self, s): |
| | raise NotImplementedError('please implement in a subclass') |
| |
|
| | def _check_compatible(self, other): |
| | if type(self) != type(other): |
| | raise TypeError('cannot compare %r and %r' % (self, other)) |
| |
|
| | def __eq__(self, other): |
| | self._check_compatible(other) |
| | return self._parts == other._parts |
| |
|
| | def __ne__(self, other): |
| | return not self.__eq__(other) |
| |
|
| | def __lt__(self, other): |
| | self._check_compatible(other) |
| | return self._parts < other._parts |
| |
|
| | def __gt__(self, other): |
| | return not (self.__lt__(other) or self.__eq__(other)) |
| |
|
| | def __le__(self, other): |
| | return self.__lt__(other) or self.__eq__(other) |
| |
|
| | def __ge__(self, other): |
| | return self.__gt__(other) or self.__eq__(other) |
| |
|
| | |
| | def __hash__(self): |
| | return hash(self._parts) |
| |
|
| | def __repr__(self): |
| | return "%s('%s')" % (self.__class__.__name__, self._string) |
| |
|
| | def __str__(self): |
| | return self._string |
| |
|
| | @property |
| | def is_prerelease(self): |
| | raise NotImplementedError('Please implement in subclasses.') |
| |
|
| |
|
| | class Matcher(object): |
| | version_class = None |
| |
|
| | |
| | _operators = { |
| | '<': lambda v, c, p: v < c, |
| | '>': lambda v, c, p: v > c, |
| | '<=': lambda v, c, p: v == c or v < c, |
| | '>=': lambda v, c, p: v == c or v > c, |
| | '==': lambda v, c, p: v == c, |
| | '===': lambda v, c, p: v == c, |
| | |
| | '~=': lambda v, c, p: v == c or v > c, |
| | '!=': lambda v, c, p: v != c, |
| | } |
| |
|
| | |
| | |
| | def parse_requirement(self, s): |
| | return parse_requirement(s) |
| |
|
| | def __init__(self, s): |
| | if self.version_class is None: |
| | raise ValueError('Please specify a version class') |
| | self._string = s = s.strip() |
| | r = self.parse_requirement(s) |
| | if not r: |
| | raise ValueError('Not valid: %r' % s) |
| | self.name = r.name |
| | self.key = self.name.lower() |
| | clist = [] |
| | if r.constraints: |
| | |
| | for op, s in r.constraints: |
| | if s.endswith('.*'): |
| | if op not in ('==', '!='): |
| | raise ValueError('\'.*\' not allowed for ' |
| | '%r constraints' % op) |
| | |
| | |
| | vn, prefix = s[:-2], True |
| | |
| | self.version_class(vn) |
| | else: |
| | |
| | |
| | vn, prefix = self.version_class(s), False |
| | clist.append((op, vn, prefix)) |
| | self._parts = tuple(clist) |
| |
|
| | def match(self, version): |
| | """ |
| | Check if the provided version matches the constraints. |
| | |
| | :param version: The version to match against this instance. |
| | :type version: String or :class:`Version` instance. |
| | """ |
| | if isinstance(version, string_types): |
| | version = self.version_class(version) |
| | for operator, constraint, prefix in self._parts: |
| | f = self._operators.get(operator) |
| | if isinstance(f, string_types): |
| | f = getattr(self, f) |
| | if not f: |
| | msg = ('%r not implemented ' |
| | 'for %s' % (operator, self.__class__.__name__)) |
| | raise NotImplementedError(msg) |
| | if not f(version, constraint, prefix): |
| | return False |
| | return True |
| |
|
| | @property |
| | def exact_version(self): |
| | result = None |
| | if len(self._parts) == 1 and self._parts[0][0] in ('==', '==='): |
| | result = self._parts[0][1] |
| | return result |
| |
|
| | def _check_compatible(self, other): |
| | if type(self) != type(other) or self.name != other.name: |
| | raise TypeError('cannot compare %s and %s' % (self, other)) |
| |
|
| | def __eq__(self, other): |
| | self._check_compatible(other) |
| | return self.key == other.key and self._parts == other._parts |
| |
|
| | def __ne__(self, other): |
| | return not self.__eq__(other) |
| |
|
| | |
| | def __hash__(self): |
| | return hash(self.key) + hash(self._parts) |
| |
|
| | def __repr__(self): |
| | return "%s(%r)" % (self.__class__.__name__, self._string) |
| |
|
| | def __str__(self): |
| | return self._string |
| |
|
| |
|
| | PEP440_VERSION_RE = re.compile(r'^v?(\d+!)?(\d+(\.\d+)*)((a|alpha|b|beta|c|rc|pre|preview)(\d+)?)?' |
| | r'(\.(post|r|rev)(\d+)?)?([._-]?(dev)(\d+)?)?' |
| | r'(\+([a-zA-Z\d]+(\.[a-zA-Z\d]+)?))?$', re.I) |
| |
|
| |
|
| | def _pep_440_key(s): |
| | s = s.strip() |
| | m = PEP440_VERSION_RE.match(s) |
| | if not m: |
| | raise UnsupportedVersionError('Not a valid version: %s' % s) |
| | groups = m.groups() |
| | nums = tuple(int(v) for v in groups[1].split('.')) |
| | while len(nums) > 1 and nums[-1] == 0: |
| | nums = nums[:-1] |
| |
|
| | if not groups[0]: |
| | epoch = 0 |
| | else: |
| | epoch = int(groups[0][:-1]) |
| | pre = groups[4:6] |
| | post = groups[7:9] |
| | dev = groups[10:12] |
| | local = groups[13] |
| | if pre == (None, None): |
| | pre = () |
| | else: |
| | if pre[1] is None: |
| | pre = pre[0], 0 |
| | else: |
| | pre = pre[0], int(pre[1]) |
| | if post == (None, None): |
| | post = () |
| | else: |
| | if post[1] is None: |
| | post = post[0], 0 |
| | else: |
| | post = post[0], int(post[1]) |
| | if dev == (None, None): |
| | dev = () |
| | else: |
| | if dev[1] is None: |
| | dev = dev[0], 0 |
| | else: |
| | dev = dev[0], int(dev[1]) |
| | if local is None: |
| | local = () |
| | else: |
| | parts = [] |
| | for part in local.split('.'): |
| | |
| | |
| | |
| | if part.isdigit(): |
| | part = (1, int(part)) |
| | else: |
| | part = (0, part) |
| | parts.append(part) |
| | local = tuple(parts) |
| | if not pre: |
| | |
| | if not post and dev: |
| | |
| | pre = ('a', -1) |
| | else: |
| | pre = ('z',) |
| | |
| | if not post: |
| | post = ('_',) |
| | if not dev: |
| | dev = ('final',) |
| |
|
| | return epoch, nums, pre, post, dev, local |
| |
|
| |
|
| | _normalized_key = _pep_440_key |
| |
|
| |
|
| | class NormalizedVersion(Version): |
| | """A rational version. |
| | |
| | Good: |
| | 1.2 # equivalent to "1.2.0" |
| | 1.2.0 |
| | 1.2a1 |
| | 1.2.3a2 |
| | 1.2.3b1 |
| | 1.2.3c1 |
| | 1.2.3.4 |
| | TODO: fill this out |
| | |
| | Bad: |
| | 1 # minimum two numbers |
| | 1.2a # release level must have a release serial |
| | 1.2.3b |
| | """ |
| | def parse(self, s): |
| | result = _normalized_key(s) |
| | |
| | |
| | |
| | |
| | m = PEP440_VERSION_RE.match(s) |
| | groups = m.groups() |
| | self._release_clause = tuple(int(v) for v in groups[1].split('.')) |
| | return result |
| |
|
| | PREREL_TAGS = set(['a', 'b', 'c', 'rc', 'dev']) |
| |
|
| | @property |
| | def is_prerelease(self): |
| | return any(t[0] in self.PREREL_TAGS for t in self._parts if t) |
| |
|
| |
|
| | def _match_prefix(x, y): |
| | x = str(x) |
| | y = str(y) |
| | if x == y: |
| | return True |
| | if not x.startswith(y): |
| | return False |
| | n = len(y) |
| | return x[n] == '.' |
| |
|
| |
|
| | class NormalizedMatcher(Matcher): |
| | version_class = NormalizedVersion |
| |
|
| | |
| | _operators = { |
| | '~=': '_match_compatible', |
| | '<': '_match_lt', |
| | '>': '_match_gt', |
| | '<=': '_match_le', |
| | '>=': '_match_ge', |
| | '==': '_match_eq', |
| | '===': '_match_arbitrary', |
| | '!=': '_match_ne', |
| | } |
| |
|
| | def _adjust_local(self, version, constraint, prefix): |
| | if prefix: |
| | strip_local = '+' not in constraint and version._parts[-1] |
| | else: |
| | |
| | |
| | |
| | |
| | strip_local = not constraint._parts[-1] and version._parts[-1] |
| | if strip_local: |
| | s = version._string.split('+', 1)[0] |
| | version = self.version_class(s) |
| | return version, constraint |
| |
|
| | def _match_lt(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | if version >= constraint: |
| | return False |
| | release_clause = constraint._release_clause |
| | pfx = '.'.join([str(i) for i in release_clause]) |
| | return not _match_prefix(version, pfx) |
| |
|
| | def _match_gt(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | if version <= constraint: |
| | return False |
| | release_clause = constraint._release_clause |
| | pfx = '.'.join([str(i) for i in release_clause]) |
| | return not _match_prefix(version, pfx) |
| |
|
| | def _match_le(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | return version <= constraint |
| |
|
| | def _match_ge(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | return version >= constraint |
| |
|
| | def _match_eq(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | if not prefix: |
| | result = (version == constraint) |
| | else: |
| | result = _match_prefix(version, constraint) |
| | return result |
| |
|
| | def _match_arbitrary(self, version, constraint, prefix): |
| | return str(version) == str(constraint) |
| |
|
| | def _match_ne(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | if not prefix: |
| | result = (version != constraint) |
| | else: |
| | result = not _match_prefix(version, constraint) |
| | return result |
| |
|
| | def _match_compatible(self, version, constraint, prefix): |
| | version, constraint = self._adjust_local(version, constraint, prefix) |
| | if version == constraint: |
| | return True |
| | if version < constraint: |
| | return False |
| | |
| | |
| | release_clause = constraint._release_clause |
| | if len(release_clause) > 1: |
| | release_clause = release_clause[:-1] |
| | pfx = '.'.join([str(i) for i in release_clause]) |
| | return _match_prefix(version, pfx) |
| |
|
| |
|
| | _REPLACEMENTS = ( |
| | (re.compile('[.+-]$'), ''), |
| | (re.compile(r'^[.](\d)'), r'0.\1'), |
| | (re.compile('^[.-]'), ''), |
| | (re.compile(r'^\((.*)\)$'), r'\1'), |
| | (re.compile(r'^v(ersion)?\s*(\d+)'), r'\2'), |
| | (re.compile(r'^r(ev)?\s*(\d+)'), r'\2'), |
| | (re.compile('[.]{2,}'), '.'), |
| | (re.compile(r'\b(alfa|apha)\b'), 'alpha'), |
| | (re.compile(r'\b(pre-alpha|prealpha)\b'), |
| | 'pre.alpha'), |
| | (re.compile(r'\(beta\)$'), 'beta'), |
| | ) |
| |
|
| | _SUFFIX_REPLACEMENTS = ( |
| | (re.compile('^[:~._+-]+'), ''), |
| | (re.compile('[,*")([\\]]'), ''), |
| | (re.compile('[~:+_ -]'), '.'), |
| | (re.compile('[.]{2,}'), '.'), |
| | (re.compile(r'\.$'), ''), |
| | ) |
| |
|
| | _NUMERIC_PREFIX = re.compile(r'(\d+(\.\d+)*)') |
| |
|
| |
|
| | def _suggest_semantic_version(s): |
| | """ |
| | Try to suggest a semantic form for a version for which |
| | _suggest_normalized_version couldn't come up with anything. |
| | """ |
| | result = s.strip().lower() |
| | for pat, repl in _REPLACEMENTS: |
| | result = pat.sub(repl, result) |
| | if not result: |
| | result = '0.0.0' |
| |
|
| | |
| | |
| | |
| | m = _NUMERIC_PREFIX.match(result) |
| | if not m: |
| | prefix = '0.0.0' |
| | suffix = result |
| | else: |
| | prefix = m.groups()[0].split('.') |
| | prefix = [int(i) for i in prefix] |
| | while len(prefix) < 3: |
| | prefix.append(0) |
| | if len(prefix) == 3: |
| | suffix = result[m.end():] |
| | else: |
| | suffix = '.'.join([str(i) for i in prefix[3:]]) + result[m.end():] |
| | prefix = prefix[:3] |
| | prefix = '.'.join([str(i) for i in prefix]) |
| | suffix = suffix.strip() |
| | if suffix: |
| | |
| | |
| | for pat, repl in _SUFFIX_REPLACEMENTS: |
| | suffix = pat.sub(repl, suffix) |
| |
|
| | if not suffix: |
| | result = prefix |
| | else: |
| | sep = '-' if 'dev' in suffix else '+' |
| | result = prefix + sep + suffix |
| | if not is_semver(result): |
| | result = None |
| | return result |
| |
|
| |
|
| | def _suggest_normalized_version(s): |
| | """Suggest a normalized version close to the given version string. |
| | |
| | If you have a version string that isn't rational (i.e. NormalizedVersion |
| | doesn't like it) then you might be able to get an equivalent (or close) |
| | rational version from this function. |
| | |
| | This does a number of simple normalizations to the given string, based |
| | on observation of versions currently in use on PyPI. Given a dump of |
| | those version during PyCon 2009, 4287 of them: |
| | - 2312 (53.93%) match NormalizedVersion without change |
| | with the automatic suggestion |
| | - 3474 (81.04%) match when using this suggestion method |
| | |
| | @param s {str} An irrational version string. |
| | @returns A rational version string, or None, if couldn't determine one. |
| | """ |
| | try: |
| | _normalized_key(s) |
| | return s |
| | except UnsupportedVersionError: |
| | pass |
| |
|
| | rs = s.lower() |
| |
|
| | |
| | for orig, repl in (('-alpha', 'a'), ('-beta', 'b'), ('alpha', 'a'), |
| | ('beta', 'b'), ('rc', 'c'), ('-final', ''), |
| | ('-pre', 'c'), |
| | ('-release', ''), ('.release', ''), ('-stable', ''), |
| | ('+', '.'), ('_', '.'), (' ', ''), ('.final', ''), |
| | ('final', '')): |
| | rs = rs.replace(orig, repl) |
| |
|
| | |
| | rs = re.sub(r"pre$", r"pre0", rs) |
| | rs = re.sub(r"dev$", r"dev0", rs) |
| |
|
| | |
| | |
| | |
| | rs = re.sub(r"([abc]|rc)[\-\.](\d+)$", r"\1\2", rs) |
| |
|
| | |
| | |
| | rs = re.sub(r"[\-\.](dev)[\-\.]?r?(\d+)$", r".\1\2", rs) |
| |
|
| | |
| | rs = re.sub(r"[.~]?([abc])\.?", r"\1", rs) |
| |
|
| | |
| | if rs.startswith('v'): |
| | rs = rs[1:] |
| |
|
| | |
| | |
| | |
| | rs = re.sub(r"\b0+(\d+)(?!\d)", r"\1", rs) |
| |
|
| | |
| | |
| | |
| | rs = re.sub(r"(\d+[abc])$", r"\g<1>0", rs) |
| |
|
| | |
| | rs = re.sub(r"\.?(dev-r|dev\.r)\.?(\d+)$", r".dev\2", rs) |
| |
|
| | |
| | rs = re.sub(r"-(a|b|c)(\d+)$", r"\1\2", rs) |
| |
|
| | |
| | rs = re.sub(r"[\.\-](dev|devel)$", r".dev0", rs) |
| |
|
| | |
| | rs = re.sub(r"(?![\.\-])dev$", r".dev0", rs) |
| |
|
| | |
| | rs = re.sub(r"(final|stable)$", "", rs) |
| |
|
| | |
| | |
| | |
| | |
| | rs = re.sub(r"\.?(r|-|-r)\.?(\d+)$", r".post\2", rs) |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | rs = re.sub(r"\.?(dev|git|bzr)\.?(\d+)$", r".dev\2", rs) |
| |
|
| | |
| | |
| | |
| | |
| | |
| | rs = re.sub(r"\.?(pre|preview|-c)(\d+)$", r"c\g<2>", rs) |
| |
|
| | |
| | rs = re.sub(r"p(\d+)$", r".post\1", rs) |
| |
|
| | try: |
| | _normalized_key(rs) |
| | except UnsupportedVersionError: |
| | rs = None |
| | return rs |
| |
|
| | |
| | |
| | |
| |
|
| |
|
| | _VERSION_PART = re.compile(r'([a-z]+|\d+|[\.-])', re.I) |
| | _VERSION_REPLACE = { |
| | 'pre': 'c', |
| | 'preview': 'c', |
| | '-': 'final-', |
| | 'rc': 'c', |
| | 'dev': '@', |
| | '': None, |
| | '.': None, |
| | } |
| |
|
| |
|
| | def _legacy_key(s): |
| | def get_parts(s): |
| | result = [] |
| | for p in _VERSION_PART.split(s.lower()): |
| | p = _VERSION_REPLACE.get(p, p) |
| | if p: |
| | if '0' <= p[:1] <= '9': |
| | p = p.zfill(8) |
| | else: |
| | p = '*' + p |
| | result.append(p) |
| | result.append('*final') |
| | return result |
| |
|
| | result = [] |
| | for p in get_parts(s): |
| | if p.startswith('*'): |
| | if p < '*final': |
| | while result and result[-1] == '*final-': |
| | result.pop() |
| | while result and result[-1] == '00000000': |
| | result.pop() |
| | result.append(p) |
| | return tuple(result) |
| |
|
| |
|
| | class LegacyVersion(Version): |
| | def parse(self, s): |
| | return _legacy_key(s) |
| |
|
| | @property |
| | def is_prerelease(self): |
| | result = False |
| | for x in self._parts: |
| | if (isinstance(x, string_types) and x.startswith('*') and x < '*final'): |
| | result = True |
| | break |
| | return result |
| |
|
| |
|
| | class LegacyMatcher(Matcher): |
| | version_class = LegacyVersion |
| |
|
| | _operators = dict(Matcher._operators) |
| | _operators['~='] = '_match_compatible' |
| |
|
| | numeric_re = re.compile(r'^(\d+(\.\d+)*)') |
| |
|
| | def _match_compatible(self, version, constraint, prefix): |
| | if version < constraint: |
| | return False |
| | m = self.numeric_re.match(str(constraint)) |
| | if not m: |
| | logger.warning('Cannot compute compatible match for version %s ' |
| | ' and constraint %s', version, constraint) |
| | return True |
| | s = m.groups()[0] |
| | if '.' in s: |
| | s = s.rsplit('.', 1)[0] |
| | return _match_prefix(version, s) |
| |
|
| | |
| | |
| | |
| |
|
| |
|
| | _SEMVER_RE = re.compile(r'^(\d+)\.(\d+)\.(\d+)' |
| | r'(-[a-z0-9]+(\.[a-z0-9-]+)*)?' |
| | r'(\+[a-z0-9]+(\.[a-z0-9-]+)*)?$', re.I) |
| |
|
| |
|
| | def is_semver(s): |
| | return _SEMVER_RE.match(s) |
| |
|
| |
|
| | def _semantic_key(s): |
| | def make_tuple(s, absent): |
| | if s is None: |
| | result = (absent,) |
| | else: |
| | parts = s[1:].split('.') |
| | |
| | |
| | result = tuple([p.zfill(8) if p.isdigit() else p for p in parts]) |
| | return result |
| |
|
| | m = is_semver(s) |
| | if not m: |
| | raise UnsupportedVersionError(s) |
| | groups = m.groups() |
| | major, minor, patch = [int(i) for i in groups[:3]] |
| | |
| | pre, build = make_tuple(groups[3], '|'), make_tuple(groups[5], '*') |
| | return (major, minor, patch), pre, build |
| |
|
| |
|
| | class SemanticVersion(Version): |
| | def parse(self, s): |
| | return _semantic_key(s) |
| |
|
| | @property |
| | def is_prerelease(self): |
| | return self._parts[1][0] != '|' |
| |
|
| |
|
| | class SemanticMatcher(Matcher): |
| | version_class = SemanticVersion |
| |
|
| |
|
| | class VersionScheme(object): |
| | def __init__(self, key, matcher, suggester=None): |
| | self.key = key |
| | self.matcher = matcher |
| | self.suggester = suggester |
| |
|
| | def is_valid_version(self, s): |
| | try: |
| | self.matcher.version_class(s) |
| | result = True |
| | except UnsupportedVersionError: |
| | result = False |
| | return result |
| |
|
| | def is_valid_matcher(self, s): |
| | try: |
| | self.matcher(s) |
| | result = True |
| | except UnsupportedVersionError: |
| | result = False |
| | return result |
| |
|
| | def is_valid_constraint_list(self, s): |
| | """ |
| | Used for processing some metadata fields |
| | """ |
| | |
| | if s.endswith(','): |
| | s = s[:-1] |
| | return self.is_valid_matcher('dummy_name (%s)' % s) |
| |
|
| | def suggest(self, s): |
| | if self.suggester is None: |
| | result = None |
| | else: |
| | result = self.suggester(s) |
| | return result |
| |
|
| |
|
| | _SCHEMES = { |
| | 'normalized': VersionScheme(_normalized_key, NormalizedMatcher, |
| | _suggest_normalized_version), |
| | 'legacy': VersionScheme(_legacy_key, LegacyMatcher, lambda self, s: s), |
| | 'semantic': VersionScheme(_semantic_key, SemanticMatcher, |
| | _suggest_semantic_version), |
| | } |
| |
|
| | _SCHEMES['default'] = _SCHEMES['normalized'] |
| |
|
| |
|
| | def get_scheme(name): |
| | if name not in _SCHEMES: |
| | raise ValueError('unknown scheme name: %r' % name) |
| | return _SCHEMES[name] |
| |
|