|
|
|
import os |
|
import subprocess |
|
import warnings |
|
|
|
from packaging.version import parse |
|
|
|
|
|
def digit_version(version_str: str, length: int = 4): |
|
"""Convert a version string into a tuple of integers. |
|
|
|
This method is usually used for comparing two versions. For pre-release |
|
versions: alpha < beta < rc. |
|
|
|
Args: |
|
version_str (str): The version string. |
|
length (int): The maximum number of version levels. Default: 4. |
|
|
|
Returns: |
|
tuple[int]: The version info in digits (integers). |
|
""" |
|
assert 'parrots' not in version_str |
|
version = parse(version_str) |
|
assert version.release, f'failed to parse version {version_str}' |
|
release = list(version.release) |
|
release = release[:length] |
|
if len(release) < length: |
|
release = release + [0] * (length - len(release)) |
|
if version.is_prerelease: |
|
mapping = {'a': -3, 'b': -2, 'rc': -1} |
|
val = -4 |
|
|
|
if version.pre: |
|
if version.pre[0] not in mapping: |
|
warnings.warn(f'unknown prerelease version {version.pre[0]}, ' |
|
'version checking may go wrong') |
|
else: |
|
val = mapping[version.pre[0]] |
|
release.extend([val, version.pre[-1]]) |
|
else: |
|
release.extend([val, 0]) |
|
|
|
elif version.is_postrelease: |
|
release.extend([1, version.post]) |
|
else: |
|
release.extend([0, 0]) |
|
return tuple(release) |
|
|
|
|
|
def _minimal_ext_cmd(cmd): |
|
|
|
env = {} |
|
for k in ['SYSTEMROOT', 'PATH', 'HOME']: |
|
v = os.environ.get(k) |
|
if v is not None: |
|
env[k] = v |
|
|
|
env['LANGUAGE'] = 'C' |
|
env['LANG'] = 'C' |
|
env['LC_ALL'] = 'C' |
|
out = subprocess.Popen( |
|
cmd, stdout=subprocess.PIPE, env=env).communicate()[0] |
|
return out |
|
|
|
|
|
def get_git_hash(fallback='unknown', digits=None): |
|
"""Get the git hash of the current repo. |
|
|
|
Args: |
|
fallback (str, optional): The fallback string when git hash is |
|
unavailable. Defaults to 'unknown'. |
|
digits (int, optional): kept digits of the hash. Defaults to None, |
|
meaning all digits are kept. |
|
|
|
Returns: |
|
str: Git commit hash. |
|
""" |
|
|
|
if digits is not None and not isinstance(digits, int): |
|
raise TypeError('digits must be None or an integer') |
|
|
|
try: |
|
out = _minimal_ext_cmd(['git', 'rev-parse', 'HEAD']) |
|
sha = out.strip().decode('ascii') |
|
if digits is not None: |
|
sha = sha[:digits] |
|
except OSError: |
|
sha = fallback |
|
|
|
return sha |
|
|