##// END OF EJS Templates
policy: add helper to import cext/pure module...
Yuya Nishihara -
r32366:8e0327da default
parent child Browse files
Show More
@@ -1,51 +1,96 b''
1 1 # policy.py - module policy logic for Mercurial.
2 2 #
3 3 # Copyright 2015 Gregory Szorc <gregory.szorc@gmail.com>
4 4 #
5 5 # This software may be used and distributed according to the terms of the
6 6 # GNU General Public License version 2 or any later version.
7 7
8 8 from __future__ import absolute_import
9 9
10 10 import os
11 11 import sys
12 12
13 13 # Rules for how modules can be loaded. Values are:
14 14 #
15 15 # c - require C extensions
16 16 # allow - allow pure Python implementation when C loading fails
17 17 # cffi - required cffi versions (implemented within pure module)
18 18 # cffi-allow - allow pure Python implementation if cffi version is missing
19 19 # py - only load pure Python modules
20 20 #
21 21 # By default, fall back to the pure modules so the in-place build can
22 22 # run without recompiling the C extensions. This will be overridden by
23 23 # __modulepolicy__ generated by setup.py.
24 24 policy = b'allow'
25 25 policynoc = (b'cffi', b'cffi-allow', b'py')
26 26 policynocffi = (b'c', b'py')
27 _packageprefs = {
28 # policy: (versioned package, pure package)
29 b'c': (r'cext', None),
30 b'allow': (r'cext', r'pure'),
31 b'cffi': (None, r'pure'), # TODO: (r'cffi', None)
32 b'cffi-allow': (None, r'pure'), # TODO: (r'cffi', r'pure')
33 b'py': (None, r'pure'),
34 }
27 35
28 36 try:
29 37 from . import __modulepolicy__
30 38 policy = __modulepolicy__.modulepolicy
31 39 except ImportError:
32 40 pass
33 41
34 42 # PyPy doesn't load C extensions.
35 43 #
36 44 # The canonical way to do this is to test platform.python_implementation().
37 45 # But we don't import platform and don't bloat for it here.
38 46 if r'__pypy__' in sys.builtin_module_names:
39 47 policy = b'cffi'
40 48
41 49 # Our C extensions aren't yet compatible with Python 3. So use pure Python
42 50 # on Python 3 for now.
43 51 if sys.version_info[0] >= 3:
44 52 policy = b'py'
45 53
46 54 # Environment variable can always force settings.
47 55 if sys.version_info[0] >= 3:
48 56 if r'HGMODULEPOLICY' in os.environ:
49 57 policy = os.environ[r'HGMODULEPOLICY'].encode(r'utf-8')
50 58 else:
51 59 policy = os.environ.get(r'HGMODULEPOLICY', policy)
60
61 def _importfrom(pkgname, modname):
62 # from .<pkgname> import <modname> (where . is looked through this module)
63 fakelocals = {}
64 pkg = __import__(pkgname, globals(), fakelocals, [modname], level=1)
65 try:
66 fakelocals[modname] = mod = getattr(pkg, modname)
67 except AttributeError:
68 raise ImportError(r'cannot import name %s' % modname)
69 # force import; fakelocals[modname] may be replaced with the real module
70 getattr(mod, r'__doc__', None)
71 return fakelocals[modname]
72
73 def _checkmod(pkgname, modname, mod):
74 expected = 1 # TODO: maybe defined in table?
75 actual = getattr(mod, r'version', None)
76 if actual != expected:
77 raise ImportError(r'cannot import module %s.%s '
78 r'(expected version: %d, actual: %r)'
79 % (pkgname, modname, expected, actual))
80
81 def importmod(modname):
82 """Import module according to policy and check API version"""
83 try:
84 verpkg, purepkg = _packageprefs[policy]
85 except KeyError:
86 raise ImportError(r'invalid HGMODULEPOLICY %r' % policy)
87 assert verpkg or purepkg
88 if verpkg:
89 try:
90 mod = _importfrom(verpkg, modname)
91 _checkmod(verpkg, modname, mod)
92 return mod
93 except ImportError:
94 if not purepkg:
95 raise
96 return _importfrom(purepkg, modname)
General Comments 0
You need to be logged in to leave comments. Login now