test_sysconfig.py 16.5 KB
Newer Older
1 2 3
import unittest
import sys
import os
4
import subprocess
Georg Brandl's avatar
Georg Brandl committed
5
import shutil
6
from copy import copy
7

8
from test.support import (run_unittest, TESTFN, unlink, check_warnings,
9
                          captured_stdout, skip_unless_symlink)
10 11 12

import sysconfig
from sysconfig import (get_paths, get_platform, get_config_vars,
13
                       get_path, get_path_names, _INSTALL_SCHEMES,
14
                       _get_default_scheme, _expand_vars,
15
                       get_scheme_names, get_config_var, _main)
16
import _osx_support
17 18 19 20 21 22 23 24 25 26 27 28

class TestSysConfig(unittest.TestCase):

    def setUp(self):
        super(TestSysConfig, self).setUp()
        self.sys_path = sys.path[:]
        # patching os.uname
        if hasattr(os, 'uname'):
            self.uname = os.uname
            self._uname = os.uname()
        else:
            self.uname = None
29
            self._set_uname(('',)*5)
30 31 32 33 34 35 36 37 38
        os.uname = self._get_uname
        # saving the environment
        self.name = os.name
        self.platform = sys.platform
        self.version = sys.version
        self.sep = os.sep
        self.join = os.path.join
        self.isabs = os.path.isabs
        self.splitdrive = os.path.splitdrive
39
        self._config_vars = sysconfig._CONFIG_VARS, copy(sysconfig._CONFIG_VARS)
40 41
        self._added_envvars = []
        self._changed_envvars = []
42
        for var in ('MACOSX_DEPLOYMENT_TARGET', 'PATH'):
43 44 45 46
            if var in os.environ:
                self._changed_envvars.append((var, os.environ[var]))
            else:
                self._added_envvars.append(var)
47 48 49 50 51 52 53 54 55 56 57 58 59 60 61

    def tearDown(self):
        sys.path[:] = self.sys_path
        self._cleanup_testfn()
        if self.uname is not None:
            os.uname = self.uname
        else:
            del os.uname
        os.name = self.name
        sys.platform = self.platform
        sys.version = self.version
        os.sep = self.sep
        os.path.join = self.join
        os.path.isabs = self.isabs
        os.path.splitdrive = self.splitdrive
62 63 64
        sysconfig._CONFIG_VARS = self._config_vars[0]
        sysconfig._CONFIG_VARS.clear()
        sysconfig._CONFIG_VARS.update(self._config_vars[1])
65 66 67 68
        for var, value in self._changed_envvars:
            os.environ[var] = value
        for var in self._added_envvars:
            os.environ.pop(var, None)
69 70 71 72

        super(TestSysConfig, self).tearDown()

    def _set_uname(self, uname):
73
        self._uname = os.uname_result(uname)
74 75 76 77 78 79 80 81 82 83 84 85

    def _get_uname(self):
        return self._uname

    def _cleanup_testfn(self):
        path = TESTFN
        if os.path.isfile(path):
            os.remove(path)
        elif os.path.isdir(path):
            shutil.rmtree(path)

    def test_get_path_names(self):
86
        self.assertEqual(get_path_names(), sysconfig._SCHEME_KEYS)
87 88 89 90 91

    def test_get_paths(self):
        scheme = get_paths()
        default_scheme = _get_default_scheme()
        wanted = _expand_vars(default_scheme, None)
92 93
        wanted = sorted(wanted.items())
        scheme = sorted(scheme.items())
94
        self.assertEqual(scheme, wanted)
95 96

    def test_get_path(self):
97
        # XXX make real tests here
98 99
        for scheme in _INSTALL_SCHEMES:
            for name in _INSTALL_SCHEMES[scheme]:
100 101 102 103
                res = get_path(name, scheme)

    def test_get_config_vars(self):
        cvars = get_config_vars()
104
        self.assertIsInstance(cvars, dict)
105 106 107 108 109 110 111 112
        self.assertTrue(cvars)

    def test_get_platform(self):
        # windows XP, 32bits
        os.name = 'nt'
        sys.version = ('2.4.4 (#71, Oct 18 2006, 08:34:43) '
                       '[MSC v.1310 32 bit (Intel)]')
        sys.platform = 'win32'
113
        self.assertEqual(get_platform(), 'win32')
114 115 116 117 118 119

        # windows XP, amd64
        os.name = 'nt'
        sys.version = ('2.4.4 (#71, Oct 18 2006, 08:34:43) '
                       '[MSC v.1310 32 bit (Amd64)]')
        sys.platform = 'win32'
120
        self.assertEqual(get_platform(), 'win-amd64')
121 122 123 124 125 126

        # windows XP, itanium
        os.name = 'nt'
        sys.version = ('2.4.4 (#71, Oct 18 2006, 08:34:43) '
                       '[MSC v.1310 32 bit (Itanium)]')
        sys.platform = 'win32'
127
        self.assertEqual(get_platform(), 'win-ia64')
128 129 130 131 132 133

        # macbook
        os.name = 'posix'
        sys.version = ('2.5 (r25:51918, Sep 19 2006, 08:49:13) '
                       '\n[GCC 4.0.1 (Apple Computer, Inc. build 5341)]')
        sys.platform = 'darwin'
134 135 136 137
        self._set_uname(('Darwin', 'macziade', '8.11.1',
                   ('Darwin Kernel Version 8.11.1: '
                    'Wed Oct 10 18:23:28 PDT 2007; '
                    'root:xnu-792.25.20~1/RELEASE_I386'), 'PowerPC'))
138
        _osx_support._remove_original_values(get_config_vars())
139
        get_config_vars()['MACOSX_DEPLOYMENT_TARGET'] = '10.3'
140 141 142 143

        get_config_vars()['CFLAGS'] = ('-fno-strict-aliasing -DNDEBUG -g '
                                       '-fwrapv -O3 -Wall -Wstrict-prototypes')

144
        maxint = sys.maxsize
145 146
        try:
            sys.maxsize = 2147483647
147
            self.assertEqual(get_platform(), 'macosx-10.3-ppc')
148
            sys.maxsize = 9223372036854775807
149
            self.assertEqual(get_platform(), 'macosx-10.3-ppc64')
150
        finally:
151
            sys.maxsize = maxint
152

153 154 155 156
        self._set_uname(('Darwin', 'macziade', '8.11.1',
                   ('Darwin Kernel Version 8.11.1: '
                    'Wed Oct 10 18:23:28 PDT 2007; '
                    'root:xnu-792.25.20~1/RELEASE_I386'), 'i386'))
157
        _osx_support._remove_original_values(get_config_vars())
158
        get_config_vars()['MACOSX_DEPLOYMENT_TARGET'] = '10.3'
159 160 161

        get_config_vars()['CFLAGS'] = ('-fno-strict-aliasing -DNDEBUG -g '
                                       '-fwrapv -O3 -Wall -Wstrict-prototypes')
162
        maxint = sys.maxsize
163 164
        try:
            sys.maxsize = 2147483647
165
            self.assertEqual(get_platform(), 'macosx-10.3-i386')
166
            sys.maxsize = 9223372036854775807
167
            self.assertEqual(get_platform(), 'macosx-10.3-x86_64')
168
        finally:
169
            sys.maxsize = maxint
170 171

        # macbook with fat binaries (fat, universal or fat64)
172
        _osx_support._remove_original_values(get_config_vars())
173
        get_config_vars()['MACOSX_DEPLOYMENT_TARGET'] = '10.4'
174 175 176 177 178
        get_config_vars()['CFLAGS'] = ('-arch ppc -arch i386 -isysroot '
                                       '/Developer/SDKs/MacOSX10.4u.sdk  '
                                       '-fno-strict-aliasing -fno-common '
                                       '-dynamic -DNDEBUG -g -O3')

179
        self.assertEqual(get_platform(), 'macosx-10.4-fat')
180

181
        _osx_support._remove_original_values(get_config_vars())
182 183 184 185 186
        get_config_vars()['CFLAGS'] = ('-arch x86_64 -arch i386 -isysroot '
                                       '/Developer/SDKs/MacOSX10.4u.sdk  '
                                       '-fno-strict-aliasing -fno-common '
                                       '-dynamic -DNDEBUG -g -O3')

187
        self.assertEqual(get_platform(), 'macosx-10.4-intel')
188

189
        _osx_support._remove_original_values(get_config_vars())
190 191 192 193
        get_config_vars()['CFLAGS'] = ('-arch x86_64 -arch ppc -arch i386 -isysroot '
                                       '/Developer/SDKs/MacOSX10.4u.sdk  '
                                       '-fno-strict-aliasing -fno-common '
                                       '-dynamic -DNDEBUG -g -O3')
194
        self.assertEqual(get_platform(), 'macosx-10.4-fat3')
195

196
        _osx_support._remove_original_values(get_config_vars())
197 198 199 200
        get_config_vars()['CFLAGS'] = ('-arch ppc64 -arch x86_64 -arch ppc -arch i386 -isysroot '
                                       '/Developer/SDKs/MacOSX10.4u.sdk  '
                                       '-fno-strict-aliasing -fno-common '
                                       '-dynamic -DNDEBUG -g -O3')
201
        self.assertEqual(get_platform(), 'macosx-10.4-universal')
202

203
        _osx_support._remove_original_values(get_config_vars())
204 205 206 207 208
        get_config_vars()['CFLAGS'] = ('-arch x86_64 -arch ppc64 -isysroot '
                                       '/Developer/SDKs/MacOSX10.4u.sdk  '
                                       '-fno-strict-aliasing -fno-common '
                                       '-dynamic -DNDEBUG -g -O3')

209
        self.assertEqual(get_platform(), 'macosx-10.4-fat64')
210 211

        for arch in ('ppc', 'i386', 'x86_64', 'ppc64'):
212
            _osx_support._remove_original_values(get_config_vars())
213 214 215
            get_config_vars()['CFLAGS'] = ('-arch %s -isysroot '
                                           '/Developer/SDKs/MacOSX10.4u.sdk  '
                                           '-fno-strict-aliasing -fno-common '
Éric Araujo's avatar
Éric Araujo committed
216
                                           '-dynamic -DNDEBUG -g -O3' % arch)
217

Éric Araujo's avatar
Éric Araujo committed
218
            self.assertEqual(get_platform(), 'macosx-10.4-%s' % arch)
219 220 221 222 223 224 225 226 227

        # linux debian sarge
        os.name = 'posix'
        sys.version = ('2.3.5 (#1, Jul  4 2007, 17:28:59) '
                       '\n[GCC 4.1.2 20061115 (prerelease) (Debian 4.1.1-21)]')
        sys.platform = 'linux2'
        self._set_uname(('Linux', 'aglae', '2.6.21.1dedibox-r7',
                    '#1 Mon Apr 30 17:25:38 CEST 2007', 'i686'))

228
        self.assertEqual(get_platform(), 'linux-i686')
229 230 231 232 233 234 235

        # XXX more platforms to tests here

    def test_get_config_h_filename(self):
        config_h = sysconfig.get_config_h_filename()
        self.assertTrue(os.path.isfile(config_h), config_h)

236
    def test_get_scheme_names(self):
237
        wanted = ('nt', 'nt_user', 'osx_framework_user',
238
                  'posix_home', 'posix_prefix', 'posix_user')
239
        self.assertEqual(get_scheme_names(), wanted)
240

241
    @skip_unless_symlink
242
    def test_symlink(self):
243 244 245
        # On Windows, the EXE needs to know where pythonXY.dll is at so we have
        # to add the directory to the path.
        if sys.platform == "win32":
246 247
            os.environ["PATH"] = "{};{}".format(
                os.path.dirname(sys.executable), os.environ["PATH"])
248

249 250 251
        # Issue 7880
        def get(python):
            cmd = [python, '-c',
252
                   'import sysconfig; print(sysconfig.get_platform())']
253
            p = subprocess.Popen(cmd, stdout=subprocess.PIPE, env=os.environ)
254 255 256
            return p.communicate()
        real = os.path.realpath(sys.executable)
        link = os.path.abspath(TESTFN)
257
        os.symlink(real, link)
258 259 260 261 262
        try:
            self.assertEqual(get(real), get(link))
        finally:
            unlink(link)

263
    def test_user_similar(self):
264
        # Issue #8759: make sure the posix scheme for the users
265 266 267
        # is similar to the global posix_prefix one
        base = get_config_var('base')
        user = get_config_var('userbase')
268 269 270
        # the global scheme mirrors the distinction between prefix and
        # exec-prefix but not the user scheme, so we have to adapt the paths
        # before comparing (issue #9100)
271
        adapt = sys.base_prefix != sys.base_exec_prefix
272 273
        for name in ('stdlib', 'platstdlib', 'purelib', 'platlib'):
            global_path = get_path(name, 'posix_prefix')
274
            if adapt:
275 276 277 278 279 280 281
                global_path = global_path.replace(sys.exec_prefix, sys.base_prefix)
                base = base.replace(sys.exec_prefix, sys.base_prefix)
            elif sys.base_prefix != sys.prefix:
                # virtual environment? Likewise, we have to adapt the paths
                # before comparing
                global_path = global_path.replace(sys.base_prefix, sys.prefix)
                base = base.replace(sys.base_prefix, sys.prefix)
282
            user_path = get_path(name, 'posix_user')
283
            self.assertEqual(user_path, global_path.replace(base, user, 1))
284

285 286 287 288 289 290
    def test_main(self):
        # just making sure _main() runs and returns things in the stdout
        with captured_stdout() as output:
            _main()
        self.assertTrue(len(output.getvalue().split('\n')) > 0)

291
    @unittest.skipIf(sys.platform == "win32", "Does not apply to Windows")
292 293 294 295 296 297
    def test_ldshared_value(self):
        ldflags = sysconfig.get_config_var('LDFLAGS')
        ldshared = sysconfig.get_config_var('LDSHARED')

        self.assertIn(ldflags, ldshared)

298 299 300 301 302 303 304 305 306 307
    @unittest.skipUnless(sys.platform == "darwin", "test only relevant on MacOSX")
    def test_platform_in_subprocess(self):
        my_platform = sysconfig.get_platform()

        # Test without MACOSX_DEPLOYMENT_TARGET in the environment

        env = os.environ.copy()
        if 'MACOSX_DEPLOYMENT_TARGET' in env:
            del env['MACOSX_DEPLOYMENT_TARGET']

308 309 310 311 312 313 314
        p = subprocess.Popen([
                sys.executable, '-c',
                'import sysconfig; print(sysconfig.get_platform())',
            ],
            stdout=subprocess.PIPE,
            stderr=subprocess.DEVNULL,
            env=env)
315 316 317 318 319 320 321 322 323 324 325 326
        test_platform = p.communicate()[0].strip()
        test_platform = test_platform.decode('utf-8')
        status = p.wait()

        self.assertEqual(status, 0)
        self.assertEqual(my_platform, test_platform)

        # Test with MACOSX_DEPLOYMENT_TARGET in the environment, and
        # using a value that is unlikely to be the default one.
        env = os.environ.copy()
        env['MACOSX_DEPLOYMENT_TARGET'] = '10.1'

327 328 329 330 331 332 333 334 335 336 337 338 339
        p = subprocess.Popen([
                sys.executable, '-c',
                'import sysconfig; print(sysconfig.get_platform())',
            ],
            stdout=subprocess.PIPE,
            stderr=subprocess.DEVNULL,
            env=env)
        test_platform = p.communicate()[0].strip()
        test_platform = test_platform.decode('utf-8')
        status = p.wait()

        self.assertEqual(status, 0)
        self.assertEqual(my_platform, test_platform)
340

341 342 343 344 345 346 347 348 349 350 351 352 353 354
    def test_srcdir(self):
        # See Issues #15322, #15364.
        srcdir = sysconfig.get_config_var('srcdir')

        self.assertTrue(os.path.isabs(srcdir), srcdir)
        self.assertTrue(os.path.isdir(srcdir), srcdir)

        if sysconfig._PYTHON_BUILD:
            # The python executable has not been installed so srcdir
            # should be a full source checkout.
            Python_h = os.path.join(srcdir, 'Include', 'Python.h')
            self.assertTrue(os.path.exists(Python_h), Python_h)
            self.assertTrue(sysconfig._is_python_source_dir(srcdir))
        elif os.name == 'posix':
355 356 357 358
            makefile_dir = os.path.dirname(sysconfig.get_makefile_filename())
            # Issue #19340: srcdir has been realpath'ed already
            makefile_dir = os.path.realpath(makefile_dir)
            self.assertEqual(makefile_dir, srcdir)
359 360 361 362 363 364 365 366 367 368 369 370 371

    def test_srcdir_independent_of_cwd(self):
        # srcdir should be independent of the current working directory
        # See Issues #15322, #15364.
        srcdir = sysconfig.get_config_var('srcdir')
        cwd = os.getcwd()
        try:
            os.chdir('..')
            srcdir2 = sysconfig.get_config_var('srcdir')
        finally:
            os.chdir(cwd)
        self.assertEqual(srcdir, srcdir2)

372 373 374 375 376 377 378 379 380
    @unittest.skipIf(sysconfig.get_config_var('EXT_SUFFIX') is None,
                     'EXT_SUFFIX required for this test')
    def test_SO_deprecation(self):
        self.assertWarns(DeprecationWarning,
                         sysconfig.get_config_var, 'SO')

    @unittest.skipIf(sysconfig.get_config_var('EXT_SUFFIX') is None,
                     'EXT_SUFFIX required for this test')
    def test_SO_value(self):
381 382 383
        with check_warnings(('', DeprecationWarning)):
            self.assertEqual(sysconfig.get_config_var('SO'),
                             sysconfig.get_config_var('EXT_SUFFIX'))
384 385 386 387 388 389 390 391

    @unittest.skipIf(sysconfig.get_config_var('EXT_SUFFIX') is None,
                     'EXT_SUFFIX required for this test')
    def test_SO_in_vars(self):
        vars = sysconfig.get_config_vars()
        self.assertIsNotNone(vars['SO'])
        self.assertEqual(vars['SO'], vars['EXT_SUFFIX'])

392

393
class MakefileTests(unittest.TestCase):
Éric Araujo's avatar
Éric Araujo committed
394

395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
    @unittest.skipIf(sys.platform.startswith('win'),
                     'Test is not Windows compatible')
    def test_get_makefile_filename(self):
        makefile = sysconfig.get_makefile_filename()
        self.assertTrue(os.path.isfile(makefile), makefile)

    def test_parse_makefile(self):
        self.addCleanup(unlink, TESTFN)
        with open(TESTFN, "w") as makefile:
            print("var1=a$(VAR2)", file=makefile)
            print("VAR2=b$(var3)", file=makefile)
            print("var3=42", file=makefile)
            print("var4=$/invalid", file=makefile)
            print("var5=dollar$$5", file=makefile)
        vars = sysconfig._parse_makefile(TESTFN)
        self.assertEqual(vars, {
            'var1': 'ab42',
            'VAR2': 'b42',
            'var3': 42,
            'var4': '$/invalid',
            'var5': 'dollar$5',
        })
417 418


419
def test_main():
420
    run_unittest(TestSysConfig, MakefileTests)
421 422 423

if __name__ == "__main__":
    test_main()