Resolve-MSBuild.ps1
<#PSScriptInfo
.VERSION 1.4.1 .AUTHOR Roman Kuzmin .COPYRIGHT (c) Roman Kuzmin .TAGS Invoke-Build, MSBuild .GUID 53c01926-4fc5-4cbd-aa46-32e415b2373b .LICENSEURI http://www.apache.org/licenses/LICENSE-2.0 .PROJECTURI https://github.com/nightroman/Invoke-Build #> <# .Synopsis Finds the specified or latest MSBuild. .Description The script finds the path to the specified or latest version of MSBuild. It is designed for MSBuild 16.0, 15.0, 14.0, 12.0, 4.0, 3.5, 2.0. For MSBuild 15+ the command uses the module VSSetup, see PSGallery. If VSSetup is not installed then the default locations are used. VSSetup is required for not default installations. MSBuild 15+ resolution: the latest major version (or absolute if -Latest), then Enterprise, Professional, Community, BuildTools, other products. For MSBuild 2.0-14.0 the information is taken from the registry. .Parameter Version Specifies the required MSBuild version. If it is omitted, empty, or * then the command finds and returns the latest installed version path. The optional suffix x86 tells to use 32-bit MSBuild. Known versions: 16.0, 15.0, 14.0, 12.0, 4.0, 3.5, 2.0 .Parameter Latest Tells to select the latest minor version if there are 2+ products with the same major version. Note that major versions have higher precedence than products regardless of -Latest. .Outputs The full path to MSBuild.exe .Example Resolve-MSBuild 16.0x86 Gets the location of 32-bit MSBuild of Visual Studio 2019. .Link https://www.powershellgallery.com/packages/VSSetup #> [OutputType([string])] [CmdletBinding()] param( [string]$Version, [switch]$Latest ) function Get-MSBuild15Path { [CmdletBinding()] param( [string]$Version, [string]$Bitness ) if ([System.IntPtr]::Size -eq 4 -or $Bitness -eq 'x86') { "MSBuild\$Version\Bin\MSBuild.exe" } else { "MSBuild\$Version\Bin\amd64\MSBuild.exe" } } function Get-MSBuild15VSSetup { [CmdletBinding()] param( [string]$Version, [string]$Bitness, [switch]$Latest, [switch]$Prerelease ) if (!(Get-Module VSSetup)) { if (!(Get-Module VSSetup -ListAvailable)) {return} Import-Module VSSetup } $items = @( $v = switch($Version) { '16.0' {'[16.0,17.0)'} '15.0' {'[15.0,16.0)'} default {'[15.0,)'} } Get-VSSetupInstance -Prerelease:$Prerelease | Select-VSSetupInstance -Version $v -Require Microsoft.Component.MSBuild -Product * ) if (!$items) { if (!$Prerelease) { Get-MSBuild15VSSetup $Version $Bitness -Latest:$Latest -Prerelease } return } if ($items.Count -ge 2) { $byVersion = if ($Latest) {{$_.InstallationVersion}} else {{$_.InstallationVersion.Major}} $byProduct = { switch ($_.Product.Id) { Microsoft.VisualStudio.Product.Enterprise {4} Microsoft.VisualStudio.Product.Professional {3} Microsoft.VisualStudio.Product.Community {2} Microsoft.VisualStudio.Product.BuildTools {1} default {0} } } $items = $items | Sort-Object $byVersion, $byProduct } $item = $items[-1] if ($item.InstallationVersion.Major -eq 15) { $Version = '15.0' } else { $Version = 'Current' } Join-Path $item.InstallationPath (Get-MSBuild15Path $Version $Bitness) } function Get-MSBuild15Guess { [CmdletBinding()] param( [string]$Version, [string]$Bitness, [switch]$Latest, [switch]$Prerelease ) if (!($root = ${env:ProgramFiles(x86)})) {$root = $env:ProgramFiles} $folders = $( if ($Prerelease) {'Preview'} elseif ($Version -eq '*') {'2019', '2017'} elseif ($Version -eq '16.0') {'2019'} else {'2017'} ) $items = @( foreach($folder in $folders) { Get-Item -ErrorAction 0 @( "$root\Microsoft Visual Studio\$folder\*\$(Get-MSBuild15Path Current $Bitness)" "$root\Microsoft Visual Studio\$folder\*\$(Get-MSBuild15Path $Version $Bitness)" ) } ) if (!$items) { if (!$Prerelease) { Get-MSBuild15Guess $Version $Bitness -Latest:$Latest -Prerelease } return } if ($items.Count -ge 2) { $byVersion = if ($Latest) {{[version]$_.VersionInfo.FileVersion}} else {{([version]$_.VersionInfo.FileVersion).Major}} $byProduct = { switch -Wildcard ($_.FullName) { *\Enterprise\* {4} *\Professional\* {3} *\Community\* {2} *\BuildTools\* {1} default {0} } } $items = $items | Sort-Object $byVersion, $byProduct } $items[-1].FullName } function Get-MSBuild15 { [CmdletBinding()] param( [string]$Version, [string]$Bitness, [switch]$Latest ) if ($path = Get-MSBuild15VSSetup $Version $Bitness -Latest:$Latest) { $path } else { Get-MSBuild15Guess $Version $Bitness -Latest:$Latest } } function Get-MSBuildOldVersion { [CmdletBinding()] param( [string]$Version, [string]$Bitness ) if ([System.IntPtr]::Size -eq 8 -and $Bitness -eq 'x86') { $key = "HKEY_LOCAL_MACHINE\SOFTWARE\Wow6432Node\Microsoft\MSBuild\ToolsVersions\$Version" } else { $key = "HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\MSBuild\ToolsVersions\$Version" } $rp = [Microsoft.Win32.Registry]::GetValue($key, 'MSBuildToolsPath', '') if ($rp) { Join-Path $rp MSBuild.exe } } function Get-MSBuildOldLatest { [CmdletBinding()] param( [string]$Bitness ) $rp = @(Get-ChildItem HKLM:\SOFTWARE\Microsoft\MSBuild\ToolsVersions | Sort-Object {[Version]$_.PSChildName}) if ($rp) { Get-MSBuildOldVersion $rp[-1].PSChildName $Bitness } } $ErrorActionPreference = 1 try { if ($Version -match '^(.*?)x86\s*$') { $Version = $matches[1] $Bitness = 'x86' } else { $Bitness = '' } $Version = $Version.Trim() $v16 = [Version]'16.0' $v15 = [Version]'15.0' $vMax = [Version]'9999.0' if (!$Version) {$Version = '*'} $vRequired = if ($Version -eq '*') {$vMax} else {[Version]$Version} if ($vRequired -eq $v16 -or $vRequired -eq $v15) { if ($path = Get-MSBuild15 $Version $Bitness -Latest:$Latest) { return $path } } elseif ($vRequired -lt $v15) { if ($path = Get-MSBuildOldVersion $Version $Bitness) { return $path } } elseif ($vRequired -eq $vMax) { if ($path = Get-MSBuild15 * $Bitness -Latest:$Latest) { return $path } if ($path = Get-MSBuildOldLatest $Bitness) { return $path } } throw 'The specified version is not found.' } catch { Write-Error "Cannot resolve MSBuild $Version : $_" } |