Commit 64110315 authored by Tarek Ziadé's avatar Tarek Ziadé

Fixed #7556: editing the MSVC manifest file with a regexp was throwing an error

parent 9414e0dd
...@@ -645,6 +645,18 @@ class MSVCCompiler(CCompiler) : ...@@ -645,6 +645,18 @@ class MSVCCompiler(CCompiler) :
mfid = 1 mfid = 1
else: else:
mfid = 2 mfid = 2
# Remove references to the Visual C runtime
self._remove_visual_c_ref(temp_manifest)
out_arg = '-outputresource:%s;%s' % (output_filename, mfid)
try:
self.spawn(['mt.exe', '-nologo', '-manifest',
temp_manifest, out_arg])
except DistutilsExecError as msg:
raise LinkError(msg)
else:
log.debug("skipping %s (up-to-date)", output_filename)
def _remove_visual_c_ref(self, manifest_file):
try: try:
# Remove references to the Visual C runtime, so they will # Remove references to the Visual C runtime, so they will
# fall through to the Visual C dependency of Python.exe. # fall through to the Visual C dependency of Python.exe.
...@@ -652,8 +664,10 @@ class MSVCCompiler(CCompiler) : ...@@ -652,8 +664,10 @@ class MSVCCompiler(CCompiler) :
# runtimes are not in WinSxS folder, but in Python's own # runtimes are not in WinSxS folder, but in Python's own
# folder), the runtimes do not need to be in every folder # folder), the runtimes do not need to be in every folder
# with .pyd's. # with .pyd's.
manifest_f = open(temp_manifest, "rb") manifest_f = open(manifest_file)
try:
manifest_buf = manifest_f.read() manifest_buf = manifest_f.read()
finally:
manifest_f.close() manifest_f.close()
pattern = re.compile( pattern = re.compile(
r"""<assemblyIdentity.*?name=("|')Microsoft\."""\ r"""<assemblyIdentity.*?name=("|')Microsoft\."""\
...@@ -662,20 +676,13 @@ class MSVCCompiler(CCompiler) : ...@@ -662,20 +676,13 @@ class MSVCCompiler(CCompiler) :
manifest_buf = re.sub(pattern, "", manifest_buf) manifest_buf = re.sub(pattern, "", manifest_buf)
pattern = "<dependentAssembly>\s*</dependentAssembly>" pattern = "<dependentAssembly>\s*</dependentAssembly>"
manifest_buf = re.sub(pattern, "", manifest_buf) manifest_buf = re.sub(pattern, "", manifest_buf)
manifest_f = open(temp_manifest, "wb") manifest_f = open(manifest_file, 'w')
try:
manifest_f.write(manifest_buf) manifest_f.write(manifest_buf)
finally:
manifest_f.close() manifest_f.close()
except IOError: except IOError:
pass pass
out_arg = '-outputresource:%s;%s' % (output_filename, mfid)
try:
self.spawn(['mt.exe', '-nologo', '-manifest',
temp_manifest, out_arg])
except DistutilsExecError as msg:
raise LinkError(msg)
else:
log.debug("skipping %s (up-to-date)", output_filename)
# -- 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
......
"""Tests for distutils.msvc9compiler.""" """Tests for distutils.msvc9compiler."""
import sys import sys
import unittest import unittest
import os
from distutils.errors import DistutilsPlatformError from distutils.errors import DistutilsPlatformError
from distutils.tests import support
class msvc9compilerTestCase(unittest.TestCase): _MANIFEST = """\
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1"
manifestVersion="1.0">
<trustInfo xmlns="urn:schemas-microsoft-com:asm.v3">
<security>
<requestedPrivileges>
<requestedExecutionLevel level="asInvoker" uiAccess="false">
</requestedExecutionLevel>
</requestedPrivileges>
</security>
</trustInfo>
<dependency>
<dependentAssembly>
<assemblyIdentity type="win32" name="Microsoft.VC90.CRT"
version="9.0.21022.8" processorArchitecture="x86"
publicKeyToken="XXXX">
</assemblyIdentity>
</dependentAssembly>
</dependency>
<dependency>
<dependentAssembly>
<assemblyIdentity type="win32" name="Microsoft.VC90.MFC"
version="9.0.21022.8" processorArchitecture="x86"
publicKeyToken="XXXX"></assemblyIdentity>
</dependentAssembly>
</dependency>
</assembly>
"""
_CLEANED_MANIFEST = """\
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1"
manifestVersion="1.0">
<trustInfo xmlns="urn:schemas-microsoft-com:asm.v3">
<security>
<requestedPrivileges>
<requestedExecutionLevel level="asInvoker" uiAccess="false">
</requestedExecutionLevel>
</requestedPrivileges>
</security>
</trustInfo>
<dependency>
</dependency>
<dependency>
<dependentAssembly>
<assemblyIdentity type="win32" name="Microsoft.VC90.MFC"
version="9.0.21022.8" processorArchitecture="x86"
publicKeyToken="XXXX"></assemblyIdentity>
</dependentAssembly>
</dependency>
</assembly>"""
@unittest.skip("These tests are only for win32")
class msvc9compilerTestCase(support.TempdirManager,
unittest.TestCase):
def test_no_compiler(self): def test_no_compiler(self):
# makes sure query_vcvarsall throws # makes sure query_vcvarsall throws
# a DistutilsPlatformError if the compiler # a DistutilsPlatformError if the compiler
# is not found # is not found
if sys.platform != 'win32':
# this test is only for win32
return
from distutils.msvccompiler import get_build_version from distutils.msvccompiler import get_build_version
if get_build_version() < 8.0: if get_build_version() < 8.0:
# this test is only for MSVC8.0 or above # this test is only for MSVC8.0 or above
...@@ -31,9 +86,6 @@ class msvc9compilerTestCase(unittest.TestCase): ...@@ -31,9 +86,6 @@ class msvc9compilerTestCase(unittest.TestCase):
msvc9compiler.find_vcvarsall = old_find_vcvarsall msvc9compiler.find_vcvarsall = old_find_vcvarsall
def test_reg_class(self): def test_reg_class(self):
if sys.platform != 'win32':
# this test is only for win32
return
from distutils.msvccompiler import get_build_version from distutils.msvccompiler import get_build_version
if get_build_version() < 8.0: if get_build_version() < 8.0:
# this test is only for MSVC8.0 or above # this test is only for MSVC8.0 or above
...@@ -56,6 +108,27 @@ class msvc9compilerTestCase(unittest.TestCase): ...@@ -56,6 +108,27 @@ class msvc9compilerTestCase(unittest.TestCase):
keys = Reg.read_keys(HKCU, r'Control Panel') keys = Reg.read_keys(HKCU, r'Control Panel')
self.assertTrue('Desktop' in keys) self.assertTrue('Desktop' in keys)
def test_remove_visual_c_ref(self):
from distutils.msvc9compiler import MSVCCompiler
tempdir = self.mkdtemp()
manifest = os.path.join(tempdir, 'manifest')
f = open(manifest, 'w')
f.write(_MANIFEST)
f.close()
compiler = MSVCCompiler()
compiler._remove_visual_c_ref(manifest)
# see what we got
f = open(manifest)
# removing trailing spaces
content = '\n'.join([line.rstrip() for line in f.readlines()])
f.close()
# makes sure the manifest was properly cleaned
self.assertEquals(content, _CLEANED_MANIFEST)
def test_suite(): def test_suite():
return unittest.makeSuite(msvc9compilerTestCase) return unittest.makeSuite(msvc9compilerTestCase)
......
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