|
|
|
import sys |
|
import os |
|
|
|
|
|
is_pypy = '__pypy__' in sys.builtin_module_names |
|
|
|
|
|
def warn_distutils_present(): |
|
if 'distutils' not in sys.modules: |
|
return |
|
if is_pypy and sys.version_info < (3, 7): |
|
|
|
|
|
return |
|
import warnings |
|
|
|
warnings.warn( |
|
"Distutils was imported before Setuptools, but importing Setuptools " |
|
"also replaces the `distutils` module in `sys.modules`. This may lead " |
|
"to undesirable behaviors or errors. To avoid these issues, avoid " |
|
"using distutils directly, ensure that setuptools is installed in the " |
|
"traditional way (e.g. not an editable install), and/or make sure " |
|
"that setuptools is always imported before distutils." |
|
) |
|
|
|
|
|
def clear_distutils(): |
|
if 'distutils' not in sys.modules: |
|
return |
|
import warnings |
|
|
|
warnings.warn("Setuptools is replacing distutils.") |
|
mods = [ |
|
name |
|
for name in sys.modules |
|
if name == "distutils" or name.startswith("distutils.") |
|
] |
|
for name in mods: |
|
del sys.modules[name] |
|
|
|
|
|
def enabled(): |
|
""" |
|
Allow selection of distutils by environment variable. |
|
""" |
|
which = os.environ.get('SETUPTOOLS_USE_DISTUTILS', 'local') |
|
return which == 'local' |
|
|
|
|
|
def ensure_local_distutils(): |
|
import importlib |
|
|
|
clear_distutils() |
|
|
|
|
|
|
|
|
|
with shim(): |
|
importlib.import_module('distutils') |
|
|
|
|
|
core = importlib.import_module('distutils.core') |
|
assert '_distutils' in core.__file__, core.__file__ |
|
assert 'setuptools._distutils.log' not in sys.modules |
|
|
|
|
|
def do_override(): |
|
""" |
|
Ensure that the local copy of distutils is preferred over stdlib. |
|
|
|
See https://github.com/pypa/setuptools/issues/417#issuecomment-392298401 |
|
for more motivation. |
|
""" |
|
if enabled(): |
|
warn_distutils_present() |
|
ensure_local_distutils() |
|
|
|
|
|
class _TrivialRe: |
|
def __init__(self, *patterns): |
|
self._patterns = patterns |
|
|
|
def match(self, string): |
|
return all(pat in string for pat in self._patterns) |
|
|
|
|
|
class DistutilsMetaFinder: |
|
def find_spec(self, fullname, path, target=None): |
|
|
|
|
|
if path is not None and not fullname.startswith('test.'): |
|
return |
|
|
|
method_name = 'spec_for_{fullname}'.format(**locals()) |
|
method = getattr(self, method_name, lambda: None) |
|
return method() |
|
|
|
def spec_for_distutils(self): |
|
if self.is_cpython(): |
|
return |
|
|
|
import importlib |
|
import importlib.abc |
|
import importlib.util |
|
|
|
try: |
|
mod = importlib.import_module('setuptools._distutils') |
|
except Exception: |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return |
|
|
|
class DistutilsLoader(importlib.abc.Loader): |
|
def create_module(self, spec): |
|
mod.__name__ = 'distutils' |
|
return mod |
|
|
|
def exec_module(self, module): |
|
pass |
|
|
|
return importlib.util.spec_from_loader( |
|
'distutils', DistutilsLoader(), origin=mod.__file__ |
|
) |
|
|
|
@staticmethod |
|
def is_cpython(): |
|
""" |
|
Suppress supplying distutils for CPython (build and tests). |
|
Ref #2965 and #3007. |
|
""" |
|
return os.path.isfile('pybuilddir.txt') |
|
|
|
def spec_for_pip(self): |
|
""" |
|
Ensure stdlib distutils when running under pip. |
|
See pypa/pip#8761 for rationale. |
|
""" |
|
if sys.version_info >= (3, 12) or self.pip_imported_during_build(): |
|
return |
|
clear_distutils() |
|
self.spec_for_distutils = lambda: None |
|
|
|
@classmethod |
|
def pip_imported_during_build(cls): |
|
""" |
|
Detect if pip is being imported in a build script. Ref #2355. |
|
""" |
|
import traceback |
|
|
|
return any( |
|
cls.frame_file_is_setup(frame) for frame, line in traceback.walk_stack(None) |
|
) |
|
|
|
@staticmethod |
|
def frame_file_is_setup(frame): |
|
""" |
|
Return True if the indicated frame suggests a setup.py file. |
|
""" |
|
|
|
return frame.f_globals.get('__file__', '').endswith('setup.py') |
|
|
|
def spec_for_sensitive_tests(self): |
|
""" |
|
Ensure stdlib distutils when running select tests under CPython. |
|
|
|
python/cpython#91169 |
|
""" |
|
clear_distutils() |
|
self.spec_for_distutils = lambda: None |
|
|
|
sensitive_tests = ( |
|
[ |
|
'test.test_distutils', |
|
'test.test_peg_generator', |
|
'test.test_importlib', |
|
] |
|
if sys.version_info < (3, 10) |
|
else [ |
|
'test.test_distutils', |
|
] |
|
) |
|
|
|
|
|
for name in DistutilsMetaFinder.sensitive_tests: |
|
setattr( |
|
DistutilsMetaFinder, |
|
f'spec_for_{name}', |
|
DistutilsMetaFinder.spec_for_sensitive_tests, |
|
) |
|
|
|
|
|
DISTUTILS_FINDER = DistutilsMetaFinder() |
|
|
|
|
|
def add_shim(): |
|
DISTUTILS_FINDER in sys.meta_path or insert_shim() |
|
|
|
|
|
class shim: |
|
def __enter__(self): |
|
insert_shim() |
|
|
|
def __exit__(self, exc, value, tb): |
|
_remove_shim() |
|
|
|
|
|
def insert_shim(): |
|
sys.meta_path.insert(0, DISTUTILS_FINDER) |
|
|
|
|
|
def _remove_shim(): |
|
try: |
|
sys.meta_path.remove(DISTUTILS_FINDER) |
|
except ValueError: |
|
pass |
|
|
|
|
|
if sys.version_info < (3, 12): |
|
|
|
remove_shim = _remove_shim |
|
|