policy.py
157 lines
| 4.8 KiB
| text/x-python
|
PythonLexer
/ mercurial / policy.py
timeless
|
r29266 | # policy.py - module policy logic for Mercurial. | ||
# | ||||
# Copyright 2015 Gregory Szorc <gregory.szorc@gmail.com> | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from __future__ import absolute_import | ||||
import os | ||||
import sys | ||||
Gregory Szorc
|
r43359 | from .pycompat import getattr | ||
timeless
|
r29266 | # Rules for how modules can be loaded. Values are: | ||
# | ||||
# c - require C extensions | ||||
Georges Racinet
|
r42651 | # rust+c - require Rust and C extensions | ||
# rust+c-allow - allow Rust and C extensions with fallback to pure Python | ||||
# for each | ||||
timeless
|
r29266 | # allow - allow pure Python implementation when C loading fails | ||
Maciej Fijalkowski
|
r29490 | # cffi - required cffi versions (implemented within pure module) | ||
# cffi-allow - allow pure Python implementation if cffi version is missing | ||||
timeless
|
r29266 | # py - only load pure Python modules | ||
# | ||||
Yuya Nishihara
|
r32251 | # By default, fall back to the pure modules so the in-place build can | ||
# run without recompiling the C extensions. This will be overridden by | ||||
# __modulepolicy__ generated by setup.py. | ||||
policy = b'allow' | ||||
Yuya Nishihara
|
r32366 | _packageprefs = { | ||
# policy: (versioned package, pure package) | ||||
b'c': (r'cext', None), | ||||
b'allow': (r'cext', r'pure'), | ||||
Yuya Nishihara
|
r32512 | b'cffi': (r'cffi', None), | ||
b'cffi-allow': (r'cffi', r'pure'), | ||||
Yuya Nishihara
|
r32366 | b'py': (None, r'pure'), | ||
Georges Racinet
|
r42651 | # For now, rust policies impact importrust only | ||
b'rust+c': (r'cext', None), | ||||
b'rust+c-allow': (r'cext', r'pure'), | ||||
Yuya Nishihara
|
r32366 | } | ||
Maciej Fijalkowski
|
r29490 | |||
timeless
|
r29266 | try: | ||
from . import __modulepolicy__ | ||||
Augie Fackler
|
r43345 | |||
timeless
|
r29266 | policy = __modulepolicy__.modulepolicy | ||
except ImportError: | ||||
pass | ||||
# PyPy doesn't load C extensions. | ||||
# | ||||
# The canonical way to do this is to test platform.python_implementation(). | ||||
# But we don't import platform and don't bloat for it here. | ||||
Yuya Nishihara
|
r32205 | if r'__pypy__' in sys.builtin_module_names: | ||
policy = b'cffi' | ||||
timeless
|
r29266 | |||
# Environment variable can always force settings. | ||||
FUJIWARA Katsunori
|
r31361 | if sys.version_info[0] >= 3: | ||
Yuya Nishihara
|
r32205 | if r'HGMODULEPOLICY' in os.environ: | ||
policy = os.environ[r'HGMODULEPOLICY'].encode(r'utf-8') | ||||
FUJIWARA Katsunori
|
r31361 | else: | ||
Yuya Nishihara
|
r32205 | policy = os.environ.get(r'HGMODULEPOLICY', policy) | ||
Yuya Nishihara
|
r32366 | |||
Augie Fackler
|
r43345 | |||
Yuya Nishihara
|
r32366 | def _importfrom(pkgname, modname): | ||
# from .<pkgname> import <modname> (where . is looked through this module) | ||||
fakelocals = {} | ||||
pkg = __import__(pkgname, globals(), fakelocals, [modname], level=1) | ||||
try: | ||||
fakelocals[modname] = mod = getattr(pkg, modname) | ||||
except AttributeError: | ||||
raise ImportError(r'cannot import name %s' % modname) | ||||
# force import; fakelocals[modname] may be replaced with the real module | ||||
Gregory Szorc
|
r43373 | getattr(mod, '__doc__', None) | ||
Yuya Nishihara
|
r32366 | return fakelocals[modname] | ||
Augie Fackler
|
r43345 | |||
Jun Wu
|
r32428 | # keep in sync with "version" in C modules | ||
_cextversions = { | ||||
Yuya Nishihara
|
r32511 | (r'cext', r'base85'): 1, | ||
Jun Wu
|
r36693 | (r'cext', r'bdiff'): 3, | ||
Yuya Nishihara
|
r32511 | (r'cext', r'mpatch'): 1, | ||
Augie Fackler
|
r36798 | (r'cext', r'osutil'): 4, | ||
Martin von Zweigbergk
|
r42530 | (r'cext', r'parsers'): 13, | ||
Jun Wu
|
r32428 | } | ||
Yuya Nishihara
|
r33755 | # map import request to other package or module | ||
_modredirects = { | ||||
Yuya Nishihara
|
r33756 | (r'cext', r'charencode'): (r'cext', r'parsers'), | ||
Yuya Nishihara
|
r33755 | (r'cffi', r'base85'): (r'pure', r'base85'), | ||
Yuya Nishihara
|
r33756 | (r'cffi', r'charencode'): (r'pure', r'charencode'), | ||
Yuya Nishihara
|
r33755 | (r'cffi', r'parsers'): (r'pure', r'parsers'), | ||
} | ||||
Augie Fackler
|
r43345 | |||
Yuya Nishihara
|
r32366 | def _checkmod(pkgname, modname, mod): | ||
Yuya Nishihara
|
r32511 | expected = _cextversions.get((pkgname, modname)) | ||
Gregory Szorc
|
r43373 | actual = getattr(mod, 'version', None) | ||
Yuya Nishihara
|
r32366 | if actual != expected: | ||
Augie Fackler
|
r43345 | raise ImportError( | ||
r'cannot import module %s.%s ' | ||||
r'(expected version: %d, actual: %r)' | ||||
% (pkgname, modname, expected, actual) | ||||
) | ||||
Yuya Nishihara
|
r32366 | |||
def importmod(modname): | ||||
"""Import module according to policy and check API version""" | ||||
try: | ||||
verpkg, purepkg = _packageprefs[policy] | ||||
except KeyError: | ||||
raise ImportError(r'invalid HGMODULEPOLICY %r' % policy) | ||||
assert verpkg or purepkg | ||||
if verpkg: | ||||
Yuya Nishihara
|
r33755 | pn, mn = _modredirects.get((verpkg, modname), (verpkg, modname)) | ||
Yuya Nishihara
|
r32366 | try: | ||
Yuya Nishihara
|
r33755 | mod = _importfrom(pn, mn) | ||
if pn == verpkg: | ||||
_checkmod(pn, mn, mod) | ||||
Yuya Nishihara
|
r32366 | return mod | ||
except ImportError: | ||||
if not purepkg: | ||||
raise | ||||
Yuya Nishihara
|
r33755 | pn, mn = _modredirects.get((purepkg, modname), (purepkg, modname)) | ||
return _importfrom(pn, mn) | ||||
Georges Racinet
|
r42651 | |||
Augie Fackler
|
r43345 | |||
Georges Racinet
|
r42651 | def _isrustpermissive(): | ||
"""Assuming the policy is a Rust one, tell if it's permissive.""" | ||||
return policy.endswith(b'-allow') | ||||
Augie Fackler
|
r43345 | |||
Georges Racinet
|
r42651 | def importrust(modname, member=None, default=None): | ||
"""Import Rust module according to policy and availability. | ||||
If policy isn't a Rust one, this returns `default`. | ||||
If either the module or its member is not available, this returns `default` | ||||
if policy is permissive and raises `ImportError` if not. | ||||
""" | ||||
if not policy.startswith(b'rust'): | ||||
return default | ||||
try: | ||||
mod = _importfrom(r'rustext', modname) | ||||
except ImportError: | ||||
if _isrustpermissive(): | ||||
return default | ||||
raise | ||||
if member is None: | ||||
return mod | ||||
try: | ||||
return getattr(mod, member) | ||||
except AttributeError: | ||||
if _isrustpermissive(): | ||||
return default | ||||
raise ImportError(r"Cannot import name %s" % member) | ||||