function Install-Binary { <# .SYNOPSIS A helper function to install executables. .DESCRIPTION Download and install .exe or .msi binaries from specified URL. .PARAMETER Url The URL from which the binary will be downloaded. Required parameter. .PARAMETER Name The Name with which binary will be downloaded. Required parameter. .PARAMETER ArgumentList The list of arguments that will be passed to the installer. Required for .exe binaries. .EXAMPLE Install-Binary -Url "https://go.microsoft.com/fwlink/p/?linkid=2083338" -Name "winsdksetup.exe" -ArgumentList ("/features", "+", "/quiet") #> Param ( [Parameter(Mandatory, ParameterSetName="Url")] [String] $Url, [Parameter(Mandatory, ParameterSetName="Url")] [String] $Name, [Parameter(Mandatory, ParameterSetName="LocalPath")] [String] $FilePath, [String[]] $ArgumentList ) if ($PSCmdlet.ParameterSetName -eq "LocalPath") { $name = Split-Path -Path $FilePath -Leaf } else { Write-Host "Downloading $Name..." $filePath = Start-DownloadWithRetry -Url $Url -Name $Name } # MSI binaries should be installed via msiexec.exe $fileExtension = ([System.IO.Path]::GetExtension($Name)).Replace(".", "") if ($fileExtension -eq "msi") { if (-not $ArgumentList) { $ArgumentList = ('/i', $filePath, '/QN', '/norestart') } $filePath = "msiexec.exe" } try { $installStartTime = Get-Date Write-Host "Starting Install $Name..." $process = Start-Process -FilePath $filePath -ArgumentList $ArgumentList -Wait -PassThru $exitCode = $process.ExitCode $installCompleteTime = [math]::Round(($(Get-Date) - $installStartTime).TotalSeconds, 2) if ($exitCode -eq 0 -or $exitCode -eq 3010) { Write-Host "Installation successful in $installCompleteTime seconds" } else { Write-Host "Non zero exit code returned by the installation process: $exitCode" Write-Host "Total time elapsed: $installCompleteTime seconds" exit $exitCode } } catch { $installCompleteTime = [math]::Round(($(Get-Date) - $installStartTime).TotalSeconds, 2) Write-Host "Failed to install the $fileExtension ${Name}: $($_.Exception.Message)" Write-Host "Installation failed after $installCompleteTime seconds" exit 1 } } function Stop-SvcWithErrHandling { <# .DESCRIPTION Function for stopping the Windows Service with error handling .PARAMETER ServiceName The name of stopping service .PARAMETER StopOnError Switch for stopping the script and exit from PowerShell if one service is absent #> Param ( [Parameter(Mandatory, ValueFromPipeLine = $true)] [string] $ServiceName, [switch] $StopOnError ) Process { $service = Get-Service $ServiceName -ErrorAction SilentlyContinue if (-not $service) { Write-Warning "[!] Service [$ServiceName] is not found" if ($StopOnError) { exit 1 } } else { Write-Host "Try to stop service [$ServiceName]" try { Stop-Service -Name $ServiceName -Force $service.WaitForStatus("Stopped", "00:01:00") Write-Host "Service [$ServiceName] has been stopped successfuly" } catch { Write-Error "[!] Failed to stop service [$ServiceName] with error:" $_ | Out-String | Write-Error } } } } function Set-SvcWithErrHandling { <# .DESCRIPTION Function for setting the Windows Service parameter with error handling .PARAMETER ServiceName The name of stopping service .PARAMETER Arguments Hashtable for service arguments #> Param ( [Parameter(Mandatory, ValueFromPipeLine = $true)] [string] $ServiceName, [Parameter(Mandatory)] [hashtable] $Arguments ) Process { $service = Get-Service $ServiceName -ErrorAction SilentlyContinue if (-not $service) { Write-Warning "[!] Service [$ServiceName] is not found" } try { Set-Service $serviceName @Arguments } catch { Write-Error "[!] Failed to set service [$ServiceName] arguments with error:" $_ | Out-String | Write-Error } } } function Start-DownloadWithRetry { Param ( [Parameter(Mandatory)] [string] $Url, [string] $Name, [string] $DownloadPath = "${env:Temp}", [int] $Retries = 20 ) if ([String]::IsNullOrEmpty($Name)) { $Name = [IO.Path]::GetFileName($Url) } $filePath = Join-Path -Path $DownloadPath -ChildPath $Name $downloadStartTime = Get-Date # Default retry logic for the package. while ($Retries -gt 0) { try { $downloadAttemptStartTime = Get-Date Write-Host "Downloading package from: $Url to path $filePath ." (New-Object System.Net.WebClient).DownloadFile($Url, $filePath) break } catch { $failTime = [math]::Round(($(Get-Date) - $downloadStartTime).TotalSeconds, 2) $attemptTime = [math]::Round(($(Get-Date) - $downloadAttemptStartTime).TotalSeconds, 2) Write-Host "There is an error encounterd after $attemptTime seconds during package downloading:`n $_" $Retries-- if ($Retries -eq 0) { Write-Host "File can't be downloaded. Please try later or check that file exists by url: $Url" Write-Host "Total time elapsed $failTime" exit 1 } Write-Host "Waiting 30 seconds before retrying. Retries left: $Retries" Start-Sleep -Seconds 30 } } $downloadCompleteTime = [math]::Round(($(Get-Date) - $downloadStartTime).TotalSeconds, 2) Write-Host "Package downloaded successfully in $downloadCompleteTime seconds" return $filePath } function Get-VsixExtenstionFromMarketplace { Param ( [string] $ExtensionMarketPlaceName, [string] $MarketplaceUri = "https://marketplace.visualstudio.com/items?itemName=" ) $extensionUri = $MarketplaceUri + $ExtensionMarketPlaceName $request = Invoke-SBWithRetry -Command { Invoke-WebRequest -Uri $extensionUri -UseBasicParsing } -RetryCount 20 -RetryIntervalSeconds 30 $request -match 'UniqueIdentifierValue":"(?[^"]*)' | Out-Null $extensionName = $Matches.extensionname $request -match 'VsixId":"(?[^"]*)' | Out-Null $vsixId = $Matches.vsixid $request -match 'AssetUri":"(?[^"]*)' | Out-Null $assetUri = $Matches.uri $request -match 'Microsoft\.VisualStudio\.Services\.Payload\.FileName":"(?[^"]*)' | Out-Null $fileName = $Matches.filename $downloadUri = $assetUri + "/" + $fileName # ProBITools.MicrosoftReportProjectsforVisualStudio2022 has different URL https://github.com/actions/runner-images/issues/5340 switch ($ExtensionMarketPlaceName) { "ProBITools.MicrosoftReportProjectsforVisualStudio2022" { $fileName = "Microsoft.DataTools.ReportingServices.vsix" $downloadUri = "https://download.microsoft.com/download/b/b/5/bb57be7e-ae72-4fc0-b528-d0ec224997bd/Microsoft.DataTools.ReportingServices.vsix" } "ProBITools.MicrosoftAnalysisServicesModelingProjects2022" { $fileName = "Microsoft.DataTools.AnalysisServices.vsix" $downloadUri = "https://download.microsoft.com/download/c/8/9/c896a7f2-d0fd-45ac-90e6-ff61f67523cb/Microsoft.DataTools.AnalysisServices.vsix" } # Starting from version 4.1 SqlServerIntegrationServicesProjects extension is distributed as exe file "SSIS.SqlServerIntegrationServicesProjects" { $fileName = "Microsoft.DataTools.IntegrationServices.exe" $downloadUri = $assetUri + "/" + $fileName } } return [PSCustomObject] @{ "ExtensionName" = $extensionName "VsixId" = $vsixId "FileName" = $fileName "DownloadUri" = $downloadUri } } function Install-VsixExtension { Param ( [string] $Url, [Parameter(Mandatory = $true)] [string] $Name, [string] $FilePath, [Parameter(Mandatory = $true)] [string] $VSversion, [int] $retries = 20, [switch] $InstallOnly ) if (-not $InstallOnly) { $FilePath = Start-DownloadWithRetry -Url $Url -Name $Name } $argumentList = ('/quiet', "`"$FilePath`"") Write-Host "Starting Install $Name..." $vsEdition = (Get-ToolsetContent).visualStudio.edition try { $installPath = ${env:ProgramFiles(x86)} if (Test-IsWin22) { $installPath = ${env:ProgramFiles} } #There are 2 types of packages at the moment - exe and vsix if ($Name -match "vsix") { $process = Start-Process -FilePath "${installPath}\Microsoft Visual Studio\${VSversion}\${vsEdition}\Common7\IDE\VSIXInstaller.exe" -ArgumentList $argumentList -Wait -PassThru } else { $process = Start-Process -FilePath ${env:Temp}\$Name /Q -Wait -PassThru } } catch { Write-Host "There is an error during $Name installation" $_ exit 1 } $exitCode = $process.ExitCode if ($exitCode -eq 0 -or $exitCode -eq 1001) # 1001 means the extension is already installed { Write-Host "$Name installed successfully" } else { Write-Host "Unsuccessful exit code returned by the installation process: $exitCode." exit 1 } #Cleanup downloaded installation files if (-not $InstallOnly) { Remove-Item -Force -Confirm:$false $FilePath } } function Get-VSExtensionVersion { Param ( [Parameter(Mandatory=$true)] [string] $packageName ) $instanceFolders = Get-ChildItem -Path "C:\ProgramData\Microsoft\VisualStudio\Packages\_Instances" if ($instanceFolders -is [array]) { Write-Host ($instanceFolders | Out-String) Write-Host ($instanceFolders | Get-ChildItem | Out-String) Write-Host "More than one instance installed" exit 1 } $stateContent = Get-Content -Path (Join-Path $instanceFolders.FullName '\state.packages.json') $state = $stateContent | ConvertFrom-Json $packageVersion = ($state.packages | Where-Object { $_.id -eq $packageName }).version if (-not $packageVersion) { Write-Host "Installed package $packageName for Visual Studio was not found" exit 1 } return $packageVersion } function Get-ToolsetContent { $toolsetPath = Join-Path "C:\\image" "toolset.json" $toolsetJson = Get-Content -Path $toolsetPath -Raw ConvertFrom-Json -InputObject $toolsetJson } function Get-ToolcacheToolDirectory { Param ([string] $ToolName) $toolcacheRootPath = Resolve-Path $env:AGENT_TOOLSDIRECTORY return Join-Path $toolcacheRootPath $ToolName } function Get-ToolsetToolFullPath { <# .DESCRIPTION Function that return full path to specified toolset tool. .PARAMETER Name The name of required tool. .PARAMETER Version The version of required tool. .PARAMETER Arch The architecture of required tool. #> Param ( [Parameter(Mandatory=$true)] [string] $Name, [Parameter(Mandatory=$true)] [string] $Version, [string] $Arch = "x64" ) $toolPath = Get-ToolcacheToolDirectory -ToolName $Name # Add wildcard if missing if ($Version.Split(".").Length -lt 3) { $Version += ".*" } $versionPath = Join-Path $toolPath $Version # Take latest installed version in case if toolset version contains wildcards $foundVersion = Get-Item $versionPath ` | Sort-Object -Property {[version]$_.name} -Descending ` | Select-Object -First 1 if (-not $foundVersion) { return $null } return Join-Path $foundVersion $Arch } function Get-WinVersion { (Get-CimInstance -ClassName Win32_OperatingSystem).Caption } function Test-IsWin22 { (Get-WinVersion) -match "2022" } function Test-IsWin19 { (Get-WinVersion) -match "2019" } function Extract-7Zip { Param ( [Parameter(Mandatory=$true)] [string]$Path, [Parameter(Mandatory=$true)] [string]$DestinationPath ) Write-Host "Expand archive '$PATH' to '$DestinationPath' directory" 7z.exe x "$Path" -o"$DestinationPath" -y | Out-Null if ($LASTEXITCODE -ne 0) { Write-Host "There is an error during expanding '$Path' to '$DestinationPath' directory" exit 1 } } function Install-AndroidSDKPackages { Param ( [Parameter(Mandatory=$true)] [string]$AndroidSDKManagerPath, [Parameter(Mandatory=$true)] [string]$AndroidSDKRootPath, [Parameter(Mandatory=$true)] [AllowEmptyCollection()] [string[]]$AndroidPackages, [string] $PrefixPackageName ) foreach ($package in $AndroidPackages) { & $AndroidSDKManagerPath --sdk_root=$AndroidSDKRootPath "$PrefixPackageName$package" } } function Get-AndroidPackages { Param ( [Parameter(Mandatory=$true)] [string]$AndroidSDKManagerPath ) return (cmd /c "$AndroidSDKManagerPath --list --verbose 2>&1").Trim() | Foreach-Object { $_.Split()[0] } | Where-Object {$_} } function Get-AndroidPackagesByName { Param ( [Parameter(Mandatory=$true)] [string[]]$AndroidPackages, [Parameter(Mandatory=$true)] [string]$PrefixPackageName ) return $AndroidPackages | Where-Object { "$_".StartsWith($PrefixPackageName) } } function Get-AndroidPackagesByVersion { Param ( [Parameter(Mandatory=$true)] [string[]]$AndroidPackages, [Parameter(Mandatory=$true)] [string]$PrefixPackageName, [object]$MinimumVersion, [char]$Delimiter, [int]$Index = 0 ) $Type = $MinimumVersion.GetType() $packagesByName = Get-AndroidPackagesByName -AndroidPackages $AndroidPackages -PrefixPackageName $PrefixPackageName $packagesByVersion = $packagesByName | Where-Object { ($_.Split($Delimiter)[$Index] -as $Type) -ge $MinimumVersion } return $packagesByVersion | Sort-Object { $_.Split($Delimiter)[$Index] -as $Type} -Unique } function Get-WindowsUpdatesHistory { $allEvents = @{} # 19 - Installation Successful: Windows successfully installed the following update # 20 - Installation Failure: Windows failed to install the following update with error # 43 - Installation Started: Windows has started installing the following update $filter = @{ LogName = "System" Id = 19, 20, 43 ProviderName = "Microsoft-Windows-WindowsUpdateClient" } $events = Get-WinEvent -FilterHashtable $filter -ErrorAction SilentlyContinue | Sort-Object Id foreach ( $event in $events ) { switch ( $event.Id ) { 19 { $status = "Successful" $title = $event.Properties[0].Value $allEvents[$title] = "" break } 20 { $status = "Failure" $title = $event.Properties[1].Value $allEvents[$title] = "" break } 43 { $status = "InProgress" $title = $event.Properties[0].Value break } } if ( $status -eq "InProgress" -and $allEvents.ContainsKey($title) ) { continue } [PSCustomObject]@{ Status = $status Title = $title } } } function Invoke-SBWithRetry { param ( [scriptblock] $Command, [int] $RetryCount = 10, [int] $RetryIntervalSeconds = 5 ) while ($RetryCount -gt 0) { try { & $Command return } catch { Write-Host "There is an error encountered:`n $_" $RetryCount-- if ($RetryCount -eq 0) { exit 1 } Write-Host "Waiting $RetryIntervalSeconds seconds before retrying. Retries left: $RetryCount" Start-Sleep -Seconds $RetryIntervalSeconds } } } function Get-GitHubPackageDownloadUrl { param ( [string]$RepoOwner, [string]$RepoName, [string]$BinaryName, [string]$Version, [string]$UrlFilter, [boolean]$IsPrerelease = $false, [int]$SearchInCount = 100 ) if ($Version -eq "latest") { $Version = "*" } $json = Invoke-RestMethod -Uri "https://api.github.com/repos/${RepoOwner}/${RepoName}/releases?per_page=${SearchInCount}" $tags = $json.Where{ $_.prerelease -eq $IsPrerelease -and $_.assets }.tag_name $versionToDownload = $tags | Select-String -Pattern "\d+.\d+.\d+" | ForEach-Object { $_.Matches.Value } | Where-Object { $_ -like "$Version.*" -or $_ -eq $Version } | Sort-Object { [version]$_ } | Select-Object -Last 1 if (-not $versionToDownload) { Write-Host "Failed to get a tag name from ${RepoOwner}/${RepoName} releases" exit 1 } $UrlFilter = $UrlFilter -replace "{BinaryName}",$BinaryName -replace "{Version}",$versionToDownload $downloadUrl = $json.assets.browser_download_url -like $UrlFilter return $downloadUrl }