| # SPDX-License-Identifier: Apache-2.0 |
| # Copyright 2016-2021 The Meson development team |
| |
| import glob, os, pathlib, shutil, subprocess, unittest |
| |
| from run_tests import ( |
| Backend |
| ) |
| |
| from .allplatformstests import git_init |
| from .baseplatformtests import BasePlatformTests |
| from .helpers import * |
| |
| from mesonbuild.mesonlib import MachineChoice, TemporaryDirectoryWinProof, is_windows |
| from mesonbuild.modules.python import PythonModule |
| |
| class PythonTests(BasePlatformTests): |
| ''' |
| Tests that verify compilation of python extension modules |
| ''' |
| |
| def test_bad_versions(self): |
| if self.backend is not Backend.ninja: |
| raise unittest.SkipTest(f'Skipping python tests with {self.backend.name} backend') |
| |
| testdir = os.path.join(self.src_root, 'test cases', 'python', '8 different python versions') |
| |
| # The test is configured to error out with MESON_SKIP_TEST |
| # in case it could not find python |
| with self.assertRaises(unittest.SkipTest): |
| self.init(testdir, extra_args=['-Dpython=not-python']) |
| self.wipe() |
| |
| # While dir is an external command on both Windows and Linux, |
| # it certainly isn't python |
| with self.assertRaises(unittest.SkipTest): |
| self.init(testdir, extra_args=['-Dpython=dir']) |
| self.wipe() |
| |
| def test_dist(self): |
| with TemporaryDirectoryWinProof() as dirstr: |
| dirobj = pathlib.Path(dirstr) |
| mesonfile = dirobj / 'meson.build' |
| mesonfile.write_text('''project('test', 'c', version: '1') |
| pymod = import('python') |
| python = pymod.find_installation('python3', required: true) |
| ''', encoding='utf-8') |
| git_init(dirstr) |
| self.init(dirstr) |
| subprocess.check_call(self.meson_command + ['dist', '-C', self.builddir], stdout=subprocess.DEVNULL) |
| |
| def _test_bytecompile(self, py2=False): |
| testdir = os.path.join(self.src_root, 'test cases', 'python', '2 extmodule') |
| |
| env = get_fake_env(testdir, self.builddir, self.prefix) |
| cc = detect_c_compiler(env, MachineChoice.HOST) |
| |
| self.init(testdir, extra_args=['-Dpython2=auto', '-Dpython.bytecompile=1']) |
| self.build() |
| self.install() |
| |
| count = 0 |
| for root, dirs, files in os.walk(self.installdir): |
| for file in files: |
| realfile = os.path.join(root, file) |
| if file.endswith('.py'): |
| cached = glob.glob(realfile+'?') + glob.glob(os.path.join(root, '__pycache__', os.path.splitext(file)[0] + '*.pyc')) |
| if py2 and cc.get_id() == 'msvc': |
| # MSVC python installs python2/python3 into the same directory |
| self.assertLength(cached, 4) |
| else: |
| self.assertLength(cached, 2) |
| count += 1 |
| # there are 5 files x 2 installations |
| if py2 and not cc.get_id() == 'msvc': |
| self.assertEqual(count, 10) |
| else: |
| self.assertEqual(count, 5) |
| |
| def test_bytecompile_multi(self): |
| if not shutil.which('python2') and not PythonModule._get_win_pythonpath('python2'): |
| raise self.skipTest('python2 not installed') |
| self._test_bytecompile(True) |
| |
| def test_bytecompile_single(self): |
| if shutil.which('python2') or PythonModule._get_win_pythonpath('python2'): |
| raise self.skipTest('python2 installed, already tested') |
| self._test_bytecompile() |
| |
| def test_limited_api_linked_correct_lib(self): |
| if not is_windows(): |
| return self.skipTest('Test only run on Windows.') |
| |
| testdir = os.path.join(self.src_root, 'test cases', 'python', '9 extmodule limited api') |
| |
| self.init(testdir) |
| self.build() |
| |
| from importlib.machinery import EXTENSION_SUFFIXES |
| limited_suffix = EXTENSION_SUFFIXES[1] |
| |
| limited_library_path = os.path.join(self.builddir, f'limited{limited_suffix}') |
| self.assertPathExists(limited_library_path) |
| |
| limited_dep_name = 'python3.dll' |
| if shutil.which('dumpbin'): |
| # MSVC |
| output = subprocess.check_output(['dumpbin', '/DEPENDENTS', limited_library_path], |
| stderr=subprocess.STDOUT) |
| self.assertIn(limited_dep_name, output.decode()) |
| elif shutil.which('objdump'): |
| # mingw |
| output = subprocess.check_output(['objdump', '-p', limited_library_path], |
| stderr=subprocess.STDOUT) |
| self.assertIn(limited_dep_name, output.decode()) |
| else: |
| raise self.skipTest('Test needs either dumpbin(MSVC) or objdump(mingw).') |