| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805 | """Improved support for Microsoft Visual C++ compilers.Known supported compilers:--------------------------Microsoft Visual C++ 9.0:    Microsoft Visual C++ Compiler for Python 2.7 (x86, amd64)    Microsoft Windows SDK 6.1 (x86, x64, ia64)    Microsoft Windows SDK 7.0 (x86, x64, ia64)Microsoft Visual C++ 10.0:    Microsoft Windows SDK 7.1 (x86, x64, ia64)Microsoft Visual C++ 14.X:    Microsoft Visual C++ Build Tools 2015 (x86, x64, arm)    Microsoft Visual Studio Build Tools 2017 (x86, x64, arm, arm64)    Microsoft Visual Studio Build Tools 2019 (x86, x64, arm, arm64)This may also support compilers shipped with compatible Visual Studio versions."""import jsonfrom io import openfrom os import listdir, pathsepfrom os.path import join, isfile, isdir, dirnameimport sysimport contextlibimport platformimport itertoolsimport subprocessimport distutils.errorsfrom setuptools.extern.packaging.version import LegacyVersionfrom setuptools.extern.more_itertools import unique_everseenfrom .monkey import get_unpatchedif platform.system() == 'Windows':    import winreg    from os import environelse:    # Mock winreg and environ so the module can be imported on this platform.    class winreg:        HKEY_USERS = None        HKEY_CURRENT_USER = None        HKEY_LOCAL_MACHINE = None        HKEY_CLASSES_ROOT = None    environ = dict()_msvc9_suppress_errors = (    # msvc9compiler isn't available on some platforms    ImportError,    # msvc9compiler raises DistutilsPlatformError in some    # environments. See #1118.    distutils.errors.DistutilsPlatformError,)try:    from distutils.msvc9compiler import Regexcept _msvc9_suppress_errors:    passdef msvc9_find_vcvarsall(version):    """    Patched "distutils.msvc9compiler.find_vcvarsall" to use the standalone    compiler build for Python    (VCForPython / Microsoft Visual C++ Compiler for Python 2.7).    Fall back to original behavior when the standalone compiler is not    available.    Redirect the path of "vcvarsall.bat".    Parameters    ----------    version: float        Required Microsoft Visual C++ version.    Return    ------    str        vcvarsall.bat path    """    vc_base = r'Software\%sMicrosoft\DevDiv\VCForPython\%0.1f'    key = vc_base % ('', version)    try:        # Per-user installs register the compiler path here        productdir = Reg.get_value(key, "installdir")    except KeyError:        try:            # All-user installs on a 64-bit system register here            key = vc_base % ('Wow6432Node\\', version)            productdir = Reg.get_value(key, "installdir")        except KeyError:            productdir = None    if productdir:        vcvarsall = join(productdir, "vcvarsall.bat")        if isfile(vcvarsall):            return vcvarsall    return get_unpatched(msvc9_find_vcvarsall)(version)def msvc9_query_vcvarsall(ver, arch='x86', *args, **kwargs):    """    Patched "distutils.msvc9compiler.query_vcvarsall" for support extra    Microsoft Visual C++ 9.0 and 10.0 compilers.    Set environment without use of "vcvarsall.bat".    Parameters    ----------    ver: float        Required Microsoft Visual C++ version.    arch: str        Target architecture.    Return    ------    dict        environment    """    # Try to get environment from vcvarsall.bat (Classical way)    try:        orig = get_unpatched(msvc9_query_vcvarsall)        return orig(ver, arch, *args, **kwargs)    except distutils.errors.DistutilsPlatformError:        # Pass error if Vcvarsall.bat is missing        pass    except ValueError:        # Pass error if environment not set after executing vcvarsall.bat        pass    # If error, try to set environment directly    try:        return EnvironmentInfo(arch, ver).return_env()    except distutils.errors.DistutilsPlatformError as exc:        _augment_exception(exc, ver, arch)        raisedef _msvc14_find_vc2015():    """Python 3.8 "distutils/_msvccompiler.py" backport"""    try:        key = winreg.OpenKey(            winreg.HKEY_LOCAL_MACHINE,            r"Software\Microsoft\VisualStudio\SxS\VC7",            0,            winreg.KEY_READ | winreg.KEY_WOW64_32KEY        )    except OSError:        return None, None    best_version = 0    best_dir = None    with key:        for i in itertools.count():            try:                v, vc_dir, vt = winreg.EnumValue(key, i)            except OSError:                break            if v and vt == winreg.REG_SZ and isdir(vc_dir):                try:                    version = int(float(v))                except (ValueError, TypeError):                    continue                if version >= 14 and version > best_version:                    best_version, best_dir = version, vc_dir    return best_version, best_dirdef _msvc14_find_vc2017():    """Python 3.8 "distutils/_msvccompiler.py" backport    Returns "15, path" based on the result of invoking vswhere.exe    If no install is found, returns "None, None"    The version is returned to avoid unnecessarily changing the function    result. It may be ignored when the path is not None.    If vswhere.exe is not available, by definition, VS 2017 is not    installed.    """    root = environ.get("ProgramFiles(x86)") or environ.get("ProgramFiles")    if not root:        return None, None    try:        path = subprocess.check_output([            join(root, "Microsoft Visual Studio", "Installer", "vswhere.exe"),            "-latest",            "-prerelease",            "-requiresAny",            "-requires", "Microsoft.VisualStudio.Component.VC.Tools.x86.x64",            "-requires", "Microsoft.VisualStudio.Workload.WDExpress",            "-property", "installationPath",            "-products", "*",        ]).decode(encoding="mbcs", errors="strict").strip()    except (subprocess.CalledProcessError, OSError, UnicodeDecodeError):        return None, None    path = join(path, "VC", "Auxiliary", "Build")    if isdir(path):        return 15, path    return None, NonePLAT_SPEC_TO_RUNTIME = {    'x86': 'x86',    'x86_amd64': 'x64',    'x86_arm': 'arm',    'x86_arm64': 'arm64'}def _msvc14_find_vcvarsall(plat_spec):    """Python 3.8 "distutils/_msvccompiler.py" backport"""    _, best_dir = _msvc14_find_vc2017()    vcruntime = None    if plat_spec in PLAT_SPEC_TO_RUNTIME:        vcruntime_plat = PLAT_SPEC_TO_RUNTIME[plat_spec]    else:        vcruntime_plat = 'x64' if 'amd64' in plat_spec else 'x86'    if best_dir:        vcredist = join(best_dir, "..", "..", "redist", "MSVC", "**",                        vcruntime_plat, "Microsoft.VC14*.CRT",                        "vcruntime140.dll")        try:            import glob            vcruntime = glob.glob(vcredist, recursive=True)[-1]        except (ImportError, OSError, LookupError):            vcruntime = None    if not best_dir:        best_version, best_dir = _msvc14_find_vc2015()        if best_version:            vcruntime = join(best_dir, 'redist', vcruntime_plat,                             "Microsoft.VC140.CRT", "vcruntime140.dll")    if not best_dir:        return None, None    vcvarsall = join(best_dir, "vcvarsall.bat")    if not isfile(vcvarsall):        return None, None    if not vcruntime or not isfile(vcruntime):        vcruntime = None    return vcvarsall, vcruntimedef _msvc14_get_vc_env(plat_spec):    """Python 3.8 "distutils/_msvccompiler.py" backport"""    if "DISTUTILS_USE_SDK" in environ:        return {            key.lower(): value            for key, value in environ.items()        }    vcvarsall, vcruntime = _msvc14_find_vcvarsall(plat_spec)    if not vcvarsall:        raise distutils.errors.DistutilsPlatformError(            "Unable to find vcvarsall.bat"        )    try:        out = subprocess.check_output(            'cmd /u /c "{}" {} && set'.format(vcvarsall, plat_spec),            stderr=subprocess.STDOUT,        ).decode('utf-16le', errors='replace')    except subprocess.CalledProcessError as exc:        raise distutils.errors.DistutilsPlatformError(            "Error executing {}".format(exc.cmd)        ) from exc    env = {        key.lower(): value        for key, _, value in        (line.partition('=') for line in out.splitlines())        if key and value    }    if vcruntime:        env['py_vcruntime_redist'] = vcruntime    return envdef msvc14_get_vc_env(plat_spec):    """    Patched "distutils._msvccompiler._get_vc_env" for support extra    Microsoft Visual C++ 14.X compilers.    Set environment without use of "vcvarsall.bat".    Parameters    ----------    plat_spec: str        Target architecture.    Return    ------    dict        environment    """    # Always use backport from CPython 3.8    try:        return _msvc14_get_vc_env(plat_spec)    except distutils.errors.DistutilsPlatformError as exc:        _augment_exception(exc, 14.0)        raisedef msvc14_gen_lib_options(*args, **kwargs):    """    Patched "distutils._msvccompiler.gen_lib_options" for fix    compatibility between "numpy.distutils" and "distutils._msvccompiler"    (for Numpy < 1.11.2)    """    if "numpy.distutils" in sys.modules:        import numpy as np        if LegacyVersion(np.__version__) < LegacyVersion('1.11.2'):            return np.distutils.ccompiler.gen_lib_options(*args, **kwargs)    return get_unpatched(msvc14_gen_lib_options)(*args, **kwargs)def _augment_exception(exc, version, arch=''):    """    Add details to the exception message to help guide the user    as to what action will resolve it.    """    # Error if MSVC++ directory not found or environment not set    message = exc.args[0]    if "vcvarsall" in message.lower() or "visual c" in message.lower():        # Special error message if MSVC++ not installed        tmpl = 'Microsoft Visual C++ {version:0.1f} or greater is required.'        message = tmpl.format(**locals())        msdownload = 'www.microsoft.com/download/details.aspx?id=%d'        if version == 9.0:            if arch.lower().find('ia64') > -1:                # For VC++ 9.0, if IA64 support is needed, redirect user                # to Windows SDK 7.0.                # Note: No download link available from Microsoft.                message += ' Get it with "Microsoft Windows SDK 7.0"'            else:                # For VC++ 9.0 redirect user to Vc++ for Python 2.7 :                # This redirection link is maintained by Microsoft.                # Contact vspython@microsoft.com if it needs updating.                message += ' Get it from http://aka.ms/vcpython27'        elif version == 10.0:            # For VC++ 10.0 Redirect user to Windows SDK 7.1            message += ' Get it with "Microsoft Windows SDK 7.1": '            message += msdownload % 8279        elif version >= 14.0:            # For VC++ 14.X Redirect user to latest Visual C++ Build Tools            message += (' Get it with "Microsoft C++ Build Tools": '                        r'https://visualstudio.microsoft.com'                        r'/visual-cpp-build-tools/')    exc.args = (message, )class PlatformInfo:    """    Current and Target Architectures information.    Parameters    ----------    arch: str        Target architecture.    """    current_cpu = environ.get('processor_architecture', '').lower()    def __init__(self, arch):        self.arch = arch.lower().replace('x64', 'amd64')    @property    def target_cpu(self):        """        Return Target CPU architecture.        Return        ------        str            Target CPU        """        return self.arch[self.arch.find('_') + 1:]    def target_is_x86(self):        """        Return True if target CPU is x86 32 bits..        Return        ------        bool            CPU is x86 32 bits        """        return self.target_cpu == 'x86'    def current_is_x86(self):        """        Return True if current CPU is x86 32 bits..        Return        ------        bool            CPU is x86 32 bits        """        return self.current_cpu == 'x86'    def current_dir(self, hidex86=False, x64=False):        """        Current platform specific subfolder.        Parameters        ----------        hidex86: bool            return '' and not '\x86' if architecture is x86.        x64: bool            return '\x64' and not '\amd64' if architecture is amd64.        Return        ------        str            subfolder: '\target', or '' (see hidex86 parameter)        """        return (            '' if (self.current_cpu == 'x86' and hidex86) else            r'\x64' if (self.current_cpu == 'amd64' and x64) else            r'\%s' % self.current_cpu        )    def target_dir(self, hidex86=False, x64=False):        r"""        Target platform specific subfolder.        Parameters        ----------        hidex86: bool            return '' and not '\x86' if architecture is x86.        x64: bool            return '\x64' and not '\amd64' if architecture is amd64.        Return        ------        str            subfolder: '\current', or '' (see hidex86 parameter)        """        return (            '' if (self.target_cpu == 'x86' and hidex86) else            r'\x64' if (self.target_cpu == 'amd64' and x64) else            r'\%s' % self.target_cpu        )    def cross_dir(self, forcex86=False):        r"""        Cross platform specific subfolder.        Parameters        ----------        forcex86: bool            Use 'x86' as current architecture even if current architecture is            not x86.        Return        ------        str            subfolder: '' if target architecture is current architecture,            '\current_target' if not.        """        current = 'x86' if forcex86 else self.current_cpu        return (            '' if self.target_cpu == current else            self.target_dir().replace('\\', '\\%s_' % current)        )class RegistryInfo:    """    Microsoft Visual Studio related registry information.    Parameters    ----------    platform_info: PlatformInfo        "PlatformInfo" instance.    """    HKEYS = (winreg.HKEY_USERS,             winreg.HKEY_CURRENT_USER,             winreg.HKEY_LOCAL_MACHINE,             winreg.HKEY_CLASSES_ROOT)    def __init__(self, platform_info):        self.pi = platform_info    @property    def visualstudio(self):        """        Microsoft Visual Studio root registry key.        Return        ------        str            Registry key        """        return 'VisualStudio'    @property    def sxs(self):        """        Microsoft Visual Studio SxS registry key.        Return        ------        str            Registry key        """        return join(self.visualstudio, 'SxS')    @property    def vc(self):        """        Microsoft Visual C++ VC7 registry key.        Return        ------        str            Registry key        """        return join(self.sxs, 'VC7')    @property    def vs(self):        """        Microsoft Visual Studio VS7 registry key.        Return        ------        str            Registry key        """        return join(self.sxs, 'VS7')    @property    def vc_for_python(self):        """        Microsoft Visual C++ for Python registry key.        Return        ------        str            Registry key        """        return r'DevDiv\VCForPython'    @property    def microsoft_sdk(self):        """        Microsoft SDK registry key.        Return        ------        str            Registry key        """        return 'Microsoft SDKs'    @property    def windows_sdk(self):        """        Microsoft Windows/Platform SDK registry key.        Return        ------        str            Registry key        """        return join(self.microsoft_sdk, 'Windows')    @property    def netfx_sdk(self):        """        Microsoft .NET Framework SDK registry key.        Return        ------        str            Registry key        """        return join(self.microsoft_sdk, 'NETFXSDK')    @property    def windows_kits_roots(self):        """        Microsoft Windows Kits Roots registry key.        Return        ------        str            Registry key        """        return r'Windows Kits\Installed Roots'    def microsoft(self, key, x86=False):        """        Return key in Microsoft software registry.        Parameters        ----------        key: str            Registry key path where look.        x86: str            Force x86 software registry.        Return        ------        str            Registry key        """        node64 = '' if self.pi.current_is_x86() or x86 else 'Wow6432Node'        return join('Software', node64, 'Microsoft', key)    def lookup(self, key, name):        """        Look for values in registry in Microsoft software registry.        Parameters        ----------        key: str            Registry key path where look.        name: str            Value name to find.        Return        ------        str            value        """        key_read = winreg.KEY_READ        openkey = winreg.OpenKey        closekey = winreg.CloseKey        ms = self.microsoft        for hkey in self.HKEYS:            bkey = None            try:                bkey = openkey(hkey, ms(key), 0, key_read)            except (OSError, IOError):                if not self.pi.current_is_x86():                    try:                        bkey = openkey(hkey, ms(key, True), 0, key_read)                    except (OSError, IOError):                        continue                else:                    continue            try:                return winreg.QueryValueEx(bkey, name)[0]            except (OSError, IOError):                pass            finally:                if bkey:                    closekey(bkey)class SystemInfo:    """    Microsoft Windows and Visual Studio related system information.    Parameters    ----------    registry_info: RegistryInfo        "RegistryInfo" instance.    vc_ver: float        Required Microsoft Visual C++ version.    """    # Variables and properties in this class use originals CamelCase variables    # names from Microsoft source files for more easy comparison.    WinDir = environ.get('WinDir', '')    ProgramFiles = environ.get('ProgramFiles', '')    ProgramFilesx86 = environ.get('ProgramFiles(x86)', ProgramFiles)    def __init__(self, registry_info, vc_ver=None):        self.ri = registry_info        self.pi = self.ri.pi        self.known_vs_paths = self.find_programdata_vs_vers()        # Except for VS15+, VC version is aligned with VS version        self.vs_ver = self.vc_ver = (            vc_ver or self._find_latest_available_vs_ver())    def _find_latest_available_vs_ver(self):        """        Find the latest VC version        Return        ------        float            version        """        reg_vc_vers = self.find_reg_vs_vers()        if not (reg_vc_vers or self.known_vs_paths):            raise distutils.errors.DistutilsPlatformError(                'No Microsoft Visual C++ version found')        vc_vers = set(reg_vc_vers)        vc_vers.update(self.known_vs_paths)        return sorted(vc_vers)[-1]    def find_reg_vs_vers(self):        """        Find Microsoft Visual Studio versions available in registry.        Return        ------        list of float            Versions        """        ms = self.ri.microsoft        vckeys = (self.ri.vc, self.ri.vc_for_python, self.ri.vs)        vs_vers = []        for hkey, key in itertools.product(self.ri.HKEYS, vckeys):            try:                bkey = winreg.OpenKey(hkey, ms(key), 0, winreg.KEY_READ)            except (OSError, IOError):                continue            with bkey:                subkeys, values, _ = winreg.QueryInfoKey(bkey)                for i in range(values):                    with contextlib.suppress(ValueError):                        ver = float(winreg.EnumValue(bkey, i)[0])                        if ver not in vs_vers:                            vs_vers.append(ver)                for i in range(subkeys):                    with contextlib.suppress(ValueError):                        ver = float(winreg.EnumKey(bkey, i))                        if ver not in vs_vers:                            vs_vers.append(ver)        return sorted(vs_vers)    def find_programdata_vs_vers(self):        r"""        Find Visual studio 2017+ versions from information in        "C:\ProgramData\Microsoft\VisualStudio\Packages\_Instances".        Return        ------        dict            float version as key, path as value.        """        vs_versions = {}        instances_dir = \            r'C:\ProgramData\Microsoft\VisualStudio\Packages\_Instances'        try:            hashed_names = listdir(instances_dir)        except (OSError, IOError):            # Directory not exists with all Visual Studio versions            return vs_versions        for name in hashed_names:            try:                # Get VS installation path from "state.json" file                state_path = join(instances_dir, name, 'state.json')                with open(state_path, 'rt', encoding='utf-8') as state_file:                    state = json.load(state_file)                vs_path = state['installationPath']                # Raises OSError if this VS installation does not contain VC                listdir(join(vs_path, r'VC\Tools\MSVC'))                # Store version and path                vs_versions[self._as_float_version(                    state['installationVersion'])] = vs_path            except (OSError, IOError, KeyError):                # Skip if "state.json" file is missing or bad format                continue        return vs_versions    @staticmethod    def _as_float_version(version):        """        Return a string version as a simplified float version (major.minor)        Parameters        ----------        version: str            Version.        Return        ------        float            version        """        return float('.'.join(version.split('.')[:2]))    @property    def VSInstallDir(self):        """        Microsoft Visual Studio directory.        Return        ------        str            path        """        # Default path        default = join(self.ProgramFilesx86,                       'Microsoft Visual Studio %0.1f' % self.vs_ver)        # Try to get path from registry, if fail use default path        return self.ri.lookup(self.ri.vs, '%0.1f' % self.vs_ver) or default    @property    def VCInstallDir(self):        """        Microsoft Visual C++ directory.        Return        ------        str            path        """        path = self._guess_vc() or self._guess_vc_legacy()        if not isdir(path):            msg = 'Microsoft Visual C++ directory not found'            raise distutils.errors.DistutilsPlatformError(msg)        return path    def _guess_vc(self):        """        Locate Visual C++ for VS2017+.        Return        ------        str            path        """        if self.vs_ver <= 14.0:            return ''        try:            # First search in known VS paths            vs_dir = self.known_vs_paths[self.vs_ver]        except KeyError:            # Else, search with path from registry            vs_dir = self.VSInstallDir        guess_vc = join(vs_dir, r'VC\Tools\MSVC')        # Subdir with VC exact version as name        try:            # Update the VC version with real one instead of VS version            vc_ver = listdir(guess_vc)[-1]            self.vc_ver = self._as_float_version(vc_ver)            return join(guess_vc, vc_ver)        except (OSError, IOError, IndexError):            return ''    def _guess_vc_legacy(self):        """        Locate Visual C++ for versions prior to 2017.        Return        ------        str            path        """        default = join(self.ProgramFilesx86,                       r'Microsoft Visual Studio %0.1f\VC' % self.vs_ver)        # Try to get "VC++ for Python" path from registry as default path        reg_path = join(self.ri.vc_for_python, '%0.1f' % self.vs_ver)        python_vc = self.ri.lookup(reg_path, 'installdir')        default_vc = join(python_vc, 'VC') if python_vc else default        # Try to get path from registry, if fail use default path        return self.ri.lookup(self.ri.vc, '%0.1f' % self.vs_ver) or default_vc    @property    def WindowsSdkVersion(self):        """        Microsoft Windows SDK versions for specified MSVC++ version.        Return        ------        tuple of str            versions        """        if self.vs_ver <= 9.0:            return '7.0', '6.1', '6.0a'        elif self.vs_ver == 10.0:            return '7.1', '7.0a'        elif self.vs_ver == 11.0:            return '8.0', '8.0a'        elif self.vs_ver == 12.0:            return '8.1', '8.1a'        elif self.vs_ver >= 14.0:            return '10.0', '8.1'    @property    def WindowsSdkLastVersion(self):        """        Microsoft Windows SDK last version.        Return        ------        str            version        """        return self._use_last_dir_name(join(self.WindowsSdkDir, 'lib'))    @property  # noqa: C901    def WindowsSdkDir(self):  # noqa: C901  # is too complex (12)  # FIXME        """        Microsoft Windows SDK directory.        Return        ------        str            path        """        sdkdir = ''        for ver in self.WindowsSdkVersion:            # Try to get it from registry            loc = join(self.ri.windows_sdk, 'v%s' % ver)            sdkdir = self.ri.lookup(loc, 'installationfolder')            if sdkdir:                break        if not sdkdir or not isdir(sdkdir):            # Try to get "VC++ for Python" version from registry            path = join(self.ri.vc_for_python, '%0.1f' % self.vc_ver)            install_base = self.ri.lookup(path, 'installdir')            if install_base:                sdkdir = join(install_base, 'WinSDK')        if not sdkdir or not isdir(sdkdir):            # If fail, use default new path            for ver in self.WindowsSdkVersion:                intver = ver[:ver.rfind('.')]                path = r'Microsoft SDKs\Windows Kits\%s' % intver                d = join(self.ProgramFiles, path)                if isdir(d):                    sdkdir = d        if not sdkdir or not isdir(sdkdir):            # If fail, use default old path            for ver in self.WindowsSdkVersion:                path = r'Microsoft SDKs\Windows\v%s' % ver                d = join(self.ProgramFiles, path)                if isdir(d):                    sdkdir = d        if not sdkdir:            # If fail, use Platform SDK            sdkdir = join(self.VCInstallDir, 'PlatformSDK')        return sdkdir    @property    def WindowsSDKExecutablePath(self):        """        Microsoft Windows SDK executable directory.        Return        ------        str            path        """        # Find WinSDK NetFx Tools registry dir name        if self.vs_ver <= 11.0:            netfxver = 35            arch = ''        else:            netfxver = 40            hidex86 = True if self.vs_ver <= 12.0 else False            arch = self.pi.current_dir(x64=True, hidex86=hidex86)        fx = 'WinSDK-NetFx%dTools%s' % (netfxver, arch.replace('\\', '-'))        # list all possibles registry paths        regpaths = []        if self.vs_ver >= 14.0:            for ver in self.NetFxSdkVersion:                regpaths += [join(self.ri.netfx_sdk, ver, fx)]        for ver in self.WindowsSdkVersion:            regpaths += [join(self.ri.windows_sdk, 'v%sA' % ver, fx)]        # Return installation folder from the more recent path        for path in regpaths:            execpath = self.ri.lookup(path, 'installationfolder')            if execpath:                return execpath    @property    def FSharpInstallDir(self):        """        Microsoft Visual F# directory.        Return        ------        str            path        """        path = join(self.ri.visualstudio, r'%0.1f\Setup\F#' % self.vs_ver)        return self.ri.lookup(path, 'productdir') or ''    @property    def UniversalCRTSdkDir(self):        """        Microsoft Universal CRT SDK directory.        Return        ------        str            path        """        # Set Kit Roots versions for specified MSVC++ version        vers = ('10', '81') if self.vs_ver >= 14.0 else ()        # Find path of the more recent Kit        for ver in vers:            sdkdir = self.ri.lookup(self.ri.windows_kits_roots,                                    'kitsroot%s' % ver)            if sdkdir:                return sdkdir or ''    @property    def UniversalCRTSdkLastVersion(self):        """        Microsoft Universal C Runtime SDK last version.        Return        ------        str            version        """        return self._use_last_dir_name(join(self.UniversalCRTSdkDir, 'lib'))    @property    def NetFxSdkVersion(self):        """        Microsoft .NET Framework SDK versions.        Return        ------        tuple of str            versions        """        # Set FxSdk versions for specified VS version        return (('4.7.2', '4.7.1', '4.7',                 '4.6.2', '4.6.1', '4.6',                 '4.5.2', '4.5.1', '4.5')                if self.vs_ver >= 14.0 else ())    @property    def NetFxSdkDir(self):        """        Microsoft .NET Framework SDK directory.        Return        ------        str            path        """        sdkdir = ''        for ver in self.NetFxSdkVersion:            loc = join(self.ri.netfx_sdk, ver)            sdkdir = self.ri.lookup(loc, 'kitsinstallationfolder')            if sdkdir:                break        return sdkdir    @property    def FrameworkDir32(self):        """        Microsoft .NET Framework 32bit directory.        Return        ------        str            path        """        # Default path        guess_fw = join(self.WinDir, r'Microsoft.NET\Framework')        # Try to get path from registry, if fail use default path        return self.ri.lookup(self.ri.vc, 'frameworkdir32') or guess_fw    @property    def FrameworkDir64(self):        """        Microsoft .NET Framework 64bit directory.        Return        ------        str            path        """        # Default path        guess_fw = join(self.WinDir, r'Microsoft.NET\Framework64')        # Try to get path from registry, if fail use default path        return self.ri.lookup(self.ri.vc, 'frameworkdir64') or guess_fw    @property    def FrameworkVersion32(self):        """        Microsoft .NET Framework 32bit versions.        Return        ------        tuple of str            versions        """        return self._find_dot_net_versions(32)    @property    def FrameworkVersion64(self):        """        Microsoft .NET Framework 64bit versions.        Return        ------        tuple of str            versions        """        return self._find_dot_net_versions(64)    def _find_dot_net_versions(self, bits):        """        Find Microsoft .NET Framework versions.        Parameters        ----------        bits: int            Platform number of bits: 32 or 64.        Return        ------        tuple of str            versions        """        # Find actual .NET version in registry        reg_ver = self.ri.lookup(self.ri.vc, 'frameworkver%d' % bits)        dot_net_dir = getattr(self, 'FrameworkDir%d' % bits)        ver = reg_ver or self._use_last_dir_name(dot_net_dir, 'v') or ''        # Set .NET versions for specified MSVC++ version        if self.vs_ver >= 12.0:            return ver, 'v4.0'        elif self.vs_ver >= 10.0:            return 'v4.0.30319' if ver.lower()[:2] != 'v4' else ver, 'v3.5'        elif self.vs_ver == 9.0:            return 'v3.5', 'v2.0.50727'        elif self.vs_ver == 8.0:            return 'v3.0', 'v2.0.50727'    @staticmethod    def _use_last_dir_name(path, prefix=''):        """        Return name of the last dir in path or '' if no dir found.        Parameters        ----------        path: str            Use dirs in this path        prefix: str            Use only dirs starting by this prefix        Return        ------        str            name        """        matching_dirs = (            dir_name            for dir_name in reversed(listdir(path))            if isdir(join(path, dir_name)) and            dir_name.startswith(prefix)        )        return next(matching_dirs, None) or ''class EnvironmentInfo:    """    Return environment variables for specified Microsoft Visual C++ version    and platform : Lib, Include, Path and libpath.    This function is compatible with Microsoft Visual C++ 9.0 to 14.X.    Script created by analysing Microsoft environment configuration files like    "vcvars[...].bat", "SetEnv.Cmd", "vcbuildtools.bat", ...    Parameters    ----------    arch: str        Target architecture.    vc_ver: float        Required Microsoft Visual C++ version. If not set, autodetect the last        version.    vc_min_ver: float        Minimum Microsoft Visual C++ version.    """    # Variables and properties in this class use originals CamelCase variables    # names from Microsoft source files for more easy comparison.    def __init__(self, arch, vc_ver=None, vc_min_ver=0):        self.pi = PlatformInfo(arch)        self.ri = RegistryInfo(self.pi)        self.si = SystemInfo(self.ri, vc_ver)        if self.vc_ver < vc_min_ver:            err = 'No suitable Microsoft Visual C++ version found'            raise distutils.errors.DistutilsPlatformError(err)    @property    def vs_ver(self):        """        Microsoft Visual Studio.        Return        ------        float            version        """        return self.si.vs_ver    @property    def vc_ver(self):        """        Microsoft Visual C++ version.        Return        ------        float            version        """        return self.si.vc_ver    @property    def VSTools(self):        """        Microsoft Visual Studio Tools.        Return        ------        list of str            paths        """        paths = [r'Common7\IDE', r'Common7\Tools']        if self.vs_ver >= 14.0:            arch_subdir = self.pi.current_dir(hidex86=True, x64=True)            paths += [r'Common7\IDE\CommonExtensions\Microsoft\TestWindow']            paths += [r'Team Tools\Performance Tools']            paths += [r'Team Tools\Performance Tools%s' % arch_subdir]        return [join(self.si.VSInstallDir, path) for path in paths]    @property    def VCIncludes(self):        """        Microsoft Visual C++ & Microsoft Foundation Class Includes.        Return        ------        list of str            paths        """        return [join(self.si.VCInstallDir, 'Include'),                join(self.si.VCInstallDir, r'ATLMFC\Include')]    @property    def VCLibraries(self):        """        Microsoft Visual C++ & Microsoft Foundation Class Libraries.        Return        ------        list of str            paths        """        if self.vs_ver >= 15.0:            arch_subdir = self.pi.target_dir(x64=True)        else:            arch_subdir = self.pi.target_dir(hidex86=True)        paths = ['Lib%s' % arch_subdir, r'ATLMFC\Lib%s' % arch_subdir]        if self.vs_ver >= 14.0:            paths += [r'Lib\store%s' % arch_subdir]        return [join(self.si.VCInstallDir, path) for path in paths]    @property    def VCStoreRefs(self):        """        Microsoft Visual C++ store references Libraries.        Return        ------        list of str            paths        """        if self.vs_ver < 14.0:            return []        return [join(self.si.VCInstallDir, r'Lib\store\references')]    @property    def VCTools(self):        """        Microsoft Visual C++ Tools.        Return        ------        list of str            paths        """        si = self.si        tools = [join(si.VCInstallDir, 'VCPackages')]        forcex86 = True if self.vs_ver <= 10.0 else False        arch_subdir = self.pi.cross_dir(forcex86)        if arch_subdir:            tools += [join(si.VCInstallDir, 'Bin%s' % arch_subdir)]        if self.vs_ver == 14.0:            path = 'Bin%s' % self.pi.current_dir(hidex86=True)            tools += [join(si.VCInstallDir, path)]        elif self.vs_ver >= 15.0:            host_dir = (r'bin\HostX86%s' if self.pi.current_is_x86() else                        r'bin\HostX64%s')            tools += [join(                si.VCInstallDir, host_dir % self.pi.target_dir(x64=True))]            if self.pi.current_cpu != self.pi.target_cpu:                tools += [join(                    si.VCInstallDir, host_dir % self.pi.current_dir(x64=True))]        else:            tools += [join(si.VCInstallDir, 'Bin')]        return tools    @property    def OSLibraries(self):        """        Microsoft Windows SDK Libraries.        Return        ------        list of str            paths        """        if self.vs_ver <= 10.0:            arch_subdir = self.pi.target_dir(hidex86=True, x64=True)            return [join(self.si.WindowsSdkDir, 'Lib%s' % arch_subdir)]        else:            arch_subdir = self.pi.target_dir(x64=True)            lib = join(self.si.WindowsSdkDir, 'lib')            libver = self._sdk_subdir            return [join(lib, '%sum%s' % (libver, arch_subdir))]    @property    def OSIncludes(self):        """        Microsoft Windows SDK Include.        Return        ------        list of str            paths        """        include = join(self.si.WindowsSdkDir, 'include')        if self.vs_ver <= 10.0:            return [include, join(include, 'gl')]        else:            if self.vs_ver >= 14.0:                sdkver = self._sdk_subdir            else:                sdkver = ''            return [join(include, '%sshared' % sdkver),                    join(include, '%sum' % sdkver),                    join(include, '%swinrt' % sdkver)]    @property    def OSLibpath(self):        """        Microsoft Windows SDK Libraries Paths.        Return        ------        list of str            paths        """        ref = join(self.si.WindowsSdkDir, 'References')        libpath = []        if self.vs_ver <= 9.0:            libpath += self.OSLibraries        if self.vs_ver >= 11.0:            libpath += [join(ref, r'CommonConfiguration\Neutral')]        if self.vs_ver >= 14.0:            libpath += [                ref,                join(self.si.WindowsSdkDir, 'UnionMetadata'),                join(                    ref, 'Windows.Foundation.UniversalApiContract', '1.0.0.0'),                join(ref, 'Windows.Foundation.FoundationContract', '1.0.0.0'),                join(                    ref, 'Windows.Networking.Connectivity.WwanContract',                    '1.0.0.0'),                join(                    self.si.WindowsSdkDir, 'ExtensionSDKs', 'Microsoft.VCLibs',                    '%0.1f' % self.vs_ver, 'References', 'CommonConfiguration',                    'neutral'),            ]        return libpath    @property    def SdkTools(self):        """        Microsoft Windows SDK Tools.        Return        ------        list of str            paths        """        return list(self._sdk_tools())    def _sdk_tools(self):        """        Microsoft Windows SDK Tools paths generator.        Return        ------        generator of str            paths        """        if self.vs_ver < 15.0:            bin_dir = 'Bin' if self.vs_ver <= 11.0 else r'Bin\x86'            yield join(self.si.WindowsSdkDir, bin_dir)        if not self.pi.current_is_x86():            arch_subdir = self.pi.current_dir(x64=True)            path = 'Bin%s' % arch_subdir            yield join(self.si.WindowsSdkDir, path)        if self.vs_ver in (10.0, 11.0):            if self.pi.target_is_x86():                arch_subdir = ''            else:                arch_subdir = self.pi.current_dir(hidex86=True, x64=True)            path = r'Bin\NETFX 4.0 Tools%s' % arch_subdir            yield join(self.si.WindowsSdkDir, path)        elif self.vs_ver >= 15.0:            path = join(self.si.WindowsSdkDir, 'Bin')            arch_subdir = self.pi.current_dir(x64=True)            sdkver = self.si.WindowsSdkLastVersion            yield join(path, '%s%s' % (sdkver, arch_subdir))        if self.si.WindowsSDKExecutablePath:            yield self.si.WindowsSDKExecutablePath    @property    def _sdk_subdir(self):        """        Microsoft Windows SDK version subdir.        Return        ------        str            subdir        """        ucrtver = self.si.WindowsSdkLastVersion        return ('%s\\' % ucrtver) if ucrtver else ''    @property    def SdkSetup(self):        """        Microsoft Windows SDK Setup.        Return        ------        list of str            paths        """        if self.vs_ver > 9.0:            return []        return [join(self.si.WindowsSdkDir, 'Setup')]    @property    def FxTools(self):        """        Microsoft .NET Framework Tools.        Return        ------        list of str            paths        """        pi = self.pi        si = self.si        if self.vs_ver <= 10.0:            include32 = True            include64 = not pi.target_is_x86() and not pi.current_is_x86()        else:            include32 = pi.target_is_x86() or pi.current_is_x86()            include64 = pi.current_cpu == 'amd64' or pi.target_cpu == 'amd64'        tools = []        if include32:            tools += [join(si.FrameworkDir32, ver)                      for ver in si.FrameworkVersion32]        if include64:            tools += [join(si.FrameworkDir64, ver)                      for ver in si.FrameworkVersion64]        return tools    @property    def NetFxSDKLibraries(self):        """        Microsoft .Net Framework SDK Libraries.        Return        ------        list of str            paths        """        if self.vs_ver < 14.0 or not self.si.NetFxSdkDir:            return []        arch_subdir = self.pi.target_dir(x64=True)        return [join(self.si.NetFxSdkDir, r'lib\um%s' % arch_subdir)]    @property    def NetFxSDKIncludes(self):        """        Microsoft .Net Framework SDK Includes.        Return        ------        list of str            paths        """        if self.vs_ver < 14.0 or not self.si.NetFxSdkDir:            return []        return [join(self.si.NetFxSdkDir, r'include\um')]    @property    def VsTDb(self):        """        Microsoft Visual Studio Team System Database.        Return        ------        list of str            paths        """        return [join(self.si.VSInstallDir, r'VSTSDB\Deploy')]    @property    def MSBuild(self):        """        Microsoft Build Engine.        Return        ------        list of str            paths        """        if self.vs_ver < 12.0:            return []        elif self.vs_ver < 15.0:            base_path = self.si.ProgramFilesx86            arch_subdir = self.pi.current_dir(hidex86=True)        else:            base_path = self.si.VSInstallDir            arch_subdir = ''        path = r'MSBuild\%0.1f\bin%s' % (self.vs_ver, arch_subdir)        build = [join(base_path, path)]        if self.vs_ver >= 15.0:            # Add Roslyn C# & Visual Basic Compiler            build += [join(base_path, path, 'Roslyn')]        return build    @property    def HTMLHelpWorkshop(self):        """        Microsoft HTML Help Workshop.        Return        ------        list of str            paths        """        if self.vs_ver < 11.0:            return []        return [join(self.si.ProgramFilesx86, 'HTML Help Workshop')]    @property    def UCRTLibraries(self):        """        Microsoft Universal C Runtime SDK Libraries.        Return        ------        list of str            paths        """        if self.vs_ver < 14.0:            return []        arch_subdir = self.pi.target_dir(x64=True)        lib = join(self.si.UniversalCRTSdkDir, 'lib')        ucrtver = self._ucrt_subdir        return [join(lib, '%sucrt%s' % (ucrtver, arch_subdir))]    @property    def UCRTIncludes(self):        """        Microsoft Universal C Runtime SDK Include.        Return        ------        list of str            paths        """        if self.vs_ver < 14.0:            return []        include = join(self.si.UniversalCRTSdkDir, 'include')        return [join(include, '%sucrt' % self._ucrt_subdir)]    @property    def _ucrt_subdir(self):        """        Microsoft Universal C Runtime SDK version subdir.        Return        ------        str            subdir        """        ucrtver = self.si.UniversalCRTSdkLastVersion        return ('%s\\' % ucrtver) if ucrtver else ''    @property    def FSharp(self):        """        Microsoft Visual F#.        Return        ------        list of str            paths        """        if 11.0 > self.vs_ver > 12.0:            return []        return [self.si.FSharpInstallDir]    @property    def VCRuntimeRedist(self):        """        Microsoft Visual C++ runtime redistributable dll.        Return        ------        str            path        """        vcruntime = 'vcruntime%d0.dll' % self.vc_ver        arch_subdir = self.pi.target_dir(x64=True).strip('\\')        # Installation prefixes candidates        prefixes = []        tools_path = self.si.VCInstallDir        redist_path = dirname(tools_path.replace(r'\Tools', r'\Redist'))        if isdir(redist_path):            # Redist version may not be exactly the same as tools            redist_path = join(redist_path, listdir(redist_path)[-1])            prefixes += [redist_path, join(redist_path, 'onecore')]        prefixes += [join(tools_path, 'redist')]  # VS14 legacy path        # CRT directory        crt_dirs = ('Microsoft.VC%d.CRT' % (self.vc_ver * 10),                    # Sometime store in directory with VS version instead of VC                    'Microsoft.VC%d.CRT' % (int(self.vs_ver) * 10))        # vcruntime path        for prefix, crt_dir in itertools.product(prefixes, crt_dirs):            path = join(prefix, arch_subdir, crt_dir, vcruntime)            if isfile(path):                return path    def return_env(self, exists=True):        """        Return environment dict.        Parameters        ----------        exists: bool            It True, only return existing paths.        Return        ------        dict            environment        """        env = dict(            include=self._build_paths('include',                                      [self.VCIncludes,                                       self.OSIncludes,                                       self.UCRTIncludes,                                       self.NetFxSDKIncludes],                                      exists),            lib=self._build_paths('lib',                                  [self.VCLibraries,                                   self.OSLibraries,                                   self.FxTools,                                   self.UCRTLibraries,                                   self.NetFxSDKLibraries],                                  exists),            libpath=self._build_paths('libpath',                                      [self.VCLibraries,                                       self.FxTools,                                       self.VCStoreRefs,                                       self.OSLibpath],                                      exists),            path=self._build_paths('path',                                   [self.VCTools,                                    self.VSTools,                                    self.VsTDb,                                    self.SdkTools,                                    self.SdkSetup,                                    self.FxTools,                                    self.MSBuild,                                    self.HTMLHelpWorkshop,                                    self.FSharp],                                   exists),        )        if self.vs_ver >= 14 and isfile(self.VCRuntimeRedist):            env['py_vcruntime_redist'] = self.VCRuntimeRedist        return env    def _build_paths(self, name, spec_path_lists, exists):        """        Given an environment variable name and specified paths,        return a pathsep-separated string of paths containing        unique, extant, directories from those paths and from        the environment variable. Raise an error if no paths        are resolved.        Parameters        ----------        name: str            Environment variable name        spec_path_lists: list of str            Paths        exists: bool            It True, only return existing paths.        Return        ------        str            Pathsep-separated paths        """        # flatten spec_path_lists        spec_paths = itertools.chain.from_iterable(spec_path_lists)        env_paths = environ.get(name, '').split(pathsep)        paths = itertools.chain(spec_paths, env_paths)        extant_paths = list(filter(isdir, paths)) if exists else paths        if not extant_paths:            msg = "%s environment variable is empty" % name.upper()            raise distutils.errors.DistutilsPlatformError(msg)        unique_paths = unique_everseen(extant_paths)        return pathsep.join(unique_paths)
 |