Commit 0a665b2a authored by Jason R. Coombs's avatar Jason R. Coombs
parents 03d36b9e 2888d3c0
...@@ -16,6 +16,8 @@ for older versions in distutils.msvc9compiler and distutils.msvccompiler. ...@@ -16,6 +16,8 @@ for older versions in distutils.msvc9compiler and distutils.msvccompiler.
import os import os
import subprocess import subprocess
import contextlib import contextlib
import warnings
import unittest.mock
with contextlib.suppress(ImportError): with contextlib.suppress(ImportError):
import winreg import winreg
...@@ -504,7 +506,29 @@ class MSVCCompiler(CCompiler) : ...@@ -504,7 +506,29 @@ class MSVCCompiler(CCompiler) :
def spawn(self, cmd): def spawn(self, cmd):
env = dict(os.environ, PATH=self._paths) env = dict(os.environ, PATH=self._paths)
return super().spawn(cmd, env=env) with self._fallback_spawn(cmd, env) as fallback:
return super().spawn(cmd, env=env)
return fallback.value
@contextlib.contextmanager
def _fallback_spawn(self, cmd, env):
"""
Discovered in pypa/distutils#15, some tools monkeypatch the compiler,
so the 'env' kwarg causes a TypeError. Detect this condition and
restore the legacy, unsafe behavior.
"""
bag = type('Bag', (), {})()
try:
yield bag
except TypeError as exc:
if "unexpected keyword argument 'env'" not in str(exc):
raise
else:
return
warnings.warn(
"Fallback spawn triggered. Please update distutils monkeypatch.")
with unittest.mock.patch('os.environ', env):
bag.value = super().spawn(cmd)
# -- Miscellaneous methods ----------------------------------------- # -- Miscellaneous methods -----------------------------------------
# These are all used by the 'gen_lib_options() function, in # These are all used by the 'gen_lib_options() function, in
......
...@@ -110,6 +110,26 @@ class TestSpawn(unittest.TestCase): ...@@ -110,6 +110,26 @@ class TestSpawn(unittest.TestCase):
thread.join() thread.join()
assert all(threads) assert all(threads)
def test_concurrent_safe_fallback(self):
"""
If CCompiler.spawn has been monkey-patched without support
for an env, it should still execute.
"""
import distutils._msvccompiler as _msvccompiler
from distutils import ccompiler
compiler = _msvccompiler.MSVCCompiler()
compiler._paths = "expected"
def CCompiler_spawn(self, cmd):
"A spawn without an env argument."
assert os.environ["PATH"] == "expected"
with unittest.mock.patch.object(
ccompiler.CCompiler, 'spawn', CCompiler_spawn):
compiler.spawn(["n/a"])
assert os.environ.get("PATH") != "expected"
def test_suite(): def test_suite():
return unittest.makeSuite(msvccompilerTestCase) return unittest.makeSuite(msvccompilerTestCase)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment