83 lines
2.3 KiB
Python
Raw Normal View History

2012-02-07 14:57:46 +02:00
# -*- coding: utf-8 -*-
import re
_REGEX = re.compile('^(?P<major>(?:0|[1-9][0-9]*))'
'\.(?P<minor>(?:0|[1-9][0-9]*))'
'\.(?P<patch>(?:0|[1-9][0-9]*))'
'(\-(?P<prerelease>[0-9A-Za-z-]+(\.[0-9A-Za-z-]+)*))?'
'(\+(?P<build>[0-9A-Za-z-]+(\.[0-9A-Za-z-]+)*))?$')
2012-02-07 14:57:46 +02:00
2012-02-08 17:14:23 +04:00
if 'cmp' not in __builtins__:
cmp = lambda a, b: (a > b) - (a < b)
2012-02-08 17:14:23 +04:00
2012-02-07 14:57:46 +02:00
def parse(version):
"""
Parse version to major, minor, patch, pre-release, build parts.
"""
match = _REGEX.match(version)
2012-02-07 14:57:46 +02:00
if match is None:
raise ValueError('%s is not valid SemVer string' % version)
2012-02-08 16:07:11 +04:00
verinfo = match.groupdict()
2012-02-08 16:07:11 +04:00
verinfo['major'] = int(verinfo['major'])
verinfo['minor'] = int(verinfo['minor'])
verinfo['patch'] = int(verinfo['patch'])
2012-02-07 14:57:46 +02:00
return verinfo
2012-02-07 14:57:46 +02:00
def compare(ver1, ver2):
def nat_cmp(a, b):
2012-02-08 17:03:20 +04:00
a, b = a or '', b or ''
2012-02-08 17:10:55 +04:00
convert = lambda text: text.isdigit() and int(text) or text.lower()
alphanum_key = lambda key: [convert(c) for c in re.split('([0-9]+)', key)]
return cmp(alphanum_key(a), alphanum_key(b))
2012-02-08 17:03:20 +04:00
def compare_by_keys(d1, d2):
for key in ['major', 'minor', 'patch']:
v = cmp(d1.get(key), d2.get(key))
if v:
2012-02-07 14:57:46 +02:00
return v
2012-02-08 17:03:20 +04:00
rc1, rc2 = d1.get('prerelease'), d2.get('prerelease')
rccmp = nat_cmp(rc1, rc2)
2014-09-24 16:21:39 +03:00
if not (rc1 or rc2):
return rccmp
if not rc1:
2012-02-08 17:03:20 +04:00
return 1
elif not rc2:
return -1
return rccmp or 0
2012-02-07 14:57:46 +02:00
v1, v2 = parse(ver1), parse(ver2)
2012-02-08 17:03:20 +04:00
return compare_by_keys(v1, v2)
2012-02-07 14:57:46 +02:00
def match(version, match_expr):
prefix = match_expr[:2]
if prefix in ('>=', '<=', '=='):
match_version = match_expr[2:]
elif prefix and prefix[0] in ('>', '<', '='):
2012-02-07 14:57:46 +02:00
prefix = prefix[0]
match_version = match_expr[1:]
else:
2012-02-08 16:07:11 +04:00
raise ValueError("match_expr parameter should be in format <op><ver>, "
"where <op> is one of ['<', '>', '==', '<=', '>=']. "
"You provided: %r" % match_expr)
2012-02-07 14:57:46 +02:00
possibilities_dict = {
'>': (1,),
'<': (-1,),
'==': (0,),
'>=': (0, 1),
2012-02-08 16:07:11 +04:00
'<=': (-1, 0)
}
2012-02-07 14:57:46 +02:00
possibilities = possibilities_dict[prefix]
cmp_res = compare(version, match_version)
2012-02-08 16:07:11 +04:00
return cmp_res in possibilities