hghave.py
1179 lines
| 30.1 KiB
| text/x-python
|
PythonLexer
/ tests / hghave.py
Augie Fackler
|
r26137 | import os | ||
Adrian Buehlmann
|
r16966 | import re | ||
Yuya Nishihara
|
r22994 | import socket | ||
Augie Fackler
|
r26137 | import stat | ||
import subprocess | ||||
Adrian Buehlmann
|
r16966 | import sys | ||
import tempfile | ||||
Mads Kiilerich
|
r52638 | try: | ||
from setuptools.extern.packaging.version import Version | ||||
except ImportError: | ||||
from distutils.version import StrictVersion as Version | ||||
Adrian Buehlmann
|
r16966 | tempprefix = 'hg-hghave-' | ||
Matt Mackall
|
r22093 | checks = { | ||
"true": (lambda: True, "yak shaving"), | ||||
"false": (lambda: False, "nail clipper"), | ||||
r46778 | "known-bad-output": (lambda: True, "use for currently known bad output"), | |||
"missing-correct-output": (lambda: False, "use for missing good output"), | ||||
Matt Mackall
|
r22093 | } | ||
Matt Harbison
|
r41039 | try: | ||
import msvcrt | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r41039 | msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY) | ||
msvcrt.setmode(sys.stderr.fileno(), os.O_BINARY) | ||||
except ImportError: | ||||
pass | ||||
stdout = getattr(sys.stdout, 'buffer', sys.stdout) | ||||
stderr = getattr(sys.stderr, 'buffer', sys.stderr) | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r51106 | def _sys2bytes(p): | ||
if p is None: | ||||
return p | ||||
return p.encode('utf-8') | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r51106 | def _bytes2sys(p): | ||
if p is None: | ||||
Matt Harbison
|
r39795 | return p | ||
Matt Harbison
|
r51106 | return p.decode('utf-8') | ||
Matt Harbison
|
r39795 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | def check(name, desc): | ||
timeless
|
r28757 | """Registers a check function for a feature.""" | ||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | def decorator(func): | ||
checks[name] = (func, desc) | ||||
return func | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | return decorator | ||
Augie Fackler
|
r43346 | |||
timeless
|
r28758 | def checkvers(name, desc, vers): | ||
"""Registers a check function for each of a series of versions. | ||||
Kyle Lippincott
|
r43383 | vers can be a list or an iterator. | ||
Produces a series of feature checks that have the form <name><vers> without | ||||
any punctuation (even if there's punctuation in 'vers'; i.e. this produces | ||||
'py38', not 'py3.8' or 'py-38').""" | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28758 | def decorator(func): | ||
def funcv(v): | ||||
def f(): | ||||
return func(v) | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28758 | return f | ||
Augie Fackler
|
r43346 | |||
timeless
|
r28758 | for v in vers: | ||
Manuel Jacob
|
r52266 | assert isinstance(v, str) | ||
timeless
|
r28758 | f = funcv(v) | ||
checks['%s%s' % (name, v.replace('.', ''))] = (f, desc % v) | ||||
return func | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28758 | return decorator | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r26067 | def checkfeatures(features): | ||
result = { | ||||
'error': [], | ||||
'missing': [], | ||||
'skipped': [], | ||||
} | ||||
for feature in features: | ||||
negate = feature.startswith('no-') | ||||
if negate: | ||||
feature = feature[3:] | ||||
if feature not in checks: | ||||
result['missing'].append(feature) | ||||
continue | ||||
check, desc = checks[feature] | ||||
try: | ||||
available = check() | ||||
Augie Fackler
|
r48082 | except Exception as e: | ||
result['error'].append('hghave check %s failed: %r' % (feature, e)) | ||||
Gregory Szorc
|
r26067 | continue | ||
if not negate and not available: | ||||
result['skipped'].append('missing feature: %s' % desc) | ||||
elif negate and available: | ||||
result['skipped'].append('system supports %s' % desc) | ||||
return result | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r26068 | def require(features): | ||
Gregory Szorc
|
r26067 | """Require that features are available, exiting if not.""" | ||
result = checkfeatures(features) | ||||
Gregory Szorc
|
r26068 | for missing in result['missing']: | ||
Augie Fackler
|
r43346 | stderr.write( | ||
('skipped: unknown feature: %s\n' % missing).encode('utf-8') | ||||
) | ||||
Gregory Szorc
|
r26068 | for msg in result['skipped']: | ||
Matt Harbison
|
r41039 | stderr.write(('skipped: %s\n' % msg).encode('utf-8')) | ||
Gregory Szorc
|
r26068 | for msg in result['error']: | ||
Matt Harbison
|
r41039 | stderr.write(('%s\n' % msg).encode('utf-8')) | ||
Gregory Szorc
|
r26067 | |||
if result['missing']: | ||||
sys.exit(2) | ||||
if result['skipped'] or result['error']: | ||||
sys.exit(1) | ||||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r16966 | def matchoutput(cmd, regexp, ignorestatus=False): | ||
timeless
|
r27114 | """Return the match object if cmd executes successfully and its output | ||
Adrian Buehlmann
|
r16966 | is matched by the supplied regular expression. | ||
""" | ||||
Matt Harbison
|
r47629 | |||
# Tests on Windows have to fake USERPROFILE to point to the test area so | ||||
# that `~` is properly expanded on py3.8+. However, some tools like black | ||||
# make calls that need the real USERPROFILE in order to run `foo --version`. | ||||
env = os.environ | ||||
if os.name == 'nt': | ||||
env = os.environ.copy() | ||||
env['USERPROFILE'] = env['REALUSERPROFILE'] | ||||
Adrian Buehlmann
|
r16966 | r = re.compile(regexp) | ||
Martin von Zweigbergk
|
r41402 | p = subprocess.Popen( | ||
Matt Harbison
|
r47629 | cmd, | ||
shell=True, | ||||
stdout=subprocess.PIPE, | ||||
stderr=subprocess.STDOUT, | ||||
env=env, | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r38205 | s = p.communicate()[0] | ||
ret = p.returncode | ||||
Augie Fackler
|
r26137 | return (ignorestatus or not ret) and r.search(s) | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("baz", "GNU Arch baz client") | ||
Adrian Buehlmann
|
r16966 | def has_baz(): | ||
timeless
|
r29140 | return matchoutput('baz --version 2>&1', br'baz Bazaar version') | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Raphaël Gomès
|
r48168 | @check("bzr", "Breezy library and executable version >= 3.1") | ||
Adrian Buehlmann
|
r16966 | def has_bzr(): | ||
try: | ||||
Raphaël Gomès
|
r48168 | # Test the Breezy python lib | ||
import breezy | ||||
import breezy.bzr.bzrdir | ||||
import breezy.errors | ||||
import breezy.revision | ||||
import breezy.revisionspec | ||||
Augie Fackler
|
r43346 | |||
Raphaël Gomès
|
r48168 | breezy.revisionspec.RevisionSpec | ||
if breezy.__doc__ is None or breezy.version_info[:2] < (3, 1): | ||||
return False | ||||
Yuya Nishihara
|
r29903 | except (AttributeError, ImportError): | ||
Adrian Buehlmann
|
r16966 | return False | ||
Raphaël Gomès
|
r48168 | # Test the executable | ||
return matchoutput('brz --version 2>&1', br'Breezy \(brz\) ') | ||||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r28880 | @check("chg", "running with chg") | ||
def has_chg(): | ||||
r48386 | return 'CHG_INSTALLED_AS_HG' in os.environ | |||
Yuya Nishihara
|
r28880 | |||
Augie Fackler
|
r43346 | |||
Simon Sapin
|
r47458 | @check("rhg", "running with rhg as 'hg'") | ||
def has_rhg(): | ||||
return 'RHG_INSTALLED_AS_HG' in os.environ | ||||
r48639 | @check("pyoxidizer", "running with pyoxidizer build as 'hg'") | |||
Manuel Jacob
|
r50176 | def has_pyoxidizer(): | ||
r48639 | return 'PYOXIDIZED_INSTALLED_AS_HG' in os.environ | |||
Matt Harbison
|
r50728 | @check( | ||
"pyoxidizer-in-memory", | ||||
"running with pyoxidizer build as 'hg' with embedded resources", | ||||
) | ||||
r50845 | def has_pyoxidizer_mem(): | |||
Matt Harbison
|
r50728 | return 'PYOXIDIZED_IN_MEMORY_RSRC' in os.environ | ||
@check( | ||||
"pyoxidizer-in-filesystem", | ||||
"running with pyoxidizer build as 'hg' with external resources", | ||||
) | ||||
r50845 | def has_pyoxidizer_fs(): | |||
Matt Harbison
|
r50728 | return 'PYOXIDIZED_FILESYSTEM_RSRC' in os.environ | ||
Matt Mackall
|
r22093 | @check("cvs", "cvs client/server") | ||
Adrian Buehlmann
|
r16966 | def has_cvs(): | ||
timeless
|
r29140 | re = br'Concurrent Versions System.*?server' | ||
Adrian Buehlmann
|
r16966 | return matchoutput('cvs --version 2>&1', re) and not has_msys() | ||
Augie Fackler
|
r43346 | |||
timeless
|
r28756 | @check("cvs112", "cvs client/server 1.12.* (not cvsnt)") | ||
Bryan O'Sullivan
|
r18285 | def has_cvs112(): | ||
timeless
|
r29140 | re = br'Concurrent Versions System \(CVS\) 1.12.*?server' | ||
Bryan O'Sullivan
|
r18285 | return matchoutput('cvs --version 2>&1', re) and not has_msys() | ||
Augie Fackler
|
r43346 | |||
timeless
|
r28796 | @check("cvsnt", "cvsnt client/server") | ||
def has_cvsnt(): | ||||
timeless
|
r29140 | re = br'Concurrent Versions System \(CVSNT\) (\d+).(\d+).*\(client/server\)' | ||
timeless
|
r28796 | return matchoutput('cvsnt --version 2>&1', re) | ||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("darcs", "darcs client") | ||
Adrian Buehlmann
|
r16966 | def has_darcs(): | ||
Yuya Nishihara
|
r30297 | return matchoutput('darcs --version', br'\b2\.([2-9]|\d{2})', True) | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("mtn", "monotone client (>= 1.0)") | ||
Adrian Buehlmann
|
r16966 | def has_mtn(): | ||
timeless
|
r29140 | return matchoutput('mtn --version', br'monotone', True) and not matchoutput( | ||
Augie Fackler
|
r43346 | 'mtn --version', br'monotone 0\.', True | ||
) | ||||
Adrian Buehlmann
|
r16966 | |||
Matt Mackall
|
r22093 | @check("eol-in-paths", "end-of-lines in paths") | ||
Adrian Buehlmann
|
r16966 | def has_eol_in_paths(): | ||
try: | ||||
Mads Kiilerich
|
r16968 | fd, path = tempfile.mkstemp(dir='.', prefix=tempprefix, suffix='\n\r') | ||
Adrian Buehlmann
|
r16966 | os.close(fd) | ||
os.remove(path) | ||||
return True | ||||
except (IOError, OSError): | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("execbit", "executable bit") | ||
Adrian Buehlmann
|
r16966 | def has_executablebit(): | ||
try: | ||||
EXECFLAGS = stat.S_IXUSR | stat.S_IXGRP | stat.S_IXOTH | ||||
Mads Kiilerich
|
r16968 | fh, fn = tempfile.mkstemp(dir='.', prefix=tempprefix) | ||
Adrian Buehlmann
|
r16966 | try: | ||
os.close(fh) | ||||
Gregory Szorc
|
r25658 | m = os.stat(fn).st_mode & 0o777 | ||
Adrian Buehlmann
|
r16966 | new_file_has_exec = m & EXECFLAGS | ||
os.chmod(fn, m ^ EXECFLAGS) | ||||
Augie Fackler
|
r43346 | exec_flags_cannot_flip = (os.stat(fn).st_mode & 0o777) == m | ||
Adrian Buehlmann
|
r16966 | finally: | ||
os.unlink(fn) | ||||
except (IOError, OSError): | ||||
# we don't care, the user probably won't be able to commit anyway | ||||
return False | ||||
return not (new_file_has_exec or exec_flags_cannot_flip) | ||||
Augie Fackler
|
r43346 | |||
pacien
|
r49133 | @check("suidbit", "setuid and setgid bit") | ||
def has_suidbit(): | ||||
Matt Harbison
|
r49281 | if ( | ||
getattr(os, "statvfs", None) is None | ||||
or getattr(os, "ST_NOSUID", None) is None | ||||
): | ||||
pacien
|
r49133 | return False | ||
return bool(os.statvfs('.').f_flag & os.ST_NOSUID) | ||||
Matt Mackall
|
r22093 | @check("icasefs", "case insensitive file system") | ||
Adrian Buehlmann
|
r16966 | def has_icasefs(): | ||
# Stolen from mercurial.util | ||||
Mads Kiilerich
|
r16968 | fd, path = tempfile.mkstemp(dir='.', prefix=tempprefix) | ||
Adrian Buehlmann
|
r16966 | os.close(fd) | ||
try: | ||||
s1 = os.stat(path) | ||||
d, b = os.path.split(path) | ||||
p2 = os.path.join(d, b.upper()) | ||||
if path == p2: | ||||
p2 = os.path.join(d, b.lower()) | ||||
try: | ||||
s2 = os.stat(p2) | ||||
return s2 == s1 | ||||
except OSError: | ||||
return False | ||||
finally: | ||||
os.remove(path) | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("fifo", "named pipes") | ||
Adrian Buehlmann
|
r16966 | def has_fifo(): | ||
Mads Kiilerich
|
r16969 | if getattr(os, "mkfifo", None) is None: | ||
return False | ||||
name = tempfile.mktemp(dir='.', prefix=tempprefix) | ||||
try: | ||||
os.mkfifo(name) | ||||
os.unlink(name) | ||||
return True | ||||
except OSError: | ||||
return False | ||||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("killdaemons", 'killdaemons.py support') | ||
Patrick Mezard
|
r17467 | def has_killdaemons(): | ||
return True | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("cacheable", "cacheable filesystem") | ||
Adrian Buehlmann
|
r16966 | def has_cacheable_fs(): | ||
from mercurial import util | ||||
Mads Kiilerich
|
r16968 | fd, path = tempfile.mkstemp(dir='.', prefix=tempprefix) | ||
Adrian Buehlmann
|
r16966 | os.close(fd) | ||
try: | ||||
Matt Harbison
|
r51105 | return util.cachestat(_sys2bytes(path)).cacheable() | ||
Adrian Buehlmann
|
r16966 | finally: | ||
os.remove(path) | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("lsprof", "python lsprof module") | ||
Adrian Buehlmann
|
r16966 | def has_lsprof(): | ||
try: | ||||
import _lsprof | ||||
Augie Fackler
|
r43346 | |||
_lsprof.Profiler # silence unused import warning | ||||
Adrian Buehlmann
|
r16966 | return True | ||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Julien Cristau
|
r44760 | def _gethgversion(): | ||
timeless
|
r29140 | m = matchoutput('hg --version --quiet 2>&1', br'(\d+)\.(\d+)') | ||
timeless
|
r28761 | if not m: | ||
return (0, 0) | ||||
return (int(m.group(1)), int(m.group(2))) | ||||
Augie Fackler
|
r43346 | |||
Julien Cristau
|
r44760 | _hgversion = None | ||
def gethgversion(): | ||||
global _hgversion | ||||
if _hgversion is None: | ||||
_hgversion = _gethgversion() | ||||
return _hgversion | ||||
Manuel Jacob
|
r52266 | @checkvers( | ||
"hg", "Mercurial >= %s", ['%d.%d' % divmod(x, 10) for x in range(9, 99)] | ||||
) | ||||
timeless
|
r28761 | def has_hg_range(v): | ||
major, minor = v.split('.')[0:2] | ||||
return gethgversion() >= (int(major), int(minor)) | ||||
Augie Fackler
|
r43346 | |||
r44955 | @check("rust", "Using the Rust extensions") | |||
def has_rust(): | ||||
"""Check is the mercurial currently running is using some rust code""" | ||||
Gregory Szorc
|
r45130 | cmd = 'hg debuginstall --quiet 2>&1' | ||
r44955 | match = br'checking module policy \(([^)]+)\)' | |||
policy = matchoutput(cmd, match) | ||||
if not policy: | ||||
return False | ||||
return b'rust' in policy.group(1) | ||||
timeless
|
r28761 | @check("hg08", "Mercurial >= 0.8") | ||
def has_hg08(): | ||||
if checks["hg09"][0](): | ||||
return True | ||||
return matchoutput('hg help annotate 2>&1', '--date') | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28761 | @check("hg07", "Mercurial >= 0.7") | ||
def has_hg07(): | ||||
if checks["hg08"][0](): | ||||
return True | ||||
return matchoutput('hg --version --quiet 2>&1', 'Mercurial Distributed SCM') | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28761 | @check("hg06", "Mercurial >= 0.6") | ||
def has_hg06(): | ||||
if checks["hg07"][0](): | ||||
return True | ||||
return matchoutput('hg --version --quiet 2>&1', 'Mercurial version') | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("gettext", "GNU Gettext (msgfmt)") | ||
Adrian Buehlmann
|
r16966 | def has_gettext(): | ||
timeless
|
r29140 | return matchoutput('msgfmt --version', br'GNU gettext-tools') | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("git", "git command line client") | ||
Adrian Buehlmann
|
r16966 | def has_git(): | ||
timeless
|
r29140 | return matchoutput('git --version 2>&1', br'^git version') | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Sean Farley
|
r32901 | def getgitversion(): | ||
m = matchoutput('git --version 2>&1', br'git version (\d+)\.(\d+)') | ||||
if not m: | ||||
return (0, 0) | ||||
return (int(m.group(1)), int(m.group(2))) | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r44967 | @check("pygit2", "pygit2 Python library") | ||
Manuel Jacob
|
r50176 | def has_pygit2(): | ||
Martin von Zweigbergk
|
r44967 | try: | ||
import pygit2 | ||||
pygit2.Oid # silence unused import | ||||
return True | ||||
except ImportError: | ||||
return False | ||||
Matt Harbison
|
r35137 | # https://github.com/git-lfs/lfs-test-server | ||
@check("lfs-test-server", "git-lfs test server") | ||||
def has_lfsserver(): | ||||
exe = 'lfs-test-server' | ||||
if has_windows(): | ||||
exe = 'lfs-test-server.exe' | ||||
return any( | ||||
os.access(os.path.join(path, exe), os.X_OK) | ||||
for path in os.environ["PATH"].split(os.pathsep) | ||||
) | ||||
Augie Fackler
|
r43346 | |||
Manuel Jacob
|
r52266 | @checkvers("git", "git client (with ext::sh support) version >= %s", ('1.9',)) | ||
Sean Farley
|
r32901 | def has_git_range(v): | ||
major, minor = v.split('.')[0:2] | ||||
return getgitversion() >= (int(major), int(minor)) | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("docutils", "Docutils text processing library") | ||
Adrian Buehlmann
|
r16966 | def has_docutils(): | ||
try: | ||||
Yuya Nishihara
|
r28779 | import docutils.core | ||
Augie Fackler
|
r43346 | |||
docutils.core.publish_cmdline # silence unused import | ||||
Adrian Buehlmann
|
r16966 | return True | ||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r16966 | def getsvnversion(): | ||
timeless
|
r29140 | m = matchoutput('svn --version --quiet 2>&1', br'^(\d+)\.(\d+)') | ||
Adrian Buehlmann
|
r16966 | if not m: | ||
return (0, 0) | ||||
return (int(m.group(1)), int(m.group(2))) | ||||
Augie Fackler
|
r43346 | |||
Manuel Jacob
|
r52266 | @checkvers("svn", "subversion client and admin tools >= %s", ('1.3', '1.5')) | ||
timeless
|
r28759 | def has_svn_range(v): | ||
major, minor = v.split('.')[0:2] | ||||
return getsvnversion() >= (int(major), int(minor)) | ||||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("svn", "subversion client and admin tools") | ||
Adrian Buehlmann
|
r16966 | def has_svn(): | ||
Augie Fackler
|
r43346 | return matchoutput('svn --version 2>&1', br'^svn, version') and matchoutput( | ||
'svnadmin --version 2>&1', br'^svnadmin, version' | ||||
) | ||||
Adrian Buehlmann
|
r16966 | |||
Matt Mackall
|
r22093 | @check("svn-bindings", "subversion python bindings") | ||
Adrian Buehlmann
|
r16966 | def has_svn_bindings(): | ||
try: | ||||
import svn.core | ||||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r16966 | version = svn.core.SVN_VER_MAJOR, svn.core.SVN_VER_MINOR | ||
if version < (1, 4): | ||||
return False | ||||
return True | ||||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("p4", "Perforce server and client") | ||
Adrian Buehlmann
|
r16966 | def has_p4(): | ||
Augie Fackler
|
r43346 | return matchoutput('p4 -V', br'Rev\. P4/') and matchoutput( | ||
'p4d -V', br'Rev\. P4D/' | ||||
) | ||||
Adrian Buehlmann
|
r16966 | |||
Matt Mackall
|
r22093 | @check("symlink", "symbolic links") | ||
Adrian Buehlmann
|
r16966 | def has_symlink(): | ||
Matt Harbison
|
r43760 | # mercurial.windows.checklink() is a hard 'no' at the moment | ||
if os.name == 'nt' or getattr(os, "symlink", None) is None: | ||||
Adrian Buehlmann
|
r16966 | return False | ||
Mads Kiilerich
|
r16968 | name = tempfile.mktemp(dir='.', prefix=tempprefix) | ||
Adrian Buehlmann
|
r16966 | try: | ||
os.symlink(".", name) | ||||
os.unlink(name) | ||||
return True | ||||
except (OSError, AttributeError): | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("hardlink", "hardlinks") | ||
Mads Kiilerich
|
r16971 | def has_hardlink(): | ||
from mercurial import util | ||||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r16971 | fh, fn = tempfile.mkstemp(dir='.', prefix=tempprefix) | ||
os.close(fh) | ||||
name = tempfile.mktemp(dir='.', prefix=tempprefix) | ||||
try: | ||||
Manuel Jacob
|
r44935 | util.oslink(_sys2bytes(fn), _sys2bytes(name)) | ||
Matt Mackall
|
r25090 | os.unlink(name) | ||
return True | ||||
except OSError: | ||||
return False | ||||
Mads Kiilerich
|
r16971 | finally: | ||
os.unlink(fn) | ||||
Augie Fackler
|
r43346 | |||
Jun Wu
|
r31576 | @check("hardlink-whitelisted", "hardlinks on whitelisted filesystems") | ||
def has_hardlink_whitelisted(): | ||||
Yuya Nishihara
|
r31674 | from mercurial import util | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r31678 | try: | ||
Augie Fackler
|
r36960 | fstype = util.getfstype(b'.') | ||
Yuya Nishihara
|
r31678 | except OSError: | ||
return False | ||||
Jun Wu
|
r31576 | return fstype in util._hardlinkfswhitelist | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r30230 | @check("rmcwd", "can remove current working directory") | ||
def has_rmcwd(): | ||||
ocwd = os.getcwd() | ||||
temp = tempfile.mkdtemp(dir='.', prefix=tempprefix) | ||||
try: | ||||
os.chdir(temp) | ||||
# On Linux, 'rmdir .' isn't allowed, but the other names are okay. | ||||
# On Solaris and Windows, the cwd can't be removed by any names. | ||||
os.rmdir(os.getcwd()) | ||||
return True | ||||
except OSError: | ||||
return False | ||||
finally: | ||||
os.chdir(ocwd) | ||||
Yuya Nishihara
|
r30242 | # clean up temp dir on platforms where cwd can't be removed | ||
try: | ||||
os.rmdir(temp) | ||||
except OSError: | ||||
pass | ||||
Yuya Nishihara
|
r30230 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("tla", "GNU Arch tla client") | ||
Adrian Buehlmann
|
r16966 | def has_tla(): | ||
timeless
|
r29140 | return matchoutput('tla --version 2>&1', br'The GNU Arch Revision') | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("gpg", "gpg client") | ||
Adrian Buehlmann
|
r16966 | def has_gpg(): | ||
timeless
|
r29140 | return matchoutput('gpg --version 2>&1', br'GnuPG') | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r29790 | @check("gpg2", "gpg client v2") | ||
def has_gpg2(): | ||||
return matchoutput('gpg --version 2>&1', br'GnuPG[^0-9]+2\.') | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r29873 | @check("gpg21", "gpg client v2.1+") | ||
def has_gpg21(): | ||||
return matchoutput('gpg --version 2>&1', br'GnuPG[^0-9]+2\.(?!0)') | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("unix-permissions", "unix-style permissions") | ||
Adrian Buehlmann
|
r16966 | def has_unix_permissions(): | ||
Mads Kiilerich
|
r16968 | d = tempfile.mkdtemp(dir='.', prefix=tempprefix) | ||
Adrian Buehlmann
|
r16966 | try: | ||
fname = os.path.join(d, 'foo') | ||||
Gregory Szorc
|
r25658 | for umask in (0o77, 0o07, 0o22): | ||
Adrian Buehlmann
|
r16966 | os.umask(umask) | ||
f = open(fname, 'w') | ||||
f.close() | ||||
mode = os.stat(fname).st_mode | ||||
os.unlink(fname) | ||||
Gregory Szorc
|
r25658 | if mode & 0o777 != ~umask & 0o666: | ||
Adrian Buehlmann
|
r16966 | return False | ||
return True | ||||
finally: | ||||
os.rmdir(d) | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r22994 | @check("unix-socket", "AF_UNIX socket family") | ||
def has_unix_socket(): | ||||
return getattr(socket, 'AF_UNIX', None) is not None | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("root", "root permissions") | ||
Matt Mackall
|
r20008 | def has_root(): | ||
Simon Heimberg
|
r20114 | return getattr(os, 'geteuid', None) and os.geteuid() == 0 | ||
Matt Mackall
|
r20008 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("pyflakes", "Pyflakes python linter") | ||
Adrian Buehlmann
|
r16966 | def has_pyflakes(): | ||
Manuel Jacob
|
r44987 | try: | ||
import pyflakes | ||||
pyflakes.__version__ | ||||
except ImportError: | ||||
return False | ||||
else: | ||||
return True | ||||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r16966 | |||
Pierre-Yves David
|
r31413 | @check("pylint", "Pylint python linter") | ||
def has_pylint(): | ||||
r50686 | return matchoutput("pylint --help", br"[Uu]sage:[ ]+pylint", True) | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r31413 | |||
r52955 | @check("clang-format", "clang-format C code formatter (11 <= … < 19)") | |||
Augie Fackler
|
r34697 | def has_clang_format(): | ||
Yuya Nishihara
|
r45744 | m = matchoutput('clang-format --version', br'clang-format version (\d+)') | ||
r52955 | # style changed somewhere between 10.x and 11.x and after 19. | |||
Kyle Lippincott
|
r49110 | if m: | ||
r52955 | major_version = int(m.group(1)) | |||
return 11 <= major_version < 19 | ||||
Kyle Lippincott
|
r49110 | # Assist Googler contributors, they have a centrally-maintained version of | ||
# clang-format that is generally very fresh, but unlike most builds (both | ||||
# official and unofficial), it does *not* include a version number. | ||||
return matchoutput( | ||||
'clang-format --version', br'clang-format .*google3-trunk \([0-9a-f]+\)' | ||||
) | ||||
Augie Fackler
|
r34697 | |||
Augie Fackler
|
r43346 | |||
r35034 | @check("jshint", "JSHint static code analysis tool") | |||
def has_jshint(): | ||||
return matchoutput("jshint --version 2>&1", br"jshint v") | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("pygments", "Pygments source highlighting library") | ||
Adrian Buehlmann
|
r16966 | def has_pygments(): | ||
try: | ||||
import pygments | ||||
Augie Fackler
|
r43346 | |||
pygments.highlight # silence unused import warning | ||||
Adrian Buehlmann
|
r16966 | return True | ||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
r50846 | def getpygmentsversion(): | |||
Cédric Krier
|
r49382 | try: | ||
import pygments | ||||
v = pygments.__version__ | ||||
r50846 | ||||
parts = v.split(".") | ||||
return (int(parts[0]), int(parts[1])) | ||||
Cédric Krier
|
r49382 | except ImportError: | ||
r50846 | return (0, 0) | |||
Cédric Krier
|
r49382 | |||
Manuel Jacob
|
r52266 | @checkvers("pygments", "Pygments version >= %s", ('2.5', '2.11', '2.14')) | ||
r50846 | def has_pygments_range(v): | |||
major, minor = v.split('.')[0:2] | ||||
return getpygmentsversion() >= (int(major), int(minor)) | ||||
r50844 | ||||
Matt Mackall
|
r22093 | @check("outer-repo", "outer repo") | ||
Adrian Buehlmann
|
r16966 | def has_outer_repo(): | ||
Mads Kiilerich
|
r17016 | # failing for other reasons than 'no repo' imply that there is a repo | ||
Augie Fackler
|
r43346 | return not matchoutput('hg root 2>&1', br'abort: no repository found', True) | ||
Adrian Buehlmann
|
r16966 | |||
Gregory Szorc
|
r28591 | @check("ssl", "ssl module available") | ||
Adrian Buehlmann
|
r16966 | def has_ssl(): | ||
try: | ||||
import ssl | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r28591 | ssl.CERT_NONE | ||
Adrian Buehlmann
|
r16966 | return True | ||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r29481 | @check("defaultcacertsloaded", "detected presence of loaded system CA certs") | ||
def has_defaultcacertsloaded(): | ||||
import ssl | ||||
Gregory Szorc
|
r29483 | from mercurial import sslutil, ui as uimod | ||
Gregory Szorc
|
r29481 | |||
Yuya Nishihara
|
r30559 | ui = uimod.ui.load() | ||
Gregory Szorc
|
r29483 | cafile = sslutil._defaultcacerts(ui) | ||
Gregory Szorc
|
r29481 | ctx = ssl.create_default_context() | ||
if cafile: | ||||
ctx.load_verify_locations(cafile=cafile) | ||||
else: | ||||
ctx.load_default_certs() | ||||
return len(ctx.get_ca_certs()) > 0 | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r29601 | @check("tls1.2", "TLS 1.2 protocol support") | ||
def has_tls1_2(): | ||||
from mercurial import sslutil | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r41426 | return b'tls1.2' in sslutil.supportedprotocols | ||
Gregory Szorc
|
r29601 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("windows", "Windows") | ||
Adrian Buehlmann
|
r16966 | def has_windows(): | ||
return os.name == 'nt' | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("system-sh", "system() uses sh") | ||
Adrian Buehlmann
|
r16966 | def has_system_sh(): | ||
return os.name != 'nt' | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("serve", "platform and python can manage 'hg serve -d'") | ||
Adrian Buehlmann
|
r16966 | def has_serve(): | ||
Matt Harbison
|
r32856 | return True | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45523 | @check("setprocname", "whether osutil.setprocname is available or not") | ||
def has_setprocname(): | ||||
try: | ||||
from mercurial.utils import procutil | ||||
procutil.setprocname | ||||
return True | ||||
except AttributeError: | ||||
return False | ||||
Ludovic Chabant
|
r52916 | @check("gui", "whether a gui environment is available or not") | ||
def has_gui(): | ||||
from mercurial.utils import procutil | ||||
return procutil.gui() | ||||
Matt Mackall
|
r22093 | @check("test-repo", "running tests from repository") | ||
Matt Mackall
|
r21208 | def has_test_repo(): | ||
t = os.environ["TESTDIR"] | ||||
return os.path.isdir(os.path.join(t, "..", ".hg")) | ||||
Augie Fackler
|
r43346 | |||
Joerg Sonnenberger
|
r47346 | @check("network-io", "whether tests are allowed to access 3rd party services") | ||
Manuel Jacob
|
r50176 | def has_network_io(): | ||
Joerg Sonnenberger
|
r47346 | t = os.environ.get("HGTESTS_ALLOW_NETIO") | ||
return t == "1" | ||||
Matt Harbison
|
r47043 | @check("curses", "terminfo compiler and curses module") | ||
def has_curses(): | ||||
Mads Kiilerich
|
r20304 | try: | ||
import curses | ||||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r20304 | curses.COLOR_BLUE | ||
Matt Harbison
|
r47038 | |||
# Windows doesn't have a `tic` executable, but the windows_curses | ||||
# package is sufficient to run the tests without it. | ||||
if os.name == 'nt': | ||||
return True | ||||
Matt Harbison
|
r47043 | return has_tic() | ||
Pulkit Goyal
|
r44736 | except (ImportError, AttributeError): | ||
Mads Kiilerich
|
r20304 | return False | ||
Adrian Buehlmann
|
r16966 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r47043 | @check("tic", "terminfo compiler") | ||
def has_tic(): | ||||
return matchoutput('test -x "`which tic`"', br'') | ||||
Matt Harbison
|
r44098 | @check("xz", "xz compression utility") | ||
def has_xz(): | ||||
# When Windows invokes a subprocess in shell mode, it uses `cmd.exe`, which | ||||
# only knows `where`, not `which`. So invoke MSYS shell explicitly. | ||||
return matchoutput("sh -c 'test -x \"`which xz`\"'", b'') | ||||
Matt Mackall
|
r22093 | @check("msys", "Windows with MSYS") | ||
Adrian Buehlmann
|
r16966 | def has_msys(): | ||
return os.getenv('MSYSTEM') | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r22093 | @check("aix", "AIX") | ||
Jim Hague
|
r19092 | def has_aix(): | ||
return sys.platform.startswith("aix") | ||||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r22575 | @check("osx", "OS X") | ||
def has_osx(): | ||||
return sys.platform == 'darwin' | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r29026 | @check("osxpackaging", "OS X packaging tools") | ||
def has_osxpackaging(): | ||||
try: | ||||
Augie Fackler
|
r43346 | return ( | ||
matchoutput('pkgbuild', br'Usage: pkgbuild ', ignorestatus=1) | ||||
and matchoutput( | ||||
'productbuild', br'Usage: productbuild ', ignorestatus=1 | ||||
) | ||||
and matchoutput('lsbom', br'Usage: lsbom', ignorestatus=1) | ||||
and matchoutput('xar --help', br'Usage: xar', ignorestatus=1) | ||||
) | ||||
Augie Fackler
|
r29026 | except ImportError: | ||
return False | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r34886 | @check('linuxormacos', 'Linux or MacOS') | ||
def has_linuxormacos(): | ||||
# This isn't a perfect test for MacOS. But it is sufficient for our needs. | ||||
return sys.platform.startswith(('linux', 'darwin')) | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r26111 | @check("docker", "docker support") | ||
def has_docker(): | ||||
timeless
|
r29140 | pat = br'A self-sufficient runtime for' | ||
Augie Fackler
|
r26111 | if matchoutput('docker --help', pat): | ||
if 'linux' not in sys.platform: | ||||
# TODO: in theory we should be able to test docker-based | ||||
# package creation on non-linux using boot2docker, but in | ||||
# practice that requires extra coordination to make sure | ||||
# $TESTTEMP is going to be visible at the same path to the | ||||
# boot2docker VM. If we figure out how to verify that, we | ||||
# can use the following instead of just saying False: | ||||
# return 'DOCKER_HOST' in os.environ | ||||
return False | ||||
return True | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r26110 | @check("debhelper", "debian packaging tools") | ||
def has_debhelper(): | ||||
Kyle Lippincott
|
r34395 | # Some versions of dpkg say `dpkg', some say 'dpkg' (` vs ' on the first | ||
# quote), so just accept anything in that spot. | ||||
Augie Fackler
|
r43346 | dpkg = matchoutput( | ||
'dpkg --version', br"Debian .dpkg' package management program" | ||||
) | ||||
dh = matchoutput( | ||||
'dh --help', br'dh is a part of debhelper.', ignorestatus=True | ||||
) | ||||
dh_py2 = matchoutput( | ||||
'dh_python2 --help', br'other supported Python versions' | ||||
) | ||||
Kyle Lippincott
|
r34400 | # debuild comes from the 'devscripts' package, though you might want | ||
# the 'build-debs' package instead, which has a dependency on devscripts. | ||||
Augie Fackler
|
r43346 | debuild = matchoutput( | ||
'debuild --help', br'to run debian/rules with given parameter' | ||||
) | ||||
Kyle Lippincott
|
r34400 | return dpkg and dh and dh_py2 and debuild | ||
Augie Fackler
|
r26110 | |||
Augie Fackler
|
r43346 | |||
@check( | ||||
"debdeps", "debian build dependencies (run dpkg-checkbuilddeps in contrib/)" | ||||
) | ||||
Kyle Lippincott
|
r34402 | def has_debdeps(): | ||
# just check exit status (ignoring output) | ||||
Gregory Szorc
|
r38029 | path = '%s/../contrib/packaging/debian/control' % os.environ['TESTDIR'] | ||
Kyle Lippincott
|
r34402 | return matchoutput('dpkg-checkbuilddeps %s' % path, br'') | ||
Augie Fackler
|
r43346 | |||
timeless
|
r29867 | @check("demandimport", "demandimport enabled") | ||
def has_demandimport(): | ||||
Saurabh Singh
|
r34841 | # chg disables demandimport intentionally for performance wins. | ||
Augie Fackler
|
r43346 | return (not has_chg()) and os.environ.get('HGDEMANDIMPORT') != 'disable' | ||
timeless
|
r29867 | |||
Matt Harbison
|
r51106 | # Add "py36", "py37", ... as possible feature checks. Note that there's no | ||
Kyle Lippincott
|
r43383 | # punctuation here. | ||
Manuel Jacob
|
r52267 | @checkvers( | ||
"py", | ||||
"Python >= %s", | ||||
('3.6', '3.7', '3.8', '3.9', '3.10', '3.11', '3.12', '3.13'), | ||||
) | ||||
Gregory Szorc
|
r41488 | def has_python_range(v): | ||
major, minor = v.split('.')[0:2] | ||||
py_major, py_minor = sys.version_info.major, sys.version_info.minor | ||||
return (py_major, py_minor) >= (int(major), int(minor)) | ||||
Augie Fackler
|
r43346 | |||
Martijn Pieters
|
r40299 | @check("py3", "running with Python 3.x") | ||
def has_py3(): | ||||
FUJIWARA Katsunori
|
r19931 | return 3 == sys.version_info[0] | ||
Yuya Nishihara
|
r25859 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r28582 | @check("py3exe", "a Python 3.x interpreter is available") | ||
def has_python3exe(): | ||||
Matt Harbison
|
r47950 | py = 'python3' | ||
if os.name == 'nt': | ||||
py = 'py -3' | ||||
Matt Harbison
|
r51106 | return matchoutput('%s -V' % py, br'^Python 3.(6|7|8|9|10|11)') | ||
Gregory Szorc
|
r28582 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r25859 | @check("pure", "running with pure Python code") | ||
def has_pure(): | ||||
Augie Fackler
|
r43346 | return any( | ||
[ | ||||
os.environ.get("HGMODULEPOLICY") == "py", | ||||
os.environ.get("HGTEST_RUN_TESTS_PURE") == "--pure", | ||||
] | ||||
) | ||||
Augie Fackler
|
r26109 | |||
Kyle Lippincott
|
r32473 | @check("slow", "allow slow tests (use --allow-slow-tests)") | ||
Augie Fackler
|
r26109 | def has_slow(): | ||
return os.environ.get('HGTEST_SLOW') == 'slow' | ||||
David R. MacIver
|
r26842 | |||
Augie Fackler
|
r43346 | |||
timeless
|
r28383 | @check("hypothesis", "Hypothesis automated test generation") | ||
David R. MacIver
|
r26842 | def has_hypothesis(): | ||
try: | ||||
import hypothesis | ||||
Augie Fackler
|
r43346 | |||
David R. MacIver
|
r26842 | hypothesis.given | ||
return True | ||||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r29843 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r29843 | @check("unziplinks", "unzip(1) understands and extracts symlinks") | ||
def unzip_understands_symlinks(): | ||||
return matchoutput('unzip --help', br'Info-ZIP') | ||||
Gregory Szorc
|
r30441 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r30441 | @check("zstd", "zstd Python module available") | ||
def has_zstd(): | ||||
try: | ||||
import mercurial.zstd | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r30441 | mercurial.zstd.__version__ | ||
return True | ||||
except ImportError: | ||||
return False | ||||
Bryan O'Sullivan
|
r31964 | |||
Augie Fackler
|
r43346 | |||
Bryan O'Sullivan
|
r31964 | @check("devfull", "/dev/full special file") | ||
def has_dev_full(): | ||||
return os.path.exists('/dev/full') | ||||
Augie Fackler
|
r32726 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r43730 | @check("ensurepip", "ensurepip module") | ||
def has_ensurepip(): | ||||
try: | ||||
import ensurepip | ||||
ensurepip.bootstrap | ||||
return True | ||||
except ImportError: | ||||
return False | ||||
Matt Harbison
|
r46713 | @check("virtualenv", "virtualenv support") | ||
def has_virtualenv(): | ||||
Augie Fackler
|
r32726 | try: | ||
import virtualenv | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r46713 | # --no-site-package became the default in 1.7 (Nov 2011), and the | ||
# argument was removed in 20.0 (Feb 2020). Rather than make the | ||||
# script complicated, just ignore ancient versions. | ||||
return int(virtualenv.__version__.split('.')[0]) > 1 | ||||
except (AttributeError, ImportError, IndexError): | ||||
Augie Fackler
|
r32726 | return False | ||
Siddharth Agarwal
|
r32770 | |||
Augie Fackler
|
r43346 | |||
Siddharth Agarwal
|
r32770 | @check("fsmonitor", "running tests with fsmonitor") | ||
def has_fsmonitor(): | ||||
return 'HGFSMONITOR_TESTS' in os.environ | ||||
Rishabh Madan
|
r33660 | |||
Augie Fackler
|
r43346 | |||
Rishabh Madan
|
r33660 | @check("fuzzywuzzy", "Fuzzy string matching library") | ||
def has_fuzzywuzzy(): | ||||
try: | ||||
import fuzzywuzzy | ||||
Augie Fackler
|
r43346 | |||
Rishabh Madan
|
r33660 | fuzzywuzzy.__version__ | ||
return True | ||||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r35686 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r35686 | @check("clang-libfuzzer", "clang new enough to include libfuzzer") | ||
def has_clang_libfuzzer(): | ||||
Gregory Szorc
|
r41684 | mat = matchoutput('clang --version', br'clang version (\d)') | ||
Augie Fackler
|
r35686 | if mat: | ||
# libfuzzer is new in clang 6 | ||||
return int(mat.group(1)) > 5 | ||||
return False | ||||
Jun Wu
|
r36696 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r38238 | @check("clang-6.0", "clang 6.0 with version suffix (libfuzzer included)") | ||
def has_clang60(): | ||||
Gregory Szorc
|
r41684 | return matchoutput('clang-6.0 --version', br'clang version 6\.') | ||
Yuya Nishihara
|
r38238 | |||
Augie Fackler
|
r43346 | |||
Jun Wu
|
r36696 | @check("xdiff", "xdiff algorithm") | ||
def has_xdiff(): | ||||
try: | ||||
from mercurial import policy | ||||
Augie Fackler
|
r43346 | |||
Jun Wu
|
r36696 | bdiff = policy.importmod('bdiff') | ||
Augie Fackler
|
r36959 | return bdiff.xdiffblocks(b'', b'') == [(0, 0, 0, 0)] | ||
Augie Fackler
|
r36712 | except (ImportError, AttributeError): | ||
Jun Wu
|
r36696 | return False | ||
Gregory Szorc
|
r37356 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37360 | @check('extraextensions', 'whether tests are running with extra extensions') | ||
def has_extraextensions(): | ||||
return 'HGTESTEXTRAEXTENSIONS' in os.environ | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37356 | def getrepofeatures(): | ||
"""Obtain set of repository features in use. | ||||
HGREPOFEATURES can be used to define or remove features. It contains | ||||
a space-delimited list of feature strings. Strings beginning with ``-`` | ||||
mean to remove. | ||||
""" | ||||
# Default list provided by core. | ||||
features = { | ||||
Gregory Szorc
|
r37364 | 'bundlerepo', | ||
Gregory Szorc
|
r37356 | 'revlogstore', | ||
Gregory Szorc
|
r37433 | 'fncache', | ||
Gregory Szorc
|
r37356 | } | ||
# Features that imply other features. | ||||
implies = { | ||||
Gregory Szorc
|
r37433 | 'simplestore': ['-revlogstore', '-bundlerepo', '-fncache'], | ||
Gregory Szorc
|
r37356 | } | ||
for override in os.environ.get('HGREPOFEATURES', '').split(' '): | ||||
if not override: | ||||
continue | ||||
if override.startswith('-'): | ||||
if override[1:] in features: | ||||
features.remove(override[1:]) | ||||
else: | ||||
features.add(override) | ||||
for imply in implies.get(override, []): | ||||
if imply.startswith('-'): | ||||
if imply[1:] in features: | ||||
features.remove(imply[1:]) | ||||
else: | ||||
features.add(imply) | ||||
return features | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37356 | @check('reporevlogstore', 'repository using the default revlog store') | ||
def has_reporevlogstore(): | ||||
return 'revlogstore' in getrepofeatures() | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37356 | @check('reposimplestore', 'repository using simple storage extension') | ||
def has_reposimplestore(): | ||||
return 'simplestore' in getrepofeatures() | ||||
Gregory Szorc
|
r37364 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37364 | @check('repobundlerepo', 'whether we can open bundle files as repos') | ||
def has_repobundlerepo(): | ||||
return 'bundlerepo' in getrepofeatures() | ||||
Gregory Szorc
|
r37433 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37433 | @check('repofncache', 'repository has an fncache') | ||
def has_repofncache(): | ||||
return 'fncache' in getrepofeatures() | ||||
Augie Fackler
|
r39684 | |||
Augie Fackler
|
r43346 | |||
Simon Sapin
|
r48054 | @check('dirstate-v2', 'using the v2 format of .hg/dirstate') | ||
def has_dirstate_v2(): | ||||
# Keep this logic in sync with `newreporequirements()` in `mercurial/localrepo.py` | ||||
Arseniy Alekseyev
|
r50515 | return matchoutput( | ||
'hg config format.use-dirstate-v2', b'(?i)1|yes|true|on|always' | ||||
Simon Sapin
|
r48054 | ) | ||
r46890 | @check('sqlite', 'sqlite3 module and matching cli is available') | |||
Gregory Szorc
|
r40362 | def has_sqlite(): | ||
try: | ||||
import sqlite3 | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r40492 | version = sqlite3.sqlite_version_info | ||
Gregory Szorc
|
r40362 | except ImportError: | ||
return False | ||||
Yuya Nishihara
|
r40492 | if version < (3, 8, 3): | ||
# WITH clause not supported | ||||
return False | ||||
Gregory Szorc
|
r41684 | return matchoutput('sqlite3 -version', br'^3\.\d+') | ||
Gregory Szorc
|
r40362 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r47122 | @check('vcr', 'vcr http mocking library (pytest-vcr)') | ||
Augie Fackler
|
r39684 | def has_vcr(): | ||
try: | ||||
import vcr | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r39684 | vcr.VCR | ||
return True | ||||
except (ImportError, AttributeError): | ||||
pass | ||||
return False | ||||
Augie Fackler
|
r41953 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r41953 | @check('emacs', 'GNU Emacs') | ||
def has_emacs(): | ||||
Augie Fackler
|
r42008 | # Our emacs lisp uses `with-eval-after-load` which is new in emacs | ||
# 24.4, so we allow emacs 24.4, 24.5, and 25+ (24.5 was the last | ||||
# 24 release) | ||||
return matchoutput('emacs --version', b'GNU Emacs 2(4.4|4.5|5|6|7|8|9)') | ||||
r43365 | ||||
Raphaël Gomès
|
r52597 | @check('black', 'the black formatter for python >=23.3.0') | ||
r43365 | def has_black(): | |||
r43664 | blackcmd = 'black --version' | |||
Manuel Jacob
|
r50162 | version_regex = b'black, (?:version )?([0-9a-b.]+)' | ||
Augie Fackler
|
r43669 | version = matchoutput(blackcmd, version_regex) | ||
Mads Kiilerich
|
r52638 | if not version: | ||
return False | ||||
return Version(_bytes2sys(version.group(1))) >= Version('23.3.0') | ||||
Augie Fackler
|
r43771 | |||
@check('pytype', 'the pytype type checker') | ||||
def has_pytype(): | ||||
pytypecmd = 'pytype --version' | ||||
version = matchoutput(pytypecmd, b'[0-9a-b.]+') | ||||
Mads Kiilerich
|
r52638 | if not version: | ||
return False | ||||
return Version(_bytes2sys(version.group(0))) >= Version('2019.10.17') | ||||
Gregory Szorc
|
r44271 | |||
Raphaël Gomès
|
r52629 | @check("rustfmt", "rustfmt tool at version nightly-2024-07-16") | ||
Gregory Szorc
|
r44271 | def has_rustfmt(): | ||
# We use Nightly's rustfmt due to current unstable config options. | ||||
return matchoutput( | ||||
Raphaël Gomès
|
r52629 | '`rustup which --toolchain nightly-2024-07-16 rustfmt` --version', | ||
r46601 | b'rustfmt', | |||
Gregory Szorc
|
r44271 | ) | ||
Manuel Jacob
|
r45601 | |||
Raphaël Gomès
|
r46130 | @check("cargo", "cargo tool") | ||
def has_cargo(): | ||||
return matchoutput('`rustup which cargo` --version', b'cargo') | ||||
Manuel Jacob
|
r45601 | @check("lzma", "python lzma module") | ||
def has_lzma(): | ||||
try: | ||||
import _lzma | ||||
_lzma.FORMAT_XZ | ||||
return True | ||||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r48081 | |||
@check("bash", "bash shell") | ||||
def has_bash(): | ||||
return matchoutput("bash -c 'echo hi'", b'^hi$') | ||||
Julien Cristau
|
r49353 | |||
@check("bigendian", "big-endian CPU") | ||||
def has_bigendian(): | ||||
return sys.byteorder == 'big' | ||||