|
import os |
|
import sys |
|
import distutils.command.build_ext as orig |
|
from distutils.sysconfig import get_config_var |
|
from importlib.util import cache_from_source as _compiled_file_name |
|
|
|
from jaraco import path |
|
|
|
from setuptools.command.build_ext import build_ext, get_abi3_suffix |
|
from setuptools.dist import Distribution |
|
from setuptools.extension import Extension |
|
from setuptools.errors import CompileError |
|
|
|
from . import environment |
|
from .textwrap import DALS |
|
|
|
import pytest |
|
|
|
|
|
IS_PYPY = '__pypy__' in sys.builtin_module_names |
|
|
|
|
|
class TestBuildExt: |
|
def test_get_ext_filename(self): |
|
""" |
|
Setuptools needs to give back the same |
|
result as distutils, even if the fullname |
|
is not in ext_map. |
|
""" |
|
dist = Distribution() |
|
cmd = build_ext(dist) |
|
cmd.ext_map['foo/bar'] = '' |
|
res = cmd.get_ext_filename('foo') |
|
wanted = orig.build_ext.get_ext_filename(cmd, 'foo') |
|
assert res == wanted |
|
|
|
def test_abi3_filename(self): |
|
""" |
|
Filename needs to be loadable by several versions |
|
of Python 3 if 'is_abi3' is truthy on Extension() |
|
""" |
|
print(get_abi3_suffix()) |
|
|
|
extension = Extension('spam.eggs', ['eggs.c'], py_limited_api=True) |
|
dist = Distribution(dict(ext_modules=[extension])) |
|
cmd = build_ext(dist) |
|
cmd.finalize_options() |
|
assert 'spam.eggs' in cmd.ext_map |
|
res = cmd.get_ext_filename('spam.eggs') |
|
|
|
if not get_abi3_suffix(): |
|
assert res.endswith(get_config_var('EXT_SUFFIX')) |
|
elif sys.platform == 'win32': |
|
assert res.endswith('eggs.pyd') |
|
else: |
|
assert 'abi3' in res |
|
|
|
def test_ext_suffix_override(self): |
|
""" |
|
SETUPTOOLS_EXT_SUFFIX variable always overrides |
|
default extension options. |
|
""" |
|
dist = Distribution() |
|
cmd = build_ext(dist) |
|
cmd.ext_map['for_abi3'] = ext = Extension( |
|
'for_abi3', |
|
['s.c'], |
|
|
|
py_limited_api=True, |
|
) |
|
|
|
ext._links_to_dynamic = False |
|
|
|
if not IS_PYPY: |
|
expect = cmd.get_ext_filename('for_abi3') |
|
else: |
|
|
|
|
|
expect = 'for_abi3.test-suffix' |
|
|
|
try: |
|
os.environ['SETUPTOOLS_EXT_SUFFIX'] = '.test-suffix' |
|
res = cmd.get_ext_filename('normal') |
|
assert 'normal.test-suffix' == res |
|
res = cmd.get_ext_filename('for_abi3') |
|
assert expect == res |
|
finally: |
|
del os.environ['SETUPTOOLS_EXT_SUFFIX'] |
|
|
|
def dist_with_example(self): |
|
files = { |
|
"src": {"mypkg": {"subpkg": {"ext2.c": ""}}}, |
|
"c-extensions": {"ext1": {"main.c": ""}}, |
|
} |
|
|
|
ext1 = Extension("mypkg.ext1", ["c-extensions/ext1/main.c"]) |
|
ext2 = Extension("mypkg.subpkg.ext2", ["src/mypkg/subpkg/ext2.c"]) |
|
ext3 = Extension("ext3", ["c-extension/ext3.c"]) |
|
|
|
path.build(files) |
|
return Distribution({ |
|
"script_name": "%test%", |
|
"ext_modules": [ext1, ext2, ext3], |
|
"package_dir": {"": "src"}, |
|
}) |
|
|
|
def test_get_outputs(self, tmpdir_cwd, monkeypatch): |
|
monkeypatch.setenv('SETUPTOOLS_EXT_SUFFIX', '.mp3') |
|
monkeypatch.setattr('setuptools.command.build_ext.use_stubs', False) |
|
dist = self.dist_with_example() |
|
|
|
|
|
build_ext = dist.get_command_obj("build_ext") |
|
build_ext.editable_mode = False |
|
build_ext.ensure_finalized() |
|
build_lib = build_ext.build_lib.replace(os.sep, "/") |
|
outputs = [x.replace(os.sep, "/") for x in build_ext.get_outputs()] |
|
assert outputs == [ |
|
f"{build_lib}/ext3.mp3", |
|
f"{build_lib}/mypkg/ext1.mp3", |
|
f"{build_lib}/mypkg/subpkg/ext2.mp3", |
|
] |
|
assert build_ext.get_output_mapping() == {} |
|
|
|
|
|
dist.reinitialize_command("build_ext") |
|
build_ext.editable_mode = True |
|
build_ext.ensure_finalized() |
|
mapping = { |
|
k.replace(os.sep, "/"): v.replace(os.sep, "/") |
|
for k, v in build_ext.get_output_mapping().items() |
|
} |
|
assert mapping == { |
|
f"{build_lib}/ext3.mp3": "src/ext3.mp3", |
|
f"{build_lib}/mypkg/ext1.mp3": "src/mypkg/ext1.mp3", |
|
f"{build_lib}/mypkg/subpkg/ext2.mp3": "src/mypkg/subpkg/ext2.mp3", |
|
} |
|
|
|
def test_get_output_mapping_with_stub(self, tmpdir_cwd, monkeypatch): |
|
monkeypatch.setenv('SETUPTOOLS_EXT_SUFFIX', '.mp3') |
|
monkeypatch.setattr('setuptools.command.build_ext.use_stubs', True) |
|
dist = self.dist_with_example() |
|
|
|
|
|
build_ext = dist.get_command_obj("build_ext") |
|
build_ext.editable_mode = True |
|
build_ext.ensure_finalized() |
|
for ext in build_ext.extensions: |
|
monkeypatch.setattr(ext, "_needs_stub", True) |
|
|
|
build_lib = build_ext.build_lib.replace(os.sep, "/") |
|
mapping = { |
|
k.replace(os.sep, "/"): v.replace(os.sep, "/") |
|
for k, v in build_ext.get_output_mapping().items() |
|
} |
|
|
|
def C(file): |
|
"""Make it possible to do comparisons and tests in a OS-independent way""" |
|
return _compiled_file_name(file).replace(os.sep, "/") |
|
|
|
assert mapping == { |
|
C(f"{build_lib}/ext3.py"): C("src/ext3.py"), |
|
f"{build_lib}/ext3.mp3": "src/ext3.mp3", |
|
C(f"{build_lib}/mypkg/ext1.py"): C("src/mypkg/ext1.py"), |
|
f"{build_lib}/mypkg/ext1.mp3": "src/mypkg/ext1.mp3", |
|
C(f"{build_lib}/mypkg/subpkg/ext2.py"): C("src/mypkg/subpkg/ext2.py"), |
|
f"{build_lib}/mypkg/subpkg/ext2.mp3": "src/mypkg/subpkg/ext2.mp3", |
|
} |
|
|
|
|
|
assert f"{build_lib}/mypkg/ext1.py" not in mapping |
|
assert f"{build_lib}/mypkg/subpkg/ext2.py" not in mapping |
|
|
|
|
|
example_stub = C(f"{build_lib}/mypkg/ext1.py") |
|
assert example_stub in mapping |
|
assert example_stub.startswith(f"{build_lib}/mypkg/__pycache__/ext1") |
|
assert example_stub.endswith(".pyc") |
|
|
|
|
|
class TestBuildExtInplace: |
|
def get_build_ext_cmd(self, optional: bool, **opts): |
|
files = { |
|
"eggs.c": "#include missingheader.h\n", |
|
".build": {"lib": {}, "tmp": {}}, |
|
} |
|
path.build(files) |
|
extension = Extension('spam.eggs', ['eggs.c'], optional=optional) |
|
dist = Distribution(dict(ext_modules=[extension])) |
|
dist.script_name = 'setup.py' |
|
cmd = build_ext(dist) |
|
vars(cmd).update(build_lib=".build/lib", build_temp=".build/tmp", **opts) |
|
cmd.ensure_finalized() |
|
return cmd |
|
|
|
def get_log_messages(self, caplog, capsys): |
|
""" |
|
Historically, distutils "logged" by printing to sys.std*. |
|
Later versions adopted the logging framework. Grab |
|
messages regardless of how they were captured. |
|
""" |
|
std = capsys.readouterr() |
|
return std.out.splitlines() + std.err.splitlines() + caplog.messages |
|
|
|
def test_optional(self, tmpdir_cwd, caplog, capsys): |
|
""" |
|
If optional extensions fail to build, setuptools should show the error |
|
in the logs but not fail to build |
|
""" |
|
cmd = self.get_build_ext_cmd(optional=True, inplace=True) |
|
cmd.run() |
|
assert any( |
|
'build_ext: building extension "spam.eggs" failed' |
|
for msg in self.get_log_messages(caplog, capsys) |
|
) |
|
|
|
|
|
def test_non_optional(self, tmpdir_cwd): |
|
|
|
cmd = self.get_build_ext_cmd(optional=False, inplace=True) |
|
with pytest.raises(CompileError): |
|
cmd.run() |
|
|
|
|
|
def test_build_ext_config_handling(tmpdir_cwd): |
|
files = { |
|
'setup.py': DALS( |
|
""" |
|
from setuptools import Extension, setup |
|
setup( |
|
name='foo', |
|
version='0.0.0', |
|
ext_modules=[Extension('foo', ['foo.c'])], |
|
) |
|
""" |
|
), |
|
'foo.c': DALS( |
|
""" |
|
#include "Python.h" |
|
|
|
#if PY_MAJOR_VERSION >= 3 |
|
|
|
static struct PyModuleDef moduledef = { |
|
PyModuleDef_HEAD_INIT, |
|
"foo", |
|
NULL, |
|
0, |
|
NULL, |
|
NULL, |
|
NULL, |
|
NULL, |
|
NULL |
|
}; |
|
|
|
#define INITERROR return NULL |
|
|
|
PyMODINIT_FUNC PyInit_foo(void) |
|
|
|
#else |
|
|
|
#define INITERROR return |
|
|
|
void initfoo(void) |
|
|
|
#endif |
|
{ |
|
#if PY_MAJOR_VERSION >= 3 |
|
PyObject *module = PyModule_Create(&moduledef); |
|
#else |
|
PyObject *module = Py_InitModule("extension", NULL); |
|
#endif |
|
if (module == NULL) |
|
INITERROR; |
|
#if PY_MAJOR_VERSION >= 3 |
|
return module; |
|
#endif |
|
} |
|
""" |
|
), |
|
'setup.cfg': DALS( |
|
""" |
|
[build] |
|
build_base = foo_build |
|
""" |
|
), |
|
} |
|
path.build(files) |
|
code, output = environment.run_setup_py( |
|
cmd=['build'], |
|
data_stream=(0, 2), |
|
) |
|
assert code == 0, '\nSTDOUT:\n%s\nSTDERR:\n%s' % output |
|
|