mirror of
https://github.com/microsoft/vcpkg.git
synced 2025-01-18 21:13:04 +08:00
209 lines
7.8 KiB
PowerShell
209 lines
7.8 KiB
PowerShell
[CmdletBinding()]
|
|
param(
|
|
[string]$Dependency
|
|
)
|
|
|
|
Import-Module BitsTransfer
|
|
|
|
$scriptsDir = split-path -parent $MyInvocation.MyCommand.Definition
|
|
$vcpkgRootDir = & $scriptsDir\findFileRecursivelyUp.ps1 $scriptsDir .vcpkg-root
|
|
|
|
$downloadsDir = "$vcpkgRootDir\downloads"
|
|
|
|
function SelectProgram([Parameter(Mandatory=$true)][string]$Dependency)
|
|
{
|
|
function promptForDownload([string]$title, [string]$message, [string]$yesDescription, [string]$noDescription)
|
|
{
|
|
if ((Test-Path "$downloadsDir\AlwaysAllowEverything") -Or (Test-Path "$downloadsDir\AlwaysAllowDownloads"))
|
|
{
|
|
return $true
|
|
}
|
|
|
|
$yes = New-Object System.Management.Automation.Host.ChoiceDescription "&Yes", $yesDescription
|
|
$no = New-Object System.Management.Automation.Host.ChoiceDescription "&No", $noDescription
|
|
$AlwaysAllowDownloads = New-Object System.Management.Automation.Host.ChoiceDescription "&Always Allow Downloads", ($yesDescription + "(Future download prompts will not be displayed)")
|
|
|
|
$options = [System.Management.Automation.Host.ChoiceDescription[]]($yes, $no, $AlwaysAllowDownloads)
|
|
$result = $host.ui.PromptForChoice($title, $message, $options, 0)
|
|
|
|
switch ($result)
|
|
{
|
|
0 {return $true}
|
|
1 {return $false}
|
|
2 {
|
|
New-Item "$downloadsDir\AlwaysAllowDownloads" -type file -force | Out-Null
|
|
return $true
|
|
}
|
|
}
|
|
|
|
throw "Unexpected result"
|
|
}
|
|
|
|
|
|
function performDownload( [Parameter(Mandatory=$true)][string]$Dependency,
|
|
[Parameter(Mandatory=$true)][string]$url,
|
|
[Parameter(Mandatory=$true)][string]$downloadDir,
|
|
[Parameter(Mandatory=$true)][string]$downloadPath,
|
|
[Parameter(Mandatory=$true)][string]$downloadVersion,
|
|
[Parameter(Mandatory=$true)][string]$requiredVersion)
|
|
{
|
|
if (Test-Path $downloadPath)
|
|
{
|
|
return
|
|
}
|
|
|
|
$title = "Download " + $Dependency
|
|
$message = ("No suitable version of " + $Dependency + " was found (requires $requiredVersion or higher). Download portable version?")
|
|
$yesDescription = "Downloads " + $Dependency + " v" + $downloadVersion +" app-locally."
|
|
$noDescription = "Does not download " + $Dependency + "."
|
|
|
|
$userAllowedDownload = promptForDownload $title $message $yesDescription $noDescription
|
|
if (!$userAllowedDownload)
|
|
{
|
|
throw [System.IO.FileNotFoundException] ("Could not detect suitable version of " + $Dependency + " and download not allowed")
|
|
}
|
|
|
|
if (!(Test-Path $downloadDir))
|
|
{
|
|
New-Item -ItemType directory -Path $downloadDir | Out-Null
|
|
}
|
|
|
|
if ($Dependency -ne "git") # git fails with BITS
|
|
{
|
|
Start-BitsTransfer -Source $url -Destination $downloadPath -ErrorAction SilentlyContinue
|
|
}
|
|
else
|
|
{
|
|
if (!(Test-Path $downloadPath))
|
|
{
|
|
Write-Host("Downloading $Dependency...")
|
|
(New-Object System.Net.WebClient).DownloadFile($url, $downloadPath)
|
|
}
|
|
}
|
|
}
|
|
|
|
# Enums (without resorting to C#) are only available on powershell 5+.
|
|
$ExtractionType_NO_EXTRACTION_REQUIRED = 0
|
|
$ExtractionType_ZIP = 1
|
|
$ExtractionType_SELF_EXTRACTING_7Z = 2
|
|
|
|
|
|
# Using this to wait for the execution to finish
|
|
function Invoke-Command()
|
|
{
|
|
param ( [string]$program = $(throw "Please specify a program" ),
|
|
[string]$argumentString = "",
|
|
[switch]$waitForExit )
|
|
|
|
$psi = new-object "Diagnostics.ProcessStartInfo"
|
|
$psi.FileName = $program
|
|
$psi.Arguments = $argumentString
|
|
$proc = [Diagnostics.Process]::Start($psi)
|
|
if ( $waitForExit )
|
|
{
|
|
$proc.WaitForExit();
|
|
}
|
|
}
|
|
|
|
function Expand-ZIPFile($file, $destination)
|
|
{
|
|
Write-Host($file)
|
|
Write-Host($destination)
|
|
$shell = new-object -com shell.application
|
|
$zip = $shell.NameSpace($file)
|
|
foreach($item in $zip.items())
|
|
{
|
|
# Piping to Out-Null is used to block until finished
|
|
$shell.Namespace($destination).copyhere($item) | Out-Null
|
|
}
|
|
}
|
|
|
|
if($Dependency -eq "cmake")
|
|
{
|
|
$requiredVersion = "3.7.2"
|
|
$downloadVersion = "3.7.2"
|
|
$url = "https://cmake.org/files/v3.7/cmake-3.7.2-win32-x86.zip"
|
|
$downloadPath = "$downloadsDir\cmake-3.7.2-win32-x86.zip"
|
|
$expectedDownloadedFileHash = "ec5e299d412e0272e01d4de5bf07718f42c96361f83d51cc39f91bf49cc3e5c3"
|
|
$executableFromDownload = "$downloadsDir\cmake-3.7.2-win32-x86\bin\cmake.exe"
|
|
$extractionType = $ExtractionType_ZIP
|
|
}
|
|
elseif($Dependency -eq "nuget")
|
|
{
|
|
$requiredVersion = "3.3.0"
|
|
$downloadVersion = "3.5.0"
|
|
$url = "https://dist.nuget.org/win-x86-commandline/v3.5.0/nuget.exe"
|
|
$downloadPath = "$downloadsDir\nuget-3.5.0\nuget.exe"
|
|
$expectedDownloadedFileHash = "399ec24c26ed54d6887cde61994bb3d1cada7956c1b19ff880f06f060c039918"
|
|
$executableFromDownload = $downloadPath
|
|
$extractionType = $ExtractionType_NO_EXTRACTION_REQUIRED
|
|
}
|
|
elseif($Dependency -eq "git")
|
|
{
|
|
$requiredVersion = "2.0.0"
|
|
$downloadVersion = "2.11.0"
|
|
$url = "https://github.com/git-for-windows/git/releases/download/v2.11.0.windows.3/PortableGit-2.11.0.3-32-bit.7z.exe" # We choose the 32-bit version
|
|
$downloadPath = "$downloadsDir\PortableGit-2.11.0.3-32-bit.7z.exe"
|
|
$expectedDownloadedFileHash = "8bf3769c37945e991903dd1b988c6b1d97bbf0f3afc9851508974f38bf94dc01"
|
|
# There is another copy of git.exe in PortableGit\bin. However, an installed version of git add the cmd dir to the PATH.
|
|
# Therefore, choosing the cmd dir here as well.
|
|
$executableFromDownload = "$downloadsDir\PortableGit\cmd\git.exe"
|
|
$extractionType = $ExtractionType_SELF_EXTRACTING_7Z
|
|
}
|
|
else
|
|
{
|
|
throw "Unknown program requested"
|
|
}
|
|
|
|
$downloadSubdir = Split-path $downloadPath -Parent
|
|
if (!(Test-Path $downloadSubdir))
|
|
{
|
|
New-Item -ItemType Directory -Path $downloadSubdir | Out-Null
|
|
}
|
|
|
|
performDownload $Dependency $url $downloadsDir $downloadPath $downloadVersion $requiredVersion
|
|
|
|
#calculating the hash
|
|
$hashAlgorithm = [Security.Cryptography.HashAlgorithm]::Create("SHA256")
|
|
$fileAsByteArray = [io.File]::ReadAllBytes($downloadPath)
|
|
$hashByteArray = $hashAlgorithm.ComputeHash($fileAsByteArray)
|
|
$downloadedFileHash = -Join ($hashByteArray | ForEach {"{0:x2}" -f $_})
|
|
|
|
if ($expectedDownloadedFileHash -ne $downloadedFileHash)
|
|
{
|
|
throw [System.IO.FileNotFoundException] ("Mismatching hash of the downloaded " + $Dependency)
|
|
}
|
|
|
|
if ($extractionType -eq $ExtractionType_NO_EXTRACTION_REQUIRED)
|
|
{
|
|
# do nothing
|
|
}
|
|
elseif($extractionType -eq $ExtractionType_ZIP)
|
|
{
|
|
if (-not (Test-Path $executableFromDownload)) # consider renaming the extraction folder to make sure the extraction finished
|
|
{
|
|
# Expand-Archive $downloadPath -dest "$downloadsDir" -Force # Requires powershell 5+
|
|
Expand-ZIPFile -File $downloadPath -Destination $downloadsDir
|
|
}
|
|
}
|
|
elseif($extractionType -eq $ExtractionType_SELF_EXTRACTING_7Z)
|
|
{
|
|
if (-not (Test-Path $executableFromDownload))
|
|
{
|
|
Invoke-Command $downloadPath "-y" -waitForExit:$true
|
|
}
|
|
}
|
|
else
|
|
{
|
|
throw "Invalid extraction type"
|
|
}
|
|
|
|
if (-not (Test-Path $executableFromDownload))
|
|
{
|
|
throw [System.IO.FileNotFoundException] ("Could not detect or download " + $Dependency)
|
|
}
|
|
|
|
return $downloadPath
|
|
}
|
|
|
|
SelectProgram $Dependency |