src/web.ps1

[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingInvokeExpression', '', Scope = 'Function', Target = 'Register-GitlabRunner')]
[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingPlainTextForPassword', '', Scope = 'Function', Target = 'Invoke-WebRequestBasicAuth')]
[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingUsernameAndPasswordParams', '', Scope = 'Function', Target = 'Invoke-WebRequestBasicAuth')]
[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSReviewUnusedParameter', '', Scope = 'Function', Target = 'Add-Metadata')]
[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSReviewUnusedParameter', '', Scope = 'Function', Target = 'Invoke-WebRequestBasicAuth')]
[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSReviewUnusedParameter', '', Scope = 'Function', Target = 'Out-Browser')]
[Diagnostics.CodeAnalysis.SuppressMessageAttribute('VariablePascalCase', '', Scope = 'Function', Target = 'Register-GitlabRunner')]
Param()

class Options {
    [String[]] GetProperties() {
        return $this | Get-Member -MemberType Properties | Select-Object -ExpandProperty Name
    }
    [PSObject] SetProperties($Object) {
        $this.GetProperties() | ForEach-Object { $Object.$_ = $this.$_ }
        return $Object
    }
}
class FormOptions: Options {
    [Int] $Width = 960
    [Int] $Height = 700
    [Int] $FormBorderStyle = 3
    [Double] $Opacity = 1.0
    [Bool] $ControlBox = $True
    [Bool] $MaximizeBox = $False
    [Bool] $MinimizeBox = $False
}
class BrowserOptions: Options {
    [String] $Anchor = 'Left,Top,Right,Bottom'
    [PSObject] $Size = @{ Height = 700; Width = 960 }
    [Bool] $IsWebBrowserContextMenuEnabled = $False
}
function Add-Metadata {
    <#
    .SYNOPSIS
    Identify certain elements and wrap them in semantic HTML tags.
    .EXAMPLE
    'My email is foo@bar.com' | ConvertTo-Html
    #>

    [CmdletBinding()]
    [OutputType([System.String])]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [String] $Text,
        [String[]] $Keyword,
        [Hashtable] $Abbreviations,
        [Switch] $Microformat,
        [ValidateSet('all', 'date', 'duration', 'email', 'url', 'ip')]
        [String[]] $Disable
    )
    Begin {
        $Custom = [Regex](($Keyword | ForEach-Object { "(\b${_}\b)" }) -join '|' )
        $Date = [Regex](New-RegexString -Date)
        $Duration = [RegEx](New-RegexString -Duration)
        $Email = [Regex](New-RegexString -Email)
        $Url = [Regex](New-RegexString -Url)
        $IpAdress = [Regex](New-RegexString -IPv4 -IPv6)
        $Attributes = @{
            Custom = 'itemprop="thing"'
            Date = 'itemscope itemtype="https://schema.org/DateTime" class="dt-event"'
            Duration = 'itemscope itemprop="event" itemtype="https://schema.org/Event" class="duration dt-event"'
            Email = 'itemscope itemprop="email" itemtype="https://schema.org/email" class="u-email"'
            End = 'itemscope itemprop="endTime" itemtype="https://schema.org/Time" class="dt-end"'
            Start = 'itemscope itemprop="startTime" itemtype="https://schema.org/Time" class="dt-start"'
            Url = 'itemscope itemprop="url" itemtype="https://schema.org/URL" class="u-url"'
        }
        $Options = [Text.RegularExpressions.RegexOptions]'IgnoreCase, CultureInvariant'
    }
    Process {
        If ($Keyword.Count -gt 0) {
            $Text = [Regex]::Replace(
                $Text,
                $Custom,
                {
                    Param($Match)
                    $Value = $Match.Value
                    $ClassName = $Value -replace '\s', '-'
                    if ($Microformat) {
                        "<span $($Attributes.Custom) class=`"keyword p-item`" data-keyword=`"${ClassName}`">${Value}</span>"
                    } else {
                        "<span class=`"keyword`" data-keyword=`"${ClassName}`">${Value}</span>"
                    }
                }
            )
        }
        if ($Abbreviations.Count -gt 0) {
            $Items = $Abbreviations.GetEnumerator()
            foreach ($Item in $Items) {
                $Name = $Item.Name
                $Value = $Item.Value
                $Text = [Regex]::Replace(
                    $Text,
                    "\b${Value}\b",
                    {
                        Param($Match)
                        $Value = $Match.Value
                        "<abbr title=`"${Name}`">${Value}</abbr>"
                    }
                )
            }
        }
        if ('all' -notin $Disable) {
            switch ($True) {
                { 'url' -notin $Disable } {
                    $Text = [Regex]::Replace(
                        $Text,
                        $Url,
                        {
                            Param($Match)
                            $Value = $Match.Groups[1].Value
                            if ($Microformat) {
                                "<a $($Attributes.Url) href=`"${Value}`">${Value}</a>"
                            } else {
                                "<a href=`"${Value}`">${Value}</a>"
                            }
                        },
                        $Options
                    )
                }
                { 'date' -notin $Disable } {
                    $Text = [Regex]::Replace(
                        $Text,
                        $Date,
                        {
                            Param($Match)
                            $Value = $Match.Groups[1].value
                            $Data = $Value | Test-Match -Date
                            $IsoValue = [DateTime]"$($Data.Month)/$($Data.Day)/$($Data.Year)" | ConvertTo-Iso8601
                            if ($Microformat) {
                                "<time $($Attributes.Date) datetime=`"${IsoValue}`">${Value}</time>"
                            } else {
                                "<time datetime=`"${IsoValue}`">${Value}</time>"
                            }
                        },
                        $Options
                    )
                }
                { 'duration' -notin $Disable } {
                    $Text = [Regex]::Replace(
                        $Text,
                        $Duration,
                        {
                            Param($Match)
                            $Value = $Match.Groups[1].value
                            $Data = $Value | Test-Match -Duration
                            $Start = $Data.Start
                            $End = $Data.End
                            $Timezone = if ($Data.IsZulu) { ' data-timezone="Zulu"' } else { '' }
                            if ($Microformat) {
                                "<span $($Attributes.Duration)${Timezone}><time $($Attributes.Start) datetime=`"${Start}`">${Start}</time> - <time $($Attributes.End) datetime=`"${End}`">${End}</time></span>"
                            } else {
                                "<span class=`"duration`"${Timezone}><time datetime=`"${Start}`">${Start}</time> - <time datetime=`"${End}`">${End}</time></span>"
                            }
                        },
                        $Options
                    )
                }
                { 'email' -notin $Disable } {
                    $Text = [Regex]::Replace(
                        $Text,
                        $Email,
                        {
                            Param($Match)
                            $Value = $Match.Groups[1].Value
                            if ($Microformat) {
                                "<a $($Attributes.Email) href=`"mailto:${Value}`">${Value}</a>"
                            } else {
                                "<a href=`"mailto:${Value}`">${Value}</a>"
                            }
                        },
                        $Options
                    )
                }
                { 'ip' -notin $Disable } {
                    $Text = [Regex]::Replace(
                        $Text,
                        $IpAdress,
                        {
                            Param($Match)
                            $Value = $Match.Groups[1].Value
                            "<a class=`"ip`" href=`"${Value}`">${Value}</a>"
                        },
                        $Options
                    )
                }
            }
        }
        $Text
    }
}
function ConvertFrom-ByteArray {
    <#
    .SYNOPSIS
    Converts bytes to human-readable text
    #>

    [CmdletBinding()]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [Array] $Data
    )
    Begin {
        function Invoke-Convert {
            Param(
                [Parameter(Position = 0)]
                $Data
            )
            if ($Data.Length -gt 0) {
                if ($Data -is [Byte] -or $Data[0] -is [Byte]) {
                    [System.Text.Encoding]::ASCII.GetString($Data)
                } else {
                    $Data
                }
            }
        }
        Invoke-Convert $Data
    }
    End {
        Invoke-Convert $Input
    }
}
function ConvertFrom-EpochDate () {
    <#
    .SYNOPSIS
    Converts epoch dates into datetime values
    .PARAMETER Epoch
    The epoch to use in conversion
    (Default value is '01.01.1970')
    .EXAMPLE
    '1577836800' | ConvertFrom-EpochDate -AsString
    # '1/1/20'
    .EXAMPLE
    '1577836800000000' | ConvertFrom-EpochDate -Microseconds -AsString
    # '1/1/20'
    #>

    [CmdletBinding()]
    [OutputType([System.String])]
    [OutputType([DateTime])]
    Param(
        [Parameter(Position = 0, ValueFromPipeline = $True)]
        [BigInt] $Value,
        [Switch] $Milliseconds,
        [Switch] $Microseconds,
        [Switch] $AsString,
        [String] $Epoch = '01.01.1970',
        [String] $Format = 'M/d/y'
    )
    $Units = if ($Milliseconds) {
        1000
    } elseif ($Microseconds) {
        1000000
    } else {
        1
    }
    $Result = (Get-Date $Epoch) + ([System.TimeSpan]::fromseconds($Value / $Units))
    if ($AsString) {
        $Result.ToString($Format)
    } else {
        $Result
    }
}
function ConvertFrom-Html {
    <#
    .SYNOPSIS
    Convert HTML string into object.
    .EXAMPLE
    '<html><body><h1>hello</h1></body></html>' | ConvertFrom-Html
    #>

    [CmdletBinding()]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [String] $Value
    )
    $Html = New-Object -ComObject 'HTMLFile'
    try {
        # This works in PowerShell with Office installed
        $Html.IHTMLDocument2_write($Value)
    } catch {
        # This works when Office is not installed
        $Content = [System.Text.Encoding]::Unicode.GetBytes($Value)
        $Html.Write($Content)
    }
    $Html
}
function ConvertFrom-QueryString {
    <#
    .SYNOPSIS
    Returns parsed query parameters
    #>

    [CmdletBinding()]
    [OutputType([Object[]])]
    [OutputType([String])]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [String] $Query
    )
    Begin {
        Use-Web
    }
    Process {
        $Decoded = [System.Web.HttpUtility]::UrlDecode($Query)
        if ($Decoded -match '=') {
            $Decoded -split '&' | Invoke-Reduce {
                Param($Acc, $Item)
                $Key, $Value = $Item -split '='
                $Acc.$Key = $Value.Trim()
            } -InitialValue @{}
        } else {
            $Decoded
        }
    }
}
function ConvertTo-Iso8601 {
    <#
    .SYNOPSIS
    Convert value to date in ISO 8601 format
    .NOTES
    See https://www.iso.org/iso-8601-date-and-time-format.html
    #>

    [CmdletBinding()]
    Param(
        [Parameter(Position = 0, ValueFromPipeline = $True)]
        [String] $Value
    )
    Process {
        $Value | Get-Date -UFormat '+%Y-%m-%dT%H:%M:%S.000Z'
    }
}
function ConvertTo-JavaScript {
    <#
    .SYNOPSIS
    Convert PowerShell values to JavaScript strings. It is similar to ConvertTo-Json, but with broader support for Prelude types.
    .EXAMPLE
    $A = [Node]'A'
    $B = [Node]'B'
    $A, $B | ConvertTo-JavaScript
    .EXAMPLE
    @{ foo = 'bar' } | ConvertTo-JavaScript
    # {"foo":"bar"}
    .NOTES
    The ConvertTo-JavaScript cmdlet is not intended to be used as a data serializer as data is removed during conversion.
    #>

    [CmdletBinding()]
    [OutputType([System.String])]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        $Value
    )
    Begin {
        $CoordinateTemplate = {
            Param($Value)
            "{latitude: $($Value.Latitude), longitude: $($Value.Longitude), height: $($Value.Height), hemisphere: '$($Value.Hemisphere -join '')'}"
        }
        $MatrixTemplate = {
            Param($Value)
            $Rows = $Value.Values.Real |
                Invoke-Chunk -Size $Value.Size[1] |
                ForEach-Object { $_ -join ', ' } |
                ForEach-Object { "[$_]" }
            "[$($Rows -join ', ')]"
        }
        $NodeTemplate = {
            Param($Value)
            "{id: '$($Value.Id)', label: '$($Value.Label)'}"
        }
        $EdgeTemplate = {
            Param($Value)
            $Source = (& $NodeTemplate -Value $Value.Source)
            $Target = (& $NodeTemplate -Value $Value.Target)
            "{source: $Source, target: $Target}"
        }
        $GraphTemplate = {
            Param($Value)
            "{nodes: $($Value.Nodes | ConvertTo-JavaScript), edges: $($Value.Edges | ConvertTo-JavaScript)}"
        }
        $DefaultTemplate = {
            Param($Value)
            $Value | ConvertTo-Json -Compress
        }
        function Invoke-Convert {
            Param($Value)
            $Type = $Value.GetType().Name
            $Template = switch ($Type) {
                'Coordinate' { $CoordinateTemplate }
                'Matrix' { $MatrixTemplate }
                'Node' { $NodeTemplate }
                'DirectedEdge' { $EdgeTemplate }
                'Edge' { $EdgeTemplate }
                'Graph' { $GraphTemplate }
                Default { $DefaultTemplate }
            }
            & $Template -Value $Value
        }
        switch ($Value.Count) {
            1 {
                Invoke-Convert -Value $Value
            }
            { $_ -gt 1 } {
                "[$(($Value | ForEach-Object { Invoke-Convert -Value $_ }) -join ', ')]"
            }
        }
    }
    End {
        switch ($Input.Count) {
            1 {
                Invoke-Convert -Value $Input[0]
            }
            { $_ -gt 1 } {
                "[$(($Input | ForEach-Object { Invoke-Convert -Value $_ }) -join ', ')]"
            }
        }
    }
}
function ConvertTo-QueryString {
    <#
    .SYNOPSIS
    Returns URL-encoded query string
    #>

    [CmdletBinding()]
    [OutputType([String])]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [PSObject] $InputObject,
        [Switch] $UrlEncode
    )
    Begin {
        Use-Web
    }
    Process {
        $Callback = {
            Param($Acc, $Item)
            $Key = $Item.Name
            $Value = $Item.Value
            "${Acc}$(if ($Acc -ne '') { '&' } else { '' })${Key}=${Value}"
        }
        $QueryString = $InputObject.GetEnumerator() | Sort-Object Name | Invoke-Reduce $Callback ''
        if (-not $QueryString) {
            $QueryString = ''
        }
        if ($UrlEncode) {
            Add-Type -AssemblyName System.Web
            [System.Web.HttpUtility]::UrlEncode($QueryString)
        } else {
            $QueryString
        }
    }
}
function Get-HostsContent {
    <#
    .SYNOPSIS
    Get and parse contents of hosts file
    .PARAMETER Path
    Specifies an alternate hosts path. Defaults to %SystemRoot%\System32\drivers\etc\hosts.
    .EXAMPLE
    Get-HostsContent
    .EXAMPLE
    Get-HostsContent '.\hosts'
    #>

    [CmdletBinding()]
    [OutputType([String])]
    Param(
        [Parameter(Position = 0, ValueFromPipeline = $True)]
        [ValidateScript( { Test-Path $_ })]
        [String] $Path
    )
    if (-not $Path) {
        $Path = if (-not $IsLinux) {
            Join-Path $Env:SystemRoot 'System32\drivers\etc\hosts'
        } else {
            '/etc/hosts'
        }
    }
    $CommentLine = '^\s*#'
    $HostLine = '^\s*(?<IPAddress>[01-9\.\:]+)\s+(?<Hostname>[^#]+)(\s*|\s+#(?<Comment>.*))$'
    $HomeAddress = [Net.IPAddress]'127.0.0.1'
    $LineNumber = 0
    (Get-Content $Path -ErrorAction Stop) | ForEach-Object {
        if (($_ -match $HostLine) -and ($_ -notmatch $CommentLine)) {
            $IpAddress = $Matches['IPAddress']
            $Comment = if ($Matches['Comment']) { $Matches['Comment'] } else { '' }
            $Result = [PSCustomObject]@{
                LineNumber = $LineNumber
                IPAddress = $IpAddress
                IsValidIP = [Net.IPAddress]::TryParse($IPAddress, [Ref] $HomeAddress)
                Hostname = $Matches['Hostname'].Trim()
                Comment = $Comment.Trim()
            }
            $Result.PSObject.TypeNames.Insert(0, 'Hosts.Entry')
            $Result
        }
        $LineNumber++
    }
}
function Get-HtmlElement {
    <#
    .SYNOPSIS
    Helper utility for getting elements as an array from HTML formatted input using tagname, id, or class name
    .EXAMPLE
    $Html | Get-HtmlElement 'div'
    .EXAMPLE
    $Html | Get-HtmlElement '.some-class'
    .EXAMPLE
    $Html | Get-HtmlElement '#some-identifier'
    #>

    [CmdletBinding()]
    [OutputType([System.Object[]])]
    Param(
        [Parameter(ValueFromPipeline = $True)]
        $InputObject,
        [Parameter(Mandatory = $True, Position = 0)]
        [String] $Selector
    )
    Process {
        $InputType = $InputObject.GetType().Name
        $Html = if ($InputType -eq 'String') {
            $InputObject | ConvertFrom-Html -Verbose:$False
        } else {
            $InputObject
        }
        $Elements = @()
        switch -Regex ($Selector) {
            '^[.].*' {
                $ClassName = $_ | Remove-Character -At 0
                foreach ($Element in $Html.all) {
                    if ($Element.className -eq $ClassName) {
                        $Elements += $Element
                    }
                }
                "==> [INFO] Found $($Elements.Count) element(s) with ${ClassName} class name" | Write-Verbose
            }
            '^#.*' {
                $Id = $_ | Remove-Character -At 0
                foreach ($Element in $Html.getElementById($Id)) {
                    $Elements += $Element
                }
                "==> [INFO] Found $($Elements.Count) element(s) with ${Id} ID" | Write-Verbose
            }
            Default {
                foreach ($Element in $Html.all.tags($Selector)) {
                    $Elements += $Element
                }
                "==> [INFO] Found $($Elements.Count) element(s) that match `"${_}`"" | Write-Verbose
            }
        }
        $Elements
    }
}
function Import-Html {
    <#
    .SYNOPSIS
    Import and parse an a local HTML file or web page.
    .EXAMPLE
    Import-Html example.com | ForEach-Object { $_.body.innerHTML }
    .EXAMPLE
    Import-Html .\bookmarks.html | ForEach-Object { $_.all.tags('a') } | Selelct-Object -ExpandProperty textContent
    #>

    [CmdletBinding()]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [Alias('Uri')]
        [String] $Path
    )
    if (Test-Path $Path) {
        $Content = Get-Content -Path $Path -Raw
    } else {
        $Content = (Invoke-WebRequest -Uri $Path).Content
    }
    ConvertFrom-Html $Content
}
function Invoke-WebRequestBasicAuth {
    <#
    .SYNOPSIS
    Invoke-WebRequest wrapper that makes it easier to use basic authentication
    .PARAMETER TwoFactorAuthentication
    Name of API that requires 2FA. Use 'none' when 2FA is not required.
    Possible values:
      - 'Github'
      - 'none' [Default]
    .PARAMETER Data
    Data (Body) payload for HTTP request. Will only function with PUT and POST requests.
    ==> Analogous to the '-d' cURL flag
    ==> Data object will be converted to JSON string
    .PARAMETER Session
    Name for custom web session variable.
    This cmdlet will try to remember the session variable, by default. Use -DisableSession to disable this behavior.
    See https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.utility/invoke-webrequest?view=powershell-7.2#example-2-use-a-stateful-web-service
    .PARAMETER Download
    Download content at URI.
    .PARAMETER WebRequestParameters
    Object for passing parameters to underlying invocation of Invoke-WebRequest
    Note: "Custom" is an alias for this parameter
    .EXAMPLE
    $Uri = 'https://api.github.com/notifications'
    $Query = @{ per_page = 100; page = 1 }
    $Response = Invoke-WebRequestBasicAuth $Uri -Token $Token -Query $Query -ParseContent
    $Response | Format-Table -AutoSize
 
    # Authenticate a GET request with a token
    .EXAMPLE
    $Uri = 'https://api.github.com/notifications'
    $Query = @{ per_page = 100; page = 1 }
    $Response = Invoke-WebRequestBasicAuth $Uri -Username $Username -Password $Token -Query $Query -ParseContent
    $Response | Format-Table -AutoSize
 
    # Use basic authentication with a username and password
    .EXAMPLE
    $Uri = 'https://api.github.com/notifications'
    @{ last_read_at = '' } | BasicAuth $Uri -Put -Token $Token
 
    # Execute a PUT request with a data payload
    .EXAMPLE
    $Uri = 'https://api.github.com/notifications'
    $Parameters = @{ SkipCertificateChecks = $True }
    @{ last_read_at = '' } | basicauth $Uri -Put -Token $Token -Custom $Parameters
    .EXAMPLE
    $Uri = 'https://db.ygoprodeck.com/api/v7/cardinfo.php
    basicauth $Uri -Query @{ name = 'Galaxy-Eyes Photon Dragon' } -ParseContent
 
    # Download and parse an API JSON response (can also parse HTML and CSV content)
    #>

    [CmdletBinding(DefaultParameterSetName = 'none', SupportsShouldProcess = $True)]
    [Alias('basicauth')]
    Param(
        [Parameter(ParameterSetName = 'basic')]
        [String] $Username,
        [Parameter(ParameterSetName = 'basic')]
        [String] $Password,
        [Parameter(ParameterSetName = 'token')]
        [String] $Token,
        [PSObject] $Headers = @{},
        [String] $Session = 'PreludeBasicAuthSession',
        [Switch] $DisableSession,
        [Parameter(Mandatory = $True, Position = 0)]
        [UriBuilder] $Uri,
        [PSObject] $Query = @{},
        [Switch] $UrlEncode,
        [Switch] $Download,
        [ValidateScript( { Test-Path $_ })]
        [String] $Folder = (Get-Location).Path,
        [Switch] $ParseContent,
        [Alias('OTP', '2FA')]
        [Switch] $TwoFactorAuthentication,
        [Switch] $Get,
        [Switch] $Post,
        [Switch] $Put,
        [Switch] $Delete,
        [Switch] $Github,
        [Switch] $Gitlab,
        [Parameter(ValueFromPipeline = $True)]
        [PSObject] $Data = @{},
        [Alias('Custom')]
        [PSObject] $WebRequestParameters = @{}
    )
    Begin {
        function Get-ParsedContent {
            Param(
                [Parameter(Mandatory = $True, Position = 0)]
                $Request
            )
            $Content = $Request.Content
            $Type = $Request.Headers.'Content-Type'
            switch -Regex ($Type) {
                '^text\/csv' {
                    $Content | ConvertFrom-Csv
                }
                '^text\/html' {
                    $Content | ConvertFrom-Html
                }
                '^application\/xhtml[+]xml' {
                    $Content | ConvertFrom-Html
                }
                '^application\/json' {
                    $Content | ConvertFrom-Json
                }
                '^text\/(plain|css|javascript)' {
                    "==> [INFO] Cannot parse content of type, ${Type}" | Write-Verbose
                    $Content
                }
                '^(image|video|font|audio|model)\/' {
                    "==> [INFO] Cannot parse content of type, ${Type}" | Write-Verbose
                    $Content
                }
                Default {
                    "==> [WARN] Unable to resolve type, ${Type}" | Write-Warning
                    $Content
                }
            }
        }
    }
    Process {
        if ($PSBoundParameters.ContainsKey('Password') -or $PSBoundParameters.ContainsKey('Token')) {
            $HeaderData = if ($Gitlab) {
                @{ 'PRIVATE-TOKEN' = $Token }
            } else {
                $Authorization = if ($Token.Length -gt 0) {
                    "Bearer $Token"
                } else {
                    $Credential = [Convert]::ToBase64String([System.Text.Encoding]::Ascii.GetBytes("${Username}:${Password}"))
                    "Basic $Credential"
                }
                @{ 'Authorization' = $Authorization }
            }
            $Headers = $Headers, $HeaderData | Invoke-ObjectMerge
        }
        if ($TwoFactorAuthentication) {
            if ($Github) {
                if ($PSCmdlet.ShouldProcess('GitHub 2FA')) {
                    'GitHub 2FA' | Write-Title -Green
                    $Code = 'Code:' | Invoke-Input -Number -Indent 4
                    $Headers.Accept = 'application/vnd.github.v3+json'
                    $Headers['x-github-otp'] = $Code
                } else {
                    '==> [DRYRUN] Would have set Accept and x-github-otp headers' | Write-Color -DarkGray
                }
            }
        }
        $Method = Find-FirstTrueVariable 'Get', 'Post', 'Put', 'Delete'
        $QueryString = $Query | ConvertTo-QueryString -UrlEncode:$UrlEncode
        $Parameters = @{
            Headers = $Headers
            Method = $Method
            Uri = "$($Uri.Uri)$(if ($QueryString.Length -gt 0) { '?' })${QueryString}"
        }
        if ($Method -in 'Post', 'Put') {
            $Parameters.Body = $Data | ConvertTo-Json
        }
        if (-not $DisableSession) {
            $WebSession = Get-Variable -Name $Session -ValueOnly -ErrorAction Ignore
            if ($WebSession -is [Microsoft.PowerShell.Commands.WebRequestSession]) {
                $Parameters.WebSession = $WebSession
            } else {
                $Parameters.SessionVariable = $Session
            }
        }
        $OutFile = Join-Path $Folder (Split-Path $Uri.Path -Leaf)
        if ($Download) {
            $Parameters.OutFile = $OutFile
        }
        if ($PSCmdlet.ShouldProcess('Invoke-WebRequest')) {
            $Parameters, $WebRequestParameters | Invoke-ObjectMerge | ConvertTo-Json | Write-Verbose
            $Request = Invoke-WebRequest @Parameters @WebRequestParameters -UseBasicParsing
        } else {
            '==> [DRYRUN] Would have called Invoke-WebRequest with the parameters:' | Write-Color -DarkGray
            $Parameters, $WebRequestParameters | Invoke-ObjectMerge | ConvertTo-Json | Write-Color -DarkGray
        }
        if ($ParseContent) {
            if ($PSCmdlet.ShouldProcess('Parse content')) {
                $Content = Get-ParsedContent $Request
                if ($Download) {
                    $Parameters = @{
                        Encoding = 'default'
                        FilePath = $OutFile
                    }
                    $Content | Out-File @Parameters | Out-Null
                } else {
                    $Content
                }
            } else {
                '==> [DRYRUN] Would have returned parsed response content' | Write-Color -DarkGray
            }
        } else {
            if ($PSCmdlet.ShouldProcess('Return request response')) {
                $Request
            } else {
                '==> [DRYRUN] Would have returned response' | Write-Color -DarkGray
            }
        }
    }
}
function New-GitlabRunner {
    <#
    .SYNOPSIS
    Create a new GitLab runner using the GitLab API (as opposed to using the GitLab GUI).
    .EXAMPLE
    [PSCustomObject]@{
        'Token' = $Token
        'Endpoint' = 'https://gitlab.com'
        'RunnerType' = 'Group'
        'Identifier' = $GroupId
        'Description' = 'My first runner'
    } | New-GitlabRunner
 
    .EXAMPLE
    Get-Content '.\runners.json' |
        ConvertFrom-Json |
        Get-Property 'items' |
        New-GitlabRunner -Token $PERSONAL_ACCESS_TOKEN
    #>

    [CmdletBinding(SupportsShouldProcess = $True)]
    [OutputType([String])]
    Param(
        [ValidateSet('https://gitlab.com', 'https://code.ornl.gov')]
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
        [Alias('url')]
        [String] $Endpoint,
        [ValidateLength(20, 20)]
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
        [String] $Token,
        [ValidateSet('Group', 'Project')]
        [Parameter(Mandatory = $False)]
        [Alias('type')]
        [String] $RunnerType = 'Group',
        [ValidateLength(5, 5)]
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
        [Alias('id')]
        [String] $Identifier,
        [Parameter(Mandatory = $False, ValueFromPipelineByPropertyName = $True)]
        [String] $Description = 'Runner created with PowerShell',
        [Parameter(Mandatory = $False, ValueFromPipelineByPropertyName = $True)]
        [Switch] $Gpu = $False
    )
    Process {
        $Uri = "${Endpoint}/api/v4/user/runners"
        $RunnerData = switch ($RunnerType) {
            'Group' {
                @{
                    'group_id' = $Identifier
                    'runner_type' = 'group_type'
                }
            }
            'Project' {
                @{
                    'project_id' = $Identifier
                    'runner_type' = 'project_type'
                }
            }
        }
        $TagData = if ($Gpu.IsPresent) {
            @{ 'tag_list' = @('gpu') }
        } else {
            @{ 'tag_list' = @('cpu') }
        }
        $TagData | ConvertTo-Json | Write-Verbose
        $Query = $RunnerData, $TagData, @{
            'run_untagged' = $True
            'description' = $Description
        } | Invoke-ObjectMerge
        $Parameters = @{
            'Uri' = $Uri
            'Query' = $Query
            'Token' = $Token
            'Post' = $True
            'Gitlab' = $True
            'ParseContent' = $True
        }
        '==> GitLab API request parameters:' | Write-Verbose
        $Parameters | ConvertTo-Json | Write-Verbose
        if ($PSCmdlet.ShouldProcess('Make request to GitLab API to create a runner')) {
            $Response = Invoke-WebRequestBasicAuth @Parameters
            '==> GitLab API response:' | Write-Verbose
            $Response | ConvertTo-Json | Write-Verbose
            $Output = $Response, @{ 'gpu' = $Gpu.IsPresent } | Invoke-ObjectMerge
            $Output | ConvertTo-Json | Write-Verbose
            $Output
        }
    }
}
function Out-Browser {
    <#
    .SYNOPSIS
    Display HTML content (string, file, or URI) in a web browser Windows form. Out-Browser will auto-detect content type.
    Returns [System.Windows.Forms.HtmlDocument] object
    .PARAMETER OnShown
    Function to be executed once form is shown. $Form and $Browser variables are available within function scope.
    .PARAMETER OnComplete
    Function to be executed whenever the Document within the browser is loaded. $Form and $Browser variables are available within function scope.
    .PARAMETER OnClose
    Function to be executed immediately before form is disposed. $Form and $Browser variables are available within function scope.
    .PARAMETER Default
    Use operating system default browser (i.e. Firefox, Chrome, etc...) instead of WebBrowser control.
    Note: OnShown, OnComplete, and OnClose will not be run when the Default parameter is used.
    .PARAMETER PassThru
    - Return WebBrowser document object when not using -Default parameter
    - Return process when using -Default parameter
    .EXAMPLE
    '<h1>Hello World</h1>' | Out-Browser
    .EXAMPLE
    'https://google.com' | Out-Browser
    .EXAMPLE
    '.\file.html' | Out-Browser
    .EXAMPLE
    $OnClose = {
        Param($Browser)
        $Browser.Document.GetElementsByTagName('h1').innerText | Write-Color -Green
    }
    '<h1 contenteditable="true">Type Here</h1>' | Out-Browser -OnClose $OnClose | Out-Null
    #>

    [CmdletBinding()]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [String] $Content,
        [FormOptions] $FormOptions = @{},
        [BrowserOptions] $BrowserOptions = @{},
        [ScriptBlock] $OnShown = {},
        [ScriptBlock] $OnComplete = {},
        [ScriptBlock] $OnClose = {},
        [Switch] $Default,
        [Switch] $PassThru
    )
    Begin {
        Use-Web -Browser
        $Form = $FormOptions.SetProperties((New-Object 'Windows.Forms.Form'))
        $Browser = $BrowserOptions.SetProperties((New-Object 'Windows.Forms.WebBrowser'))
        $Browser.Size = @{ Width = $Form.Width; Height = $Form.Height }
        $Form.Controls.Add($Browser)
        $ShownCallback = {
            '==> Form shown' | Write-Verbose
            $Form.BringToFront()
            & $OnShown -Form $Form -Browser $Browser
        }
        $CompletedCallback = {
            "==> Document load complete ($($_.Url))" | Write-Verbose
            & $OnComplete -Form $Form -Browser $Browser
        }
        $Form.Add_Shown($ShownCallback);
        $Browser.Add_DocumentCompleted($CompletedCallback)
    }
    Process {
        "==> Browser is $(if($Browser.IsOffline) { 'OFFLINE' } else { 'ONLINE' })" | Write-Verbose
        $IsFile = if (Test-Path $Content -IsValid) { Test-Path $Content } else { $False }
        $IsUri = ([Uri]$Content).IsAbsoluteUri
        if ($Default) {
            $FilePath = if ($IsFile -or $IsUri) {
                $Content
            } else {
                $TempRoot = if ($IsLinux) { '/tmp' } else { $Env:temp }
                $Path = Join-Path $TempRoot 'content.html'
                $Content | Set-Content -Path $Path
                $Path
            }
            $Process = Start-Process -FilePath $FilePath -PassThru
            if ($PassThru) {
                return $Process
            }
        } else {
            if ($IsFile) {
                "==> Opening ${Content}..." | Write-Verbose
                $Browser.Navigate("file:///$(Get-Item $Content | Get-StringPath)")
            } elseif ($IsUri) {
                "==> Navigating to ${Content}..." | Write-Verbose
                $Browser.Navigate([Uri]$Content)
            } else {
                '==> Opening HTML in WebBrowser control...' | Write-Verbose
                $Browser.DocumentText = "$Content"
            }
            if ($Form.ShowDialog() -ne 'OK') {
                $Document = $Browser.Document
                '==> Browser closing...' | Write-Verbose
                & $OnClose -Form $Form -Browser $Browser
                $Form.Dispose()
                '==> Form disposed' | Write-Verbose
                if ($PassThru) {
                    return $Document
                }
            }
        }
    }
}
function Register-GitlabRunner {
    <#
    .SYNOPSIS
    Register new GitLab docker executor runner within a gitlab/gitlab-runner Docker container.
    .DESCRIPTION
    - Docker must be installed and running
    - If gum is not installed, Force will effectively be treated as True
    .PARAMETER Token
    GitLab runner registration token
    https://docs.gitlab.com/runner/register/index.html
    .EXAMPLE
    Register-GitlabRunner -Token $Token -Identifier $RunnerId
    .EXAMPLE
    [PSCustomObject]@{
        'Token' = $Token
        'RunnerType' = 'Group'
        'Identifier' = $GroupId
        'Endpoint' = $Endpoint
        'Description' = $Description
    } | New-GitlabRunner | Register-GitlabRunner
    #>

    [CmdletBinding()]
    Param(
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
        [String] $Token,
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
        [Alias('id')]
        [String] $Identifier,
        [Parameter(Mandatory = $False, ValueFromPipelineByPropertyName = $True)]
        [Switch] $Gpu = $False,
        [Switch] $Force
    )
    Begin {
        $Tasks = @('Create', 'Register')
        $Image = 'gitlab/gitlab-runner:latest'
        $Passthru = '/var/run/docker.sock:/var/run/docker.sock'
        $LogLevel = if ($PSCmdlet.MyInvocation.BoundParameters.Verbose) { 'debug' } else { 'panic' }
    }
    Process {
        $Url = 'https://code.ornl.gov'
        $GpuOptions = if ($Gpu.IsPresent) { @{ 'gpus' = 'all' } } else { @{} }
        $Parameters = @{
            'Create' = $GpuOptions, @{
                'detach' = $True
                'name' = "runner_${Identifier}"
                'restart' = 'always'
                'volume' = '/srv/gitlab-runner/config:/etc/gitlab-runner'
            } | Invoke-ObjectMerge | ConvertTo-ParameterString
            'Register' = @{
                'non-interactive' = $True
                'url' = $Url
                'token' = $Token
                'executor' = 'docker'
                'docker-image' = 'docker'
                'description' = "'GitLab Runner [${Identifier}]'"
                'docker-volumes' = $Passthru
            } | ConvertTo-ParameterString
        }
        $Command = @{
            'Create' = "docker run $($Parameters.Create) --volume ${Passthru} ${Image}"
            'Register' = "docker exec runner_${Identifier} gitlab-runner --log-level ${LogLevel} register $($Parameters.Register)"
        }
        $Execute = if (-not $Force -and (Test-Command 'gum')) {
            gum confirm 'Execute Docker commands?'; if ($?) { $True } else { $False }
        } else {
            $True
        }
        if ($Execute) {
            $Tasks | ForEach-Object {
                '==> Executing command:' | Write-Verbose
                $Command[$_] | Write-Verbose
                Invoke-Expression $Command[$_] | Out-Null
            }
        } else {
            '==>[ERROR] User cancelled operation!' | Write-Color -Red
            $Parameters = @{
                'Delete' = $True
                'Gitlab' = $True
                'Token' = $Token
                'Uri' = "${Url}/api/v4/runners/${Identifier}"
            }
            Invoke-WebRequestBasicAuth @Parameters
        }
    }
}
function Save-File {
    <#
    .SYNOPSIS
    Download and save a file from a local or remote location.
    .PARAMETER SleepInterval
    Initial number of seconds to wait before checking if BitsTransfer job is complete.
    .PARAMETER WebClient
    Use .NET Web Client class instead of BitsTransfer.
    See https://docs.microsoft.com/en-us/dotnet/api/system.net.webclient
    .PARAMETER CustomParameters
    Parameters to pass to Start-BitsTransfer.
    .EXAMPLE
    'https://storage.googleapis.com/ygoprodeck.com/pics/93717133.jpg' | Save-File 'dragon.jpg'
    .EXAMPLE
    'https://storage.googleapis.com/ygoprodeck.com/pics/93717133.jpg' | Save-File 'dragon.jpg' -Asynchronous -Priority 'High'
    #>

    [CmdletBinding(DefaultParameterSetName = 'normal', SupportsShouldProcess = $True)]
    Param(
        [Parameter(Mandatory = $True, ValueFromPipeline = $True)]
        [UriBuilder] $Uri,
        [ValidateScript( { Test-Path $_ })]
        [String] $Destination = (Get-Location).Path,
        [Parameter(Position = 0)]
        [String[]] $Filename,
        [Parameter(ParameterSetName = 'asynchronous')]
        [Switch] $Asynchronous,
        [Parameter(ParameterSetName = 'asynchronous')]
        [Switch] $PassThru,
        [Parameter(ParameterSetName = 'asynchronous')]
        [ValidateSet('Foreground', 'High', 'Normal', 'Low')]
        [String] $Priority = 'Foreground',
        [Parameter(ParameterSetName = 'asynchronous')]
        [Int] $SleepInterval = 1,
        [Switch] $WebClient,
        [PSObject] $CustomParameters = @{}
    )
    Begin {
        function Format-FileVersion {
            Param(
                [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
                [String] $Name
            )
            $Elapsed = [DateTimeOffset]::UtcNow.ToUnixTimeMilliseconds()
            $Extension = [System.IO.Path]::GetExtension($Name)
            $Filename = $Name.Substring(0, $Name.Length - $Extension.Length)
            "${Filename}-${Elapsed}${Extension}"
        }
        function Test-JobComplete {
            Param(
                [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
                $BitsJob
            )
            $State = $BitsJob.JobState
            ($State -ne 'Transferring') -and ($State -ne 'Connecting')
        }
        $Count = 0
        $CanUseBitsTransfer = Test-Command 'Start-BitsTransfer' -Silent
        $Client = if ($CanUseBitsTransfer -and (-not $WebClient)) {
            '==> [INFO] Using Start-BitsTransfer' | Write-Verbose
            $Null
        } else {
            '==> [INFO] Using .NET Web Client class' | Write-Verbose
            New-Object 'System.Net.WebClient'
        }
    }
    Process {
        $Name = if ($Filename.Count -gt 0) { $Filename[$Count] } else { Split-Path $Uri.Path -Leaf }
        $Path = Join-Path $Destination $Name
        if (Test-Path -Path $Path) {
            $Name = $Name | Format-FileVersion
            $Path = Join-Path $Destination $Name
        }
        if ($Client) {
            if ($PSCmdlet.ShouldProcess($Path)) {
                if ($Asynchronous) {
                    $Client.DownloadFileAsync($Uri.Uri, $Path)
                } else {
                    $Client.DownloadFile($Uri.Uri, $Path)
                }
                "==> [INFO] Saved file to ${Path} using WebClient." | Write-Verbose
            }
        } elseif ($CanUseBitsTransfer) {
            $Parameters = @{
                Asynchronous = $Asynchronous
                Destination = $Path
                DisplayName = 'PreludeBitsJob'
                Priority = $Priority
                Source = $Uri.Uri
                TransferType = 'Download'
            }
            $Job = Start-BitsTransfer @Parameters @CustomParameters
            if ($Asynchronous) {
                $Id = $Job.JobId
                "==> [INFO] Finishing BitsTransfer job [${Id}]..." | Write-Verbose
                if ($PassThru) {
                    return $Job
                } else {
                    $Seconds = $SleepInterval
                    while (-not (Test-JobComplete $Job)) {
                        "==> [INFO] BitsTransfer status [${Id}]: $($Job.JobState)" | Write-Verbose
                        Start-Sleep -Seconds $Seconds
                        $Seconds += 1
                    }
                    switch ($Job.JobState) {
                        'Transferred' {
                            Complete-BitsTransfer -BitsJob $Job
                            "==> [INFO] BitsTransfer Job [${Id}], complete." | Write-Verbose
                        }
                        'Error' {
                            "==> [ERROR] BitsTransfer Job [${Id}], failed." | Write-Error
                            $Job | Format-List
                        }
                        Default {
                            # Do nothing
                        }
                    }
                }
            }
            "==> [INFO] Saved file to ${Path} using BitsTransfer." | Write-Verbose
        }
        $Count += 1
    }
}
function Test-Url {
    <#
    .SYNOPSIS
    Test if a URL is accessible
    .PARAMETER Code
    Return status code as a string instead of boolean value
    .PARAMETER WebRequestParameters
    Object for passing parameters to underlying invocation of Invoke-WebRequest
    .EXAMPLE
    'https://google.com' | Test-Url
    #>

    [CmdletBinding()]
    [OutputType([Bool])]
    [OutputType([String])]
    Param(
        [Parameter(Mandatory = $True, Position = 0, ValueFromPipeline = $True)]
        [UriBuilder] $Value,
        [Switch] $Code,
        [PSObject] $WebRequestParameters = @{}
    )
    Process {
        $Response = try {
            Invoke-WebRequestBasicAuth -Uri $Value.Uri -WebRequestParameters $WebRequestParameters
        } catch {
            @{ StatusCode = '404' }
        }
        $StatusCode = $Response | Get-Property StatusCode
        switch ($StatusCode) {
            200 {
                if ($Code) { '200' } else { $True }
            }
            Default {
                if ($Code) { $StatusCode.ToString() } else { $False }
            }
        }
    }
}
function Update-HostsFile {
    <#
    .SYNOPSIS
    Update and/or add entries of a hosts file.
    .PARAMETER Path
    Specifies an alternate hosts path. Defaults to %SystemRoot%\System32\drivers\etc\hosts.
    .PARAMETER PassThru
    Outputs parsed HOSTS file upon completion.
    .EXAMPLE
    Update-HostsFile -IPAddress '127.0.0.1' -Hostname 'c2.evil.com'
    .EXAMPLE
    Update-HostsFile -IPAddress '127.0.0.1' -Hostname 'c2.evil.com' -Comment 'Malware C2'
    #>

    [CmdletBinding(SupportsShouldProcess = $True)]
    Param(
        [Parameter(Mandatory = $True, Position = 0)]
        [Alias('IP')]
        [Net.IpAddress] $IPAddress,
        [Parameter(Mandatory = $True, Position = 1)]
        [ValidateNotNullOrEmpty()]
        [Alias('Name')]
        [String] $Hostname,
        [Parameter(Position = 2)]
        [String] $Comment,
        [ValidateScript( { Test-Path $_ })]
        [String] $Path = (Join-Path $Env:SystemRoot 'System32\drivers\etc\hosts'),
        [Switch] $PassThru
    )
    $Raw = Get-Content $Path
    $Hosts = Get-HostsContent $Path
    $Comment = if ($Comment) { "# $Comment" } else { '' }
    $Entry = "$IpAddress $Hostname $Comment"
    $HostExists = $Hostname -in $Hosts.Hostname
    $Hosts | Where-Object { $_.Hostname -eq $Hostname } | ForEach-Object {
        if ($_.IpAddress -eq $IPAddress) {
            "The hostname, '$Hostname', and IP address, '$IPAddress', already exist in $Path." | Write-Verbose
        } else {
            if ($PSCmdlet.ShouldProcess($Path)) {
                "Replacing hostname, '$Hostname', in $Path." | Write-Verbose
                $Raw[$_.LineNumber] = $Entry
            } else {
                "==> Would be replacing hostname, '$Hostname', in $Path." | Write-Color -DarkGray
            }
        }
    }
    if (-not $HostExists) {
        if ($PSCmdlet.ShouldProcess($Path)) {
            "Appending '$Hostname' at '$IPAddress' to $Path." | Write-Verbose
            $Raw += "`n$Entry"
        } else {
            "==> Would be appending '$Hostname' at '$IPAddress' to $Path." | Write-Color -DarkGray
        }
    }
    $Raw | Out-File -Encoding ascii -FilePath $Path -ErrorAction Stop
    if ($PassThru) {
        Get-HostsContent $Path
    }
}
function Use-Web {
    <#
    .SYNOPSIS
    Load related types for using web (with or without a web browser), if types are not already loaded.
    .PARAMETER Browser
    Whether or not to load WebBrowser type
    #>

    [CmdletBinding()]
    Param(
        [Switch] $Browser,
        [Switch] $PassThru
    )
    if (-not ('System.Web.HttpUtility' -as [Type])) {
        '==> Adding System.Web types' | Write-Verbose
        Add-Type -AssemblyName System.Web
    } else {
        '==> System.Web is already loaded' | Write-Verbose
    }
    if ($Browser) {
        if (-not ('System.Windows.Forms.WebBrowser' -as [Type])) {
            '==> Adding System.Windows.Forms types' | Write-Verbose
            Add-Type -AssemblyName System.Windows.Forms
        } else {
            '==> System.Windows.Forms is already loaded' | Write-Verbose
        }
    }
    if ($PassThru) {
        $True
    }
}