policy.py
116 lines
| 3.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 | ||||
# Rules for how modules can be loaded. Values are: | ||||
# | ||||
# c - require C extensions | ||||
# 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'), | ||
} | ||||
Maciej Fijalkowski
|
r29490 | |||
timeless
|
r29266 | try: | ||
from . import __modulepolicy__ | ||||
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 | |||
# Our C extensions aren't yet compatible with Python 3. So use pure Python | ||||
# on Python 3 for now. | ||||
if sys.version_info[0] >= 3: | ||||
Augie Fackler
|
r31308 | policy = b'py' | ||
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 | |||
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 | ||||
getattr(mod, r'__doc__', None) | ||||
return fakelocals[modname] | ||||
Jun Wu
|
r32428 | # keep in sync with "version" in C modules | ||
_cextversions = { | ||||
Yuya Nishihara
|
r32511 | (r'cext', r'base85'): 1, | ||
(r'cext', r'bdiff'): 1, | ||||
(r'cext', r'diffhelpers'): 1, | ||||
(r'cext', r'mpatch'): 1, | ||||
(r'cext', r'osutil'): 1, | ||||
Yuya Nishihara
|
r33927 | (r'cext', r'parsers'): 3, | ||
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'diffhelpers'): (r'pure', r'diffhelpers'), | ||
(r'cffi', r'parsers'): (r'pure', r'parsers'), | ||||
} | ||||
Yuya Nishihara
|
r32366 | def _checkmod(pkgname, modname, mod): | ||
Yuya Nishihara
|
r32511 | expected = _cextversions.get((pkgname, modname)) | ||
Yuya Nishihara
|
r32366 | actual = getattr(mod, r'version', None) | ||
if actual != expected: | ||||
raise ImportError(r'cannot import module %s.%s ' | ||||
r'(expected version: %d, actual: %r)' | ||||
% (pkgname, modname, expected, actual)) | ||||
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) | ||||