Shared/Get-GitHubRepoRelease.ps1
function Get-GitHubRepoRelease { <# .SYNOPSIS Calls the GitHub Releases API passed via $Uri, validates the response and returns a formatted object Example: https://api.github.com/repos/PowerShell/PowerShell/releases/latest TODO: support Basic or OAuth authentication to GitHub #> [OutputType([System.Management.Automation.PSObject])] [CmdletBinding(SupportsShouldProcess = $false)] param ( [Parameter(Mandatory = $true, Position = 0)] [ValidateScript( { if ($_ -match "^(https://api\.github\.com/repos/)([a-zA-Z0-9_-]+)/([a-zA-Z0-9_-]+)(/tags|/releases)") { $true } else { throw "'$_' must be in the format 'https://api.github.com/repos/user/repository/releases/latest'. Replace 'user' with the user or organisation and 'repository' with the target repository name." } })] [System.String] $Uri, [Parameter(Mandatory = $false, Position = 1)] [ValidateNotNullOrEmpty()] [System.String] $MatchVersion = "(\d+(\.\d+){1,4}).*", [Parameter(Mandatory = $false, Position = 2)] [ValidateNotNullOrEmpty()] [System.String] $VersionTag = "tag_name", [Parameter(Mandatory = $false, Position = 3)] [ValidateNotNullOrEmpty()] [System.String] $Filter = "\.exe$|\.msi$|\.msp$|\.zip$", [Parameter(Mandatory = $false, Position = 4)] [System.Array] $VersionReplace, [Parameter()] [System.Management.Automation.SwitchParameter] $ReturnVersionOnly ) begin { $RateLimit = Get-GitHubRateLimit } process { if ($RateLimit.remaining -eq 0) { # We're rate limited, so output a special object [PSCustomObject] @{ Version = "RateLimited" URI = "https://docs.github.com/en/rest/overview/resources-in-the-rest-api#rate-limiting" } | Write-Output } else { try { # Retrieve the releases from the GitHub API # Use TLS for connections $SslProtocol = "Tls12" Write-Verbose -Message "$($MyInvocation.MyCommand): Set TLS to $SslProtocol." [Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::$SslProtocol # Invoke the GitHub releases REST API # Note that the API performs rate limiting. # https://docs.github.com/en/free-pro-team@latest/rest/reference/repos#get-the-latest-release $params = @{ ContentType = "application/vnd.github.v3+json" ErrorAction = "Stop" MaximumRedirection = 0 DisableKeepAlive = $true UseBasicParsing = $true UserAgent = "github-aaronparker-evergreen" Uri = $Uri } if (Test-ProxyEnv) { $params.Proxy = $script:EvergreenProxy } if (Test-ProxyEnv -Creds) { $params.ProxyCredential = $script:EvergreenProxyCreds } # If GITHUB_TOKEN or GH_TOKEN exists, let's add that to the API request if (Test-Path -Path "env:GITHUB_TOKEN") { $params.Headers = @{ Authorization = "token $env:GITHUB_TOKEN" } } elseif (Test-Path -Path "env:GH_TOKEN") { $params.Headers = @{ Authorization = "token $env:GH_TOKEN" } } Write-Verbose -Message "$($MyInvocation.MyCommand): Get GitHub release from: $Uri." $release = Invoke-RestMethod @params } catch { throw $_ } if ($null -eq $script:resourceStrings.Properties.GitHub) { Write-Warning -Message "$($MyInvocation.MyCommand): Unable to validate release against GitHub releases property object because we can't find the module resource." } else { # Validate that $release has the expected properties Write-Verbose -Message "$($MyInvocation.MyCommand): Validating GitHub release object." foreach ($item in $release) { # Compare the GitHub release object with properties that we expect $params = @{ ReferenceObject = $script:resourceStrings.Properties.GitHub DifferenceObject = (Get-Member -InputObject $item -MemberType NoteProperty) PassThru = $true ErrorAction = "Continue" } $missingProperties = Compare-Object @params # Throw an error for missing properties if ($null -ne $missingProperties) { Write-Verbose -Message "$($MyInvocation.MyCommand): Validated successfully." } else { Write-Verbose -Message "$($MyInvocation.MyCommand): Validation failed." $missingProperties | ForEach-Object { throw [System.Management.Automation.ValidationMetadataException]::New("$($MyInvocation.MyCommand): Property: '$_' missing") } } } } # Build and array of the latest release and download URLs Write-Verbose -Message "$($MyInvocation.MyCommand): Found $($release.count) release/s." Write-Verbose -Message "$($MyInvocation.MyCommand): Found $($release.assets.count) asset/s." if ($PSBoundParameters.ContainsKey("ReturnVersionOnly")) { if ($Uri -match "^*tags$") { try { # Uri matches tags fo the repo; find the latest tag $Version = [RegEx]::Match($release[0].name, $MatchVersion).Captures.Groups[1].Value } catch { Write-Verbose -Message "$($MyInvocation.MyCommand): Failed to match version number as-is, returning: $($release[0].$VersionTag)." $Version = $release[0].name } } else { try { # Uri matches releases for the repo; return just the version string $Version = [RegEx]::Match($release[0].$VersionTag, $MatchVersion).Captures.Groups[1].Value } catch { Write-Verbose -Message "$($MyInvocation.MyCommand): Failed to match version number as-is, returning: $($release[0].$VersionTag)." $Version = $item.$VersionTag } } if ($PSBoundParameters.ContainsKey("VersionReplace")) { # Replace string in version $Version = $Version -replace $res.Get.VersionReplace[0], $res.Get.VersionReplace[1] } # Build the output object $PSObject = [PSCustomObject] @{ Version = $Version } Write-Output -InputObject $PSObject } else { foreach ($item in $release) { foreach ($asset in $item.assets) { # Filter downloads by matching the RegEx in the manifest. The the RegEx may perform includes and excludes Write-Verbose -Message "$($MyInvocation.MyCommand): Match $Filter to $($asset.browser_download_url)." if ($asset.browser_download_url -match $Filter) { Write-Verbose -Message "$($MyInvocation.MyCommand): Building Windows release output object with: $($asset.browser_download_url)." # Capture the version string from the specified release tag try { Write-Verbose -Message "$($MyInvocation.MyCommand): Matching version number against: $($item.$VersionTag)." $Version = [RegEx]::Match($item.$VersionTag, $MatchVersion).Captures.Groups[1].Value Write-Verbose -Message "$($MyInvocation.MyCommand): Found version number: $Version." } catch { Write-Verbose -Message "$($MyInvocation.MyCommand): Failed to match version number, returning: $($item.$VersionTag)." $Version = $item.$VersionTag } if ($PSBoundParameters.ContainsKey("VersionReplace")) { # Replace string in version Write-Verbose -Message "$($MyInvocation.MyCommand): Replace $($res.Get.VersionReplace[0])." $Version = $Version -replace $res.Get.VersionReplace[0], $res.Get.VersionReplace[1] } # Build the output object $PSObject = [PSCustomObject] @{ Version = $Version Platform = Get-Platform -String $asset.browser_download_url Architecture = Get-Architecture -String $asset.browser_download_url Type = [System.IO.Path]::GetExtension($asset.browser_download_url).Split(".")[-1] InstallerType = Get-InstallerType -String $asset.browser_download_url Date = ConvertTo-DateTime -DateTime $item.created_at -Pattern "MM/dd/yyyy HH:mm:ss" Size = $asset.size URI = $asset.browser_download_url } if ($PSObject.Platform -eq "Windows") { Write-Output -InputObject $PSObject } } else { Write-Verbose -Message "$($MyInvocation.MyCommand): Skip: $($asset.browser_download_url)." } } } } } } } |