| # SPDX-License-Identifier: Apache-2.0 |
| # Copyright 2021 The Meson development team |
| # Copyright © 2024 Intel Corporation |
| |
| from __future__ import annotations |
| import json |
| import os |
| import pickle |
| import tempfile |
| import subprocess |
| import textwrap |
| import shutil |
| from unittest import skipIf, SkipTest |
| from pathlib import Path |
| |
| from .baseplatformtests import BasePlatformTests |
| from .helpers import is_ci |
| from mesonbuild.mesonlib import EnvironmentVariables, ExecutableSerialisation, MesonException, is_linux, python_command |
| from mesonbuild.mformat import match_path |
| from mesonbuild.optinterpreter import OptionInterpreter, OptionException |
| from run_tests import Backend |
| |
| @skipIf(is_ci() and not is_linux(), "Run only on fast platforms") |
| class PlatformAgnosticTests(BasePlatformTests): |
| ''' |
| Tests that does not need to run on all platforms during CI |
| ''' |
| |
| def test_relative_find_program(self): |
| ''' |
| Tests that find_program() with a relative path does not find the program |
| in current workdir. |
| ''' |
| testdir = os.path.join(self.unit_test_dir, '101 relative find program') |
| self.init(testdir, workdir=testdir) |
| |
| def test_invalid_option_names(self): |
| interp = OptionInterpreter('') |
| |
| def write_file(code: str): |
| with tempfile.NamedTemporaryFile('w', dir=self.builddir, encoding='utf-8', delete=False) as f: |
| f.write(code) |
| return f.name |
| |
| fname = write_file("option('default_library', type: 'string')") |
| self.assertRaisesRegex(OptionException, 'Option name default_library is reserved.', |
| interp.process, fname) |
| |
| fname = write_file("option('c_anything', type: 'string')") |
| self.assertRaisesRegex(OptionException, 'Option name c_anything is reserved.', |
| interp.process, fname) |
| |
| fname = write_file("option('b_anything', type: 'string')") |
| self.assertRaisesRegex(OptionException, 'Option name b_anything is reserved.', |
| interp.process, fname) |
| |
| fname = write_file("option('backend_anything', type: 'string')") |
| self.assertRaisesRegex(OptionException, 'Option name backend_anything is reserved.', |
| interp.process, fname) |
| |
| fname = write_file("option('foo.bar', type: 'string')") |
| self.assertRaisesRegex(OptionException, 'Option names can only contain letters, numbers or dashes.', |
| interp.process, fname) |
| |
| # platlib is allowed, only python.platlib is reserved. |
| fname = write_file("option('platlib', type: 'string')") |
| interp.process(fname) |
| |
| def test_option_validation(self): |
| """Test cases that are not catch by the optinterpreter itself.""" |
| interp = OptionInterpreter('') |
| |
| def write_file(code: str): |
| with tempfile.NamedTemporaryFile('w', dir=self.builddir, encoding='utf-8', delete=False) as f: |
| f.write(code) |
| return f.name |
| |
| fname = write_file("option('intminmax', type: 'integer', value: 10, min: 0, max: 5)") |
| self.assertRaisesRegex(MesonException, 'Value 10 for option "intminmax" is more than maximum value 5.', |
| interp.process, fname) |
| |
| fname = write_file("option('array', type: 'array', choices : ['one', 'two', 'three'], value : ['one', 'four'])") |
| self.assertRaisesRegex(MesonException, 'Value "four" for option "array" is not in allowed choices: "one, two, three"', |
| interp.process, fname) |
| |
| fname = write_file("option('array', type: 'array', choices : ['one', 'two', 'three'], value : ['four', 'five', 'six'])") |
| self.assertRaisesRegex(MesonException, 'Values "four, five, six" for option "array" are not in allowed choices: "one, two, three"', |
| interp.process, fname) |
| |
| def test_python_dependency_without_pkgconfig(self): |
| testdir = os.path.join(self.unit_test_dir, '103 python without pkgconfig') |
| self.init(testdir, override_envvars={'PKG_CONFIG': 'notfound'}) |
| |
| def test_debug_function_outputs_to_meson_log(self): |
| testdir = os.path.join(self.unit_test_dir, '105 debug function') |
| log_msg = 'This is an example debug output, should only end up in debug log' |
| output = self.init(testdir) |
| |
| # Check if message is not printed to stdout while configuring |
| self.assertNotIn(log_msg, output) |
| |
| # Check if message is written to the meson log |
| mesonlog = self.get_meson_log_raw() |
| self.assertIn(log_msg, mesonlog) |
| |
| def test_new_subproject_reconfigure(self): |
| testdir = os.path.join(self.unit_test_dir, '108 new subproject on reconfigure') |
| self.init(testdir) |
| self.build() |
| |
| # Enable the subproject "foo" and reconfigure, this is used to fail |
| # because per-subproject builtin options were not initialized: |
| # https://github.com/mesonbuild/meson/issues/10225. |
| self.setconf('-Dfoo=enabled') |
| self.build('reconfigure') |
| |
| def check_connectivity(self): |
| import urllib |
| try: |
| with urllib.request.urlopen('https://wrapdb.mesonbuild.com') as p: |
| pass |
| except urllib.error.URLError as e: |
| self.skipTest('No internet connectivity: ' + str(e)) |
| |
| def test_update_wrapdb(self): |
| self.check_connectivity() |
| # Write the project into a temporary directory because it will add files |
| # into subprojects/ and we don't want to pollute meson source tree. |
| with tempfile.TemporaryDirectory() as testdir: |
| with Path(testdir, 'meson.build').open('w', encoding='utf-8') as f: |
| f.write(textwrap.dedent( |
| ''' |
| project('wrap update-db', |
| default_options: ['wrap_mode=forcefallback']) |
| |
| zlib_dep = dependency('zlib') |
| assert(zlib_dep.type_name() == 'internal') |
| ''')) |
| subprocess.check_call(self.wrap_command + ['update-db'], cwd=testdir) |
| self.init(testdir, workdir=testdir) |
| |
| def test_none_backend(self): |
| testdir = os.path.join(self.python_test_dir, '7 install path') |
| |
| self.init(testdir, extra_args=['--backend=none'], override_envvars={'NINJA': 'absolutely false command'}) |
| self.assertPathDoesNotExist(os.path.join(self.builddir, 'build.ninja')) |
| |
| self.run_tests(inprocess=True, override_envvars={}) |
| |
| out = self._run(self.meson_command + ['install', f'--destdir={self.installdir}'], workdir=self.builddir) |
| self.assertNotIn('Only ninja backend is supported to rebuild the project before installation.', out) |
| |
| with open(os.path.join(testdir, 'test.json'), 'rb') as f: |
| dat = json.load(f) |
| for i in dat['installed']: |
| self.assertPathExists(os.path.join(self.installdir, i['file'])) |
| |
| def test_change_backend(self): |
| if self.backend != Backend.ninja: |
| raise SkipTest('Only useful to test if backend is ninja.') |
| |
| testdir = os.path.join(self.python_test_dir, '7 install path') |
| self.init(testdir) |
| |
| # no-op change works |
| self.setconf(f'--backend=ninja') |
| self.init(testdir, extra_args=['--reconfigure', '--backend=ninja']) |
| |
| # Change backend option is not allowed |
| with self.assertRaises(subprocess.CalledProcessError) as cm: |
| self.setconf('-Dbackend=none') |
| self.assertIn("ERROR: Tried modify read only option 'backend'", cm.exception.stdout) |
| |
| # Reconfigure with a different backend is not allowed |
| with self.assertRaises(subprocess.CalledProcessError) as cm: |
| self.init(testdir, extra_args=['--reconfigure', '--backend=none']) |
| self.assertIn("ERROR: Tried modify read only option 'backend'", cm.exception.stdout) |
| |
| # Wipe with a different backend is allowed |
| self.init(testdir, extra_args=['--wipe', '--backend=none']) |
| |
| def test_validate_dirs(self): |
| testdir = os.path.join(self.common_test_dir, '1 trivial') |
| |
| # Using parent as builddir should fail |
| self.builddir = os.path.dirname(self.builddir) |
| with self.assertRaises(subprocess.CalledProcessError) as cm: |
| self.init(testdir) |
| self.assertIn('cannot be a parent of source directory', cm.exception.stdout) |
| |
| # Reconfigure of empty builddir should work |
| self.new_builddir() |
| self.init(testdir, extra_args=['--reconfigure']) |
| |
| # Reconfigure of not empty builddir should work |
| self.new_builddir() |
| Path(self.builddir, 'dummy').touch() |
| self.init(testdir, extra_args=['--reconfigure']) |
| |
| # Setup a valid builddir should update options but not reconfigure |
| self.assertEqual(self.getconf('buildtype'), 'debug') |
| o = self.init(testdir, extra_args=['-Dbuildtype=release']) |
| self.assertIn('Directory already configured', o) |
| self.assertNotIn('The Meson build system', o) |
| self.assertEqual(self.getconf('buildtype'), 'release') |
| |
| # Wipe of empty builddir should work |
| self.new_builddir() |
| self.init(testdir, extra_args=['--wipe']) |
| |
| # Wipe of partial builddir should work |
| self.new_builddir() |
| Path(self.builddir, 'meson-private').mkdir() |
| Path(self.builddir, 'dummy').touch() |
| self.init(testdir, extra_args=['--wipe']) |
| |
| # Wipe of not empty builddir should fail |
| self.new_builddir() |
| Path(self.builddir, 'dummy').touch() |
| with self.assertRaises(subprocess.CalledProcessError) as cm: |
| self.init(testdir, extra_args=['--wipe']) |
| self.assertIn('Directory is not empty', cm.exception.stdout) |
| |
| def test_scripts_loaded_modules(self): |
| ''' |
| Simulate a wrapped command, as done for custom_target() that capture |
| output. The script will print all python modules loaded and we verify |
| that it contains only an acceptable subset. Loading too many modules |
| slows down the build when many custom targets get wrapped. |
| |
| This list must not be edited without a clear rationale for why it is |
| acceptable to do so! |
| ''' |
| es = ExecutableSerialisation(python_command + ['-c', 'exit(0)'], env=EnvironmentVariables()) |
| p = Path(self.builddir, 'exe.dat') |
| with p.open('wb') as f: |
| pickle.dump(es, f) |
| cmd = self.meson_command + ['--internal', 'test_loaded_modules', '--unpickle', str(p)] |
| p = subprocess.run(cmd, stdout=subprocess.PIPE) |
| all_modules = json.loads(p.stdout.splitlines()[0]) |
| meson_modules = [m for m in all_modules if m.startswith('mesonbuild')] |
| expected_meson_modules = [ |
| 'mesonbuild', |
| 'mesonbuild._pathlib', |
| 'mesonbuild.utils', |
| 'mesonbuild.utils.core', |
| 'mesonbuild.mesonmain', |
| 'mesonbuild.mlog', |
| 'mesonbuild.scripts', |
| 'mesonbuild.scripts.meson_exe', |
| 'mesonbuild.scripts.test_loaded_modules' |
| ] |
| self.assertEqual(sorted(expected_meson_modules), sorted(meson_modules)) |
| |
| def test_setup_loaded_modules(self): |
| ''' |
| Execute a very basic meson.build and capture a list of all python |
| modules loaded. We verify that it contains only an acceptable subset. |
| Loading too many modules slows down `meson setup` startup time and |
| gives a perception that meson is slow. |
| |
| Adding more modules to the default startup flow is not an unreasonable |
| thing to do as new features are added, but keeping track of them is |
| good. |
| ''' |
| testdir = os.path.join(self.unit_test_dir, '116 empty project') |
| |
| self.init(testdir) |
| self._run(self.meson_command + ['--internal', 'regenerate', '--profile-self', testdir, self.builddir]) |
| with open(os.path.join(self.builddir, 'meson-logs', 'profile-startup-modules.json'), encoding='utf-8') as f: |
| data = json.load(f)['meson'] |
| |
| with open(os.path.join(testdir, 'expected_mods.json'), encoding='utf-8') as f: |
| expected = json.load(f)['meson']['modules'] |
| |
| self.assertEqual(data['modules'], expected) |
| self.assertEqual(data['count'], 70) |
| |
| def test_meson_package_cache_dir(self): |
| # Copy testdir into temporary directory to not pollute meson source tree. |
| testdir = os.path.join(self.unit_test_dir, '118 meson package cache dir') |
| srcdir = os.path.join(self.builddir, 'srctree') |
| shutil.copytree(testdir, srcdir) |
| builddir = os.path.join(srcdir, '_build') |
| self.change_builddir(builddir) |
| self.init(srcdir, override_envvars={'MESON_PACKAGE_CACHE_DIR': os.path.join(srcdir, 'cache_dir')}) |
| |
| def test_cmake_openssl_not_found_bug(self): |
| """Issue #12098""" |
| testdir = os.path.join(self.unit_test_dir, '119 openssl cmake bug') |
| self.meson_native_files.append(os.path.join(testdir, 'nativefile.ini')) |
| out = self.init(testdir, allow_fail=True) |
| self.assertNotIn('Unhandled python exception', out) |
| |
| def test_editorconfig_match_path(self): |
| '''match_path function used to parse editorconfig in meson format''' |
| cases = [ |
| ('a.txt', '*.txt', True), |
| ('a.txt', '?.txt', True), |
| ('a.txt', 'a.t?t', True), |
| ('a.txt', '*.build', False), |
| |
| ('/a.txt', '*.txt', True), |
| ('/a.txt', '/*.txt', True), |
| ('a.txt', '/*.txt', False), |
| |
| ('a/b/c.txt', 'a/b/*.txt', True), |
| ('a/b/c.txt', 'a/*/*.txt', True), |
| ('a/b/c.txt', '*/*.txt', True), |
| ('a/b/c.txt', 'b/*.txt', True), |
| ('a/b/c.txt', 'a/*.txt', False), |
| |
| ('a/b/c/d.txt', 'a/**/*.txt', True), |
| ('a/b/c/d.txt', 'a/*', False), |
| ('a/b/c/d.txt', 'a/**', True), |
| |
| ('a.txt', '[abc].txt', True), |
| ('a.txt', '[!xyz].txt', True), |
| ('a.txt', '[xyz].txt', False), |
| ('a.txt', '[!abc].txt', False), |
| |
| ('a.txt', '{a,b,c}.txt', True), |
| ('a.txt', '*.{txt,tex,cpp}', True), |
| ('a.hpp', '*.{txt,tex,cpp}', False), |
| |
| ('a1.txt', 'a{0..9}.txt', True), |
| ('a001.txt', 'a{0..9}.txt', True), |
| ('a-1.txt', 'a{-10..10}.txt', True), |
| ('a99.txt', 'a{0..9}.txt', False), |
| ('a099.txt', 'a{0..9}.txt', False), |
| ('a-1.txt', 'a{0..10}.txt', False), |
| ] |
| |
| for filename, pattern, expected in cases: |
| self.assertTrue(match_path(filename, pattern) is expected, f'{filename} -> {pattern}') |
| |
| def test_error_configuring_subdir(self): |
| testdir = os.path.join(self.common_test_dir, '152 index customtarget') |
| out = self.init(os.path.join(testdir, 'subdir'), allow_fail=True) |
| |
| self.assertIn('first statement must be a call to project()', out) |
| # provide guidance diagnostics by finding a file whose first AST statement is project() |
| self.assertIn(f'Did you mean to run meson from the directory: "{testdir}"?', out) |
| |
| def test_reconfigure_base_options(self): |
| testdir = os.path.join(self.unit_test_dir, '122 reconfigure base options') |
| out = self.init(testdir, extra_args=['-Db_ndebug=true']) |
| self.assertIn('\nMessage: b_ndebug: true\n', out) |
| self.assertIn('\nMessage: c_std: c89\n', out) |
| |
| out = self.init(testdir, extra_args=['--reconfigure', '-Db_ndebug=if-release', '-Dsub:b_ndebug=false', '-Dc_std=c99', '-Dsub:c_std=c11']) |
| self.assertIn('\nMessage: b_ndebug: if-release\n', out) |
| self.assertIn('\nMessage: c_std: c99\n', out) |
| self.assertIn('\nsub| Message: b_ndebug: false\n', out) |
| self.assertIn('\nsub| Message: c_std: c11\n', out) |
| |
| def test_setup_with_unknown_option(self): |
| testdir = os.path.join(self.common_test_dir, '1 trivial') |
| |
| for option in ('not_an_option', 'b_not_an_option'): |
| out = self.init(testdir, extra_args=['--wipe', f'-D{option}=1'], allow_fail=True) |
| self.assertIn(f'ERROR: Unknown options: "{option}"', out) |
| |
| def test_configure_new_option(self) -> None: |
| """Adding a new option without reconfiguring should work.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '40 options')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'meson_options.txt'), 'a', encoding='utf-8') as f: |
| f.write("option('new_option', type : 'boolean', value : false)") |
| self.setconf('-Dnew_option=true') |
| self.assertEqual(self.getconf('new_option'), True) |
| |
| def test_configure_removed_option(self) -> None: |
| """Removing an options without reconfiguring should still give an error.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '40 options')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'meson_options.txt'), 'r', encoding='utf-8') as f: |
| opts = f.readlines() |
| with open(os.path.join(testdir, 'meson_options.txt'), 'w', encoding='utf-8') as f: |
| for line in opts: |
| if line.startswith("option('neg'"): |
| continue |
| f.write(line) |
| with self.assertRaises(subprocess.CalledProcessError) as e: |
| self.setconf('-Dneg_int_opt=0') |
| self.assertIn('Unknown options: "neg_int_opt"', e.exception.stdout) |
| |
| def test_configure_option_changed_constraints(self) -> None: |
| """Changing the constraints of an option without reconfiguring should work.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '40 options')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'meson_options.txt'), 'r', encoding='utf-8') as f: |
| opts = f.readlines() |
| with open(os.path.join(testdir, 'meson_options.txt'), 'w', encoding='utf-8') as f: |
| for line in opts: |
| if line.startswith("option('neg'"): |
| f.write("option('neg_int_opt', type : 'integer', min : -10, max : 10, value : -3)\n") |
| else: |
| f.write(line) |
| self.setconf('-Dneg_int_opt=-10') |
| self.assertEqual(self.getconf('neg_int_opt'), -10) |
| |
| def test_configure_meson_options_txt_to_meson_options(self) -> None: |
| """Changing from a meson_options.txt to meson.options should still be detected.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '40 options')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'meson_options.txt'), 'r', encoding='utf-8') as f: |
| opts = f.readlines() |
| with open(os.path.join(testdir, 'meson_options.txt'), 'w', encoding='utf-8') as f: |
| for line in opts: |
| if line.startswith("option('neg'"): |
| f.write("option('neg_int_opt', type : 'integer', min : -10, max : 10, value : -3)\n") |
| else: |
| f.write(line) |
| shutil.move(os.path.join(testdir, 'meson_options.txt'), os.path.join(testdir, 'meson.options')) |
| self.setconf('-Dneg_int_opt=-10') |
| self.assertEqual(self.getconf('neg_int_opt'), -10) |
| |
| def test_configure_options_file_deleted(self) -> None: |
| """Deleting all option files should make seting a project option an error.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '40 options')) |
| self.init(testdir) |
| os.unlink(os.path.join(testdir, 'meson_options.txt')) |
| with self.assertRaises(subprocess.CalledProcessError) as e: |
| self.setconf('-Dneg_int_opt=0') |
| self.assertIn('Unknown options: "neg_int_opt"', e.exception.stdout) |
| |
| def test_configure_options_file_added(self) -> None: |
| """A new project option file should be detected.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '1 trivial')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'meson.options'), 'w', encoding='utf-8') as f: |
| f.write("option('new_option', type : 'string', value : 'foo')") |
| self.setconf('-Dnew_option=bar') |
| self.assertEqual(self.getconf('new_option'), 'bar') |
| |
| def test_configure_options_file_added_old(self) -> None: |
| """A new project option file should be detected.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '1 trivial')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'meson_options.txt'), 'w', encoding='utf-8') as f: |
| f.write("option('new_option', type : 'string', value : 'foo')") |
| self.setconf('-Dnew_option=bar') |
| self.assertEqual(self.getconf('new_option'), 'bar') |
| |
| def test_configure_new_option_subproject(self) -> None: |
| """Adding a new option to a subproject without reconfiguring should work.""" |
| testdir = self.copy_srcdir(os.path.join(self.common_test_dir, '43 subproject options')) |
| self.init(testdir) |
| with open(os.path.join(testdir, 'subprojects/subproject/meson_options.txt'), 'a', encoding='utf-8') as f: |
| f.write("option('new_option', type : 'boolean', value : false)") |
| self.setconf('-Dsubproject:new_option=true') |
| self.assertEqual(self.getconf('subproject:new_option'), True) |