AzStackHciRemoteSupport/AzStackHCI.RemoteSupport.Helpers.psm1

<#############################################################
 # #
 # Copyright (C) Microsoft Corporation. All rights reserved. #
 # #
 #############################################################>

Import-LocalizedData -BindingVariable lvsTxt -FileName AzStackHCI.RemoteSupport.Strings.psd1

class HealthModel
{
    # Attributes for Azure Monitor schema
    [string]$Name #Name of the individual test/rule/alert that was executed. Unique, not exposed to the customer.
    [string]$Title #User-facing name; one or more sentences indicating the direct issue.
    [string]$Severity #Severity of the result (Critical, Warning, Informational, Hidden) - this answers how important the result is. Critical is the only update-blocking severity.
    [string]$Description #Detailed overview of the issue and what impact the issue has on the stamp.
    [psobject]$Tags #Key-value pairs that allow grouping/filtering individual tests. For example, "Group": "ReadinessChecks", "UpdateType": "ClusterAware"
    [string]$Status #The status of the check running (i.e. Failed, Succeeded, In Progress) - this answers whether the check ran, and passed or failed.
    [string]$Remediation #Set of steps that can be taken to resolve the issue found.
    [string]$TargetResourceID #The unique identifier for the affected resource (such as a node or drive).
    [string]$TargetResourceName #The name of the affected resource.
    [datetime]$Timestamp #The Time in which the HealthCheck was called.
    [psobject[]]$AdditionalData #Property bag of key value pairs for additional information.
    [string]$HealthCheckSource #The name of the services called for the HealthCheck (I.E. Test-AzureStack, Test-Cluster).
}

class EnableRemoteSupportAccessTarget : HealthModel
{
    #Additional Attribute
    [string]$State
    [string]$CreatedAt
    [string]$UpdatedAt
    [string]$ConnectionStatus
    [string]$ConnectionErrorMessage
    [string]$TargetService
    [string]$AccessLevel
    [string]$ExpiresAt
}

class GetRemoteSupportAccessTarget : HealthModel
{
    #Additional Attribute
    [string]$State
    [string]$CreatedAt
    [string]$UpdatedAt
    [string]$ConnectionStatus
    [string]$ConnectionErrorMessage
    [string]$TargetService
    [string]$AccessLevel
    [string]$ExpiresAt
}

class GetRemoteSupportSessionHistoryTarget : HealthModel
{
    #Additional Attribute
    [string]$SessionId
    [string]$NodeName
    [string]$StartTime
    [string]$EndTime
    [string]$TargetService
    [string]$AccessLevel
    [string]$TranscriptPath
}

function Invoke-InstallModule
{
    param (
        [Parameter(Mandatory=$true)]
        [string]
        $ModuleName
    )

    try
    {
        $module = Get-Module -Name $ModuleName -ListAvailable | Sort-Object -Property Version -Descending | Microsoft.PowerShell.Utility\Select-Object "Name", "Version"

        if([string]::IsNullOrWhiteSpace($module))
        {
            Trace-Execution "Installing $ModuleName module..."
            Install-Module $ModuleName -Force -AllowClobber | Out-Null
        }
        else
        {
            Trace-Execution "$ModuleName module already exists $($module.Name) $($module.Version)"
        }
    }
    catch
    {
        $exception = $_
        Trace-Execution "Install module failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)" 
        throw $exception
    }
}

function Install-AzStackHciRemoteSupport
{
    param (
    )

    try
    {
        $OperationType = $MyInvocation.MyCommand

        # Install Az.StackHCI module if doesn't exist
        Invoke-InstallModule -ModuleName "Az.StackHCI"

        # Install Remote Support
        Az.StackHCI\Install-AzStackHCIRemoteSupport
    }
    catch
    {
        $exception = $_
        Trace-Execution "Install-AzStackHciRemoteSupport failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)"
        throw $exception
    }
}


function Enable-AzStackHciRemoteSupport
{
    param (
        [Parameter(Mandatory=$true)]
        [ValidateSet("Diagnostics","DiagnosticsRepair")]
        [string]
        $AccessLevel,

        [Parameter(Mandatory=$false)]
        [int]
        $ExpireInMinutes = 480,

        [Parameter(Mandatory=$false)]
        [string]
        $SasCredential,

        [Parameter(Mandatory=$false)]
        [switch]
        $AgreeToRemoteSupportConsent
    )

    try
    {
        $OperationType = $MyInvocation.MyCommand

        # Install Az.StackHCI module if doesn't exist
        Invoke-InstallModule -ModuleName "Az.StackHCI"

        # Install Remote Support
        Az.StackHCI\Install-AzStackHCIRemoteSupport

        # Enable Remote Support
        $cmdResult = Az.StackHCI\Enable-AzStackHCIRemoteSupport -AccessLevel $AccessLevel -ExpireInMinutes $ExpireInMinutes -SasCredential $SasCredential -AgreeToRemoteSupportConsent:$AgreeToRemoteSupportConsent

        # Build Result
        $now = [datetime]::UtcNow
        $targetComputerName = $ENV:COMPUTERNAME

        $aggregateStatus = if( -not ([string]::IsNullOrWhiteSpace($cmdResult)) -and $cmdResult.State -eq "Active" -and $cmdResult.ConnectionStatus -eq "Connected" -or "Connecting") {'Succeeded'} else {'Failed'}

        $enableResult = New-Object -Type EnableRemoteSupportAccessTarget -Property @{
            Name = 'AzStackHci_Enable_RemoteSupport'
            Title = 'Standalone Remote Support: Enable'
            Severity = 'Info'
            Description = 'Enable AzStackHCI Remote Support'
            Tags = $OperationType
            TargetResourceID = $targetComputerName
            TargetResourceName = $targetComputerName
            Timestamp = $now
            Status = $aggregateStatus
            HealthCheckSource = ((Get-PSCallStack)[-1].Command)
            AdditionalData = $cmdResult
            State = $cmdResult.State
            CreatedAt = $cmdResult.CreatedAt
            UpdatedAt = $cmdResult.UpdatedAt
            ConnectionStatus = $cmdResult.ConnectionStatus
            ConnectionErrorMessage = $cmdResult.ConnectionErrorMessage
            TargetService = $cmdResult.TargetService
            AccessLevel = $cmdResult.AccessLevel
            ExpiresAt = $cmdResult.ExpiresAt
        }

        return $enableResult
    }
    catch
    {
        $exception = $_
        Trace-Execution "Enable-AzStackHciRemoteSupport failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)"
        throw $exception
    }
}

function Get-AzStackHCIRemoteSupportAccess
{
    Param(
        [Parameter(Mandatory=$false)]
        [switch]
        $Cluster,

        [Parameter(Mandatory=$false)]
        [switch]
        $IncludeExpired
    )

    try
    {
        $OperationType = $MyInvocation.MyCommand

        # Install Az.StackHCI module if doesn't exist
        Invoke-InstallModule -ModuleName "Az.StackHCI"

        # Get Remote Support access
        $cmdResult = Az.StackHCI\Get-AzStackHCIRemoteSupportAccess -Cluster:$Cluster -IncludeExpired:$IncludeExpired

        # Build Result
        $now = [datetime]::UtcNow
        $targetComputerName = $ENV:COMPUTERNAME
        $cmdResultEmpty = [string]::IsNullOrWhiteSpace($cmdResult)

        if($cmdResultEmpty)
        {
            $aggregateStatus = 'Succeeded'
        }
        else{
            $aggregateStatus = if( -not $cmdResultEmpty -and $cmdResult.State -eq "Active" -and $cmdResult.ConnectionStatus -eq "Connected" -or "Connecting") {'Succeeded'} else {'Failed'}
        }
        
        $getAccessResult = New-Object -Type GetRemoteSupportAccessTarget -Property @{
            Name = 'AzStackHci_Get_RemoteSupportAccess'
            Title = 'Standalone Remote Support: Get Access'
            Severity = 'Info'
            Description = 'Get AzStackHCI Remote Support Access'
            Tags = $OperationType
            TargetResourceID = $targetComputerName
            TargetResourceName = $targetComputerName
            Timestamp = $now
            Status = $aggregateStatus
            HealthCheckSource = ((Get-PSCallStack)[-1].Command)
            AdditionalData = if(-not $cmdResultEmpty) {$cmdResult} else {''}
            State = if(-not $cmdResultEmpty) {$cmdResult.State} else {''}
            CreatedAt = if(-not $cmdResultEmpty) {$cmdResult.CreatedAt} else {''}
            UpdatedAt = if(-not $cmdResultEmpty) {$cmdResult.UpdatedAt} else {''}
            ConnectionStatus = if(-not $cmdResultEmpty) {$cmdResult.ConnectionStatus} else {''}
            ConnectionErrorMessage = if(-not $cmdResultEmpty) {$cmdResult.ConnectionErrorMessage} else{''}
            TargetService = if(-not $cmdResultEmpty) {$cmdResult.TargetService} else {''}
            AccessLevel = if(-not $cmdResultEmpty) {$cmdResult.AccessLevel} else {''}
            ExpiresAt = if(-not $cmdResultEmpty) {$cmdResult.ExpiresAt} else {''}
        }

        return $getAccessResult
    }
    catch
    {
        $exception = $_
        Trace-Execution "Get-AzStackHCIRemoteSupportAccess failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)"
        throw $exception
    }
}

function Get-AzStackHCIRemoteSupportSessionHistory
{
    Param(
        [Parameter(Mandatory=$false)]
        [string]
        $SessionId,

        [Parameter(Mandatory=$false)]
        [switch]
        $IncludeSessionTranscript,

        [Parameter(Mandatory=$false)]
        [DateTime]
        $FromDate = (Get-Date).AddDays(-7)
    )

    try
    {
        $OperationType = $MyInvocation.MyCommand

        # Install Az.StackHCI module if doesn't exist
        Invoke-InstallModule -ModuleName "Az.StackHCI"

        # Get Remote Support session history
        $cmdResult = Az.StackHCI\Get-AzStackHCIRemoteSupportSessionHistory -SessionId $SessionId -FromDate $FromDate -IncludeSessionTranscript:$IncludeSessionTranscript

        # Build Result
        $now = [datetime]::UtcNow
        $targetComputerName = $ENV:COMPUTERNAME
        $cmdResultEmpty = [string]::IsNullOrWhiteSpace($cmdResult)

        if($cmdResultEmpty)
        {
            $aggregateStatus = 'Succeeded'
        }
        else{
            $aggregateStatus = if( -not $cmdResultEmpty -and $cmdResult.State -eq "Active" -and $cmdResult.ConnectionStatus -eq "Connected" -or "Connecting") {'Succeeded'} else {'Failed'}
        }
        
        $getSessionHistoryResult = New-Object -Type PsObject -Property @{
            Name = 'AzStackHci_Get_RemoteSupportSessionHistory'
            Title = 'Standalone Remote Support: Get Session History'
            Severity = 'Info'
            Description = 'Get AzStackHCI Remote Support session history'
            Tags = $OperationType
            TargetResourceID = $targetComputerName
            TargetResourceName = $targetComputerName
            Timestamp = $now
            Status = $aggregateStatus
            HealthCheckSource = ((Get-PSCallStack)[-1].Command)
            AdditionalData = $cmdResult
        }

        return $getSessionHistoryResult
    }
    catch
    {
        $exception = $_
        Trace-Execution "Get-AzStackHCIRemoteSupportSessionHistory failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)"
        throw $exception
    }
}

function Disable-AzStackHciRemoteSupport
{
    param ()

    try
    {
        $OperationType = $MyInvocation.MyCommand

        # Install Az.StackHCI module if doesn't exist
        Invoke-InstallModule -ModuleName "Az.StackHCI"

        # Disable Remote Support
        Az.StackHCI\Disable-AzStackHCIRemoteSupport

        $now = [datetime]::UtcNow
        $targetComputerName = $ENV:COMPUTERNAME

        $aggregateStatus = "Succeeded"
        return New-Object -Type PsObject -Property @{
            Name = 'AzStackHci_Disable_RemoteSupport'
            Title = 'Standalone Remote Support: Disable'
            Severity = 'Info'
            Description = 'Disable AzStackHCI Remote Support'
            Tags = $OperationType
            TargetResourceID = $targetComputerName
            TargetResourceName = $targetComputerName
            Timestamp = $now
            Status = $aggregateStatus
            HealthCheckSource = ((Get-PSCallStack)[-1].Command)
            AdditionalData = {}
            Remediation = ""
        }
    }
    catch
    {
        $exception = $_
        Trace-Execution "Disable-AzStackHciRemoteSupport failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)"
        throw $exception
    }
}

function Remove-AzStackHciRemoteSupport
{
    param ()

    try
    {
        $OperationType = $MyInvocation.MyCommand

        # Install Az.StackHCI module if doesn't exist
        Invoke-InstallModule -ModuleName "Az.StackHCI"
        
        # Install Remote Support
        Az.StackHCI\Remove-AzStackHCIRemoteSupport

        $now = [datetime]::UtcNow
        $targetComputerName = $ENV:COMPUTERNAME

        $aggregateStatus = "Succeeded"
        return New-Object -Type PsObject -Property @{
            Name = 'AzStackHci_Remove_RemoteSupport'
            Title = 'Standalone Remote Support: Remove'
            Severity = 'Info'
            Description = 'Remove AzStackHCI Remote Support'
            Tags = $OperationType
            TargetResourceID = $targetComputerName
            TargetResourceName = $targetComputerName
            Timestamp = $now
            Status = $aggregateStatus
            HealthCheckSource = ((Get-PSCallStack)[-1].Command)
            AdditionalData = {}
            Remediation = ""
        }
    }
    catch
    {
        $exception = $_
        Trace-Execution "Remove-AzStackHciRemoteSupport failed. $exception"
        Trace-Execution "$($exception.ScriptStackTrace)"
        throw $exception
    }
}

Export-ModuleMember -Function Install-AzStackHciRemoteSupport -Variable MetaData
Export-ModuleMember -Function Enable-AzStackHciRemoteSupport -Variable MetaData
Export-ModuleMember -Function Disable-AzStackHciRemoteSupport -Variable MetaData
Export-ModuleMember -Function Get-AzStackHCIRemoteSupportAccess -Variable MetaData
Export-ModuleMember -Function Get-AzStackHCIRemoteSupportSessionHistory -Variable MetaData
Export-ModuleMember -Function Remove-AzStackHciRemoteSupport -Variable MetaData
# SIG # Begin signature block
# MIIoLQYJKoZIhvcNAQcCoIIoHjCCKBoCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCByBxL/+PBPqlDu
# oZsYsInxmbNgy2GrQMe3VCE+YQurt6CCDXYwggX0MIID3KADAgECAhMzAAADTrU8
# esGEb+srAAAAAANOMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p
# bmcgUENBIDIwMTEwHhcNMjMwMzE2MTg0MzI5WhcNMjQwMzE0MTg0MzI5WjB0MQsw
# CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u
# ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB
# AQDdCKiNI6IBFWuvJUmf6WdOJqZmIwYs5G7AJD5UbcL6tsC+EBPDbr36pFGo1bsU
# p53nRyFYnncoMg8FK0d8jLlw0lgexDDr7gicf2zOBFWqfv/nSLwzJFNP5W03DF/1
# 1oZ12rSFqGlm+O46cRjTDFBpMRCZZGddZlRBjivby0eI1VgTD1TvAdfBYQe82fhm
# WQkYR/lWmAK+vW/1+bO7jHaxXTNCxLIBW07F8PBjUcwFxxyfbe2mHB4h1L4U0Ofa
# +HX/aREQ7SqYZz59sXM2ySOfvYyIjnqSO80NGBaz5DvzIG88J0+BNhOu2jl6Dfcq
# jYQs1H/PMSQIK6E7lXDXSpXzAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE
# AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUnMc7Zn/ukKBsBiWkwdNfsN5pdwAw
# RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW
# MBQGA1UEBRMNMjMwMDEyKzUwMDUxNjAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci
# tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j
# b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG
# CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu
# Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0
# MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBAD21v9pHoLdBSNlFAjmk
# mx4XxOZAPsVxxXbDyQv1+kGDe9XpgBnT1lXnx7JDpFMKBwAyIwdInmvhK9pGBa31
# TyeL3p7R2s0L8SABPPRJHAEk4NHpBXxHjm4TKjezAbSqqbgsy10Y7KApy+9UrKa2
# kGmsuASsk95PVm5vem7OmTs42vm0BJUU+JPQLg8Y/sdj3TtSfLYYZAaJwTAIgi7d
# hzn5hatLo7Dhz+4T+MrFd+6LUa2U3zr97QwzDthx+RP9/RZnur4inzSQsG5DCVIM
# pA1l2NWEA3KAca0tI2l6hQNYsaKL1kefdfHCrPxEry8onJjyGGv9YKoLv6AOO7Oh
# JEmbQlz/xksYG2N/JSOJ+QqYpGTEuYFYVWain7He6jgb41JbpOGKDdE/b+V2q/gX
# UgFe2gdwTpCDsvh8SMRoq1/BNXcr7iTAU38Vgr83iVtPYmFhZOVM0ULp/kKTVoir
# IpP2KCxT4OekOctt8grYnhJ16QMjmMv5o53hjNFXOxigkQWYzUO+6w50g0FAeFa8
# 5ugCCB6lXEk21FFB1FdIHpjSQf+LP/W2OV/HfhC3uTPgKbRtXo83TZYEudooyZ/A
# Vu08sibZ3MkGOJORLERNwKm2G7oqdOv4Qj8Z0JrGgMzj46NFKAxkLSpE5oHQYP1H
# tPx1lPfD7iNSbJsP6LiUHXH1MIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq
# hkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x
# EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv
# bjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5
# IDIwMTEwHhcNMTEwNzA4MjA1OTA5WhcNMjYwNzA4MjEwOTA5WjB+MQswCQYDVQQG
# EwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwG
# A1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYDVQQDEx9NaWNyb3NvZnQg
# Q29kZSBTaWduaW5nIFBDQSAyMDExMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIIC
# CgKCAgEAq/D6chAcLq3YbqqCEE00uvK2WCGfQhsqa+laUKq4BjgaBEm6f8MMHt03
# a8YS2AvwOMKZBrDIOdUBFDFC04kNeWSHfpRgJGyvnkmc6Whe0t+bU7IKLMOv2akr
# rnoJr9eWWcpgGgXpZnboMlImEi/nqwhQz7NEt13YxC4Ddato88tt8zpcoRb0Rrrg
# OGSsbmQ1eKagYw8t00CT+OPeBw3VXHmlSSnnDb6gE3e+lD3v++MrWhAfTVYoonpy
# 4BI6t0le2O3tQ5GD2Xuye4Yb2T6xjF3oiU+EGvKhL1nkkDstrjNYxbc+/jLTswM9
# sbKvkjh+0p2ALPVOVpEhNSXDOW5kf1O6nA+tGSOEy/S6A4aN91/w0FK/jJSHvMAh
# dCVfGCi2zCcoOCWYOUo2z3yxkq4cI6epZuxhH2rhKEmdX4jiJV3TIUs+UsS1Vz8k
# A/DRelsv1SPjcF0PUUZ3s/gA4bysAoJf28AVs70b1FVL5zmhD+kjSbwYuER8ReTB
# w3J64HLnJN+/RpnF78IcV9uDjexNSTCnq47f7Fufr/zdsGbiwZeBe+3W7UvnSSmn
# Eyimp31ngOaKYnhfsi+E11ecXL93KCjx7W3DKI8sj0A3T8HhhUSJxAlMxdSlQy90
# lfdu+HggWCwTXWCVmj5PM4TasIgX3p5O9JawvEagbJjS4NaIjAsCAwEAAaOCAe0w
# ggHpMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRIbmTlUAXTgqoXNzcitW2o
# ynUClTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMCAYYwDwYD
# VR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBRyLToCMZBDuRQFTuHqp8cx0SOJNDBa
# BgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20vcGtpL2Ny
# bC9wcm9kdWN0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3JsMF4GCCsG
# AQUFBwEBBFIwUDBOBggrBgEFBQcwAoZCaHR0cDovL3d3dy5taWNyb3NvZnQuY29t
# L3BraS9jZXJ0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3J0MIGfBgNV
# HSAEgZcwgZQwgZEGCSsGAQQBgjcuAzCBgzA/BggrBgEFBQcCARYzaHR0cDovL3d3
# dy5taWNyb3NvZnQuY29tL3BraW9wcy9kb2NzL3ByaW1hcnljcHMuaHRtMEAGCCsG
# AQUFBwICMDQeMiAdAEwAZQBnAGEAbABfAHAAbwBsAGkAYwB5AF8AcwB0AGEAdABl
# AG0AZQBuAHQALiAdMA0GCSqGSIb3DQEBCwUAA4ICAQBn8oalmOBUeRou09h0ZyKb
# C5YR4WOSmUKWfdJ5DJDBZV8uLD74w3LRbYP+vj/oCso7v0epo/Np22O/IjWll11l
# hJB9i0ZQVdgMknzSGksc8zxCi1LQsP1r4z4HLimb5j0bpdS1HXeUOeLpZMlEPXh6
# I/MTfaaQdION9MsmAkYqwooQu6SpBQyb7Wj6aC6VoCo/KmtYSWMfCWluWpiW5IP0
# wI/zRive/DvQvTXvbiWu5a8n7dDd8w6vmSiXmE0OPQvyCInWH8MyGOLwxS3OW560
# STkKxgrCxq2u5bLZ2xWIUUVYODJxJxp/sfQn+N4sOiBpmLJZiWhub6e3dMNABQam
# ASooPoI/E01mC8CzTfXhj38cbxV9Rad25UAqZaPDXVJihsMdYzaXht/a8/jyFqGa
# J+HNpZfQ7l1jQeNbB5yHPgZ3BtEGsXUfFL5hYbXw3MYbBL7fQccOKO7eZS/sl/ah
# XJbYANahRr1Z85elCUtIEJmAH9AAKcWxm6U/RXceNcbSoqKfenoi+kiVH6v7RyOA
# 9Z74v2u3S5fi63V4GuzqN5l5GEv/1rMjaHXmr/r8i+sLgOppO6/8MO0ETI7f33Vt
# Y5E90Z1WTk+/gFcioXgRMiF670EKsT/7qMykXcGhiJtXcVZOSEXAQsmbdlsKgEhr
# /Xmfwb1tbWrJUnMTDXpQzTGCGg0wghoJAgEBMIGVMH4xCzAJBgNVBAYTAlVTMRMw
# EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN
# aWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNp
# Z25pbmcgUENBIDIwMTECEzMAAANOtTx6wYRv6ysAAAAAA04wDQYJYIZIAWUDBAIB
# BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO
# MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIPjDxYn8PI1OFk9gjXKUhgWz
# QhWTJNOXgXu6hDHUi9glMEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A
# cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB
# BQAEggEAI+iWUqEJjwzz5tQb+bDKysa5JnZJ8sUbZg3dxhaF2vr7vKMI6NpbTKA4
# 3SeafEjmo+sACRzhZxCv5xz2CG7m8tKDN8McsXdXOdeBLBCTk/AeJMDa+9tYksea
# CF2Mf4H73uGfeXSrU0RSmikYDkuykw6y29UwMpCTk+RixodlDpDo47C8zAxfVfZj
# DNGRQTtBS692hkzN9i4QXcvrPDdAM5oPD2/vkeRw/EP2uhc2qspHwwbSXP/kqIBA
# 0IL3klvu3n1Ja0OZLhyHB718s3GfIRaafCSGakgehF92HCP3tgx2vBWSWdL+0r5V
# z8M1ghIKFgBXq5F+h+5qolxtK6PttqGCF5cwgheTBgorBgEEAYI3AwMBMYIXgzCC
# F38GCSqGSIb3DQEHAqCCF3AwghdsAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFSBgsq
# hkiG9w0BCRABBKCCAUEEggE9MIIBOQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl
# AwQCAQUABCD3GvjR57oYlsPRA8rIoJaEz6m/GwwI0ZCOfEqZ7yooQwIGZQQ0KAnA
# GBMyMDIzMDkyMjA4MzExOS4wMzJaMASAAgH0oIHRpIHOMIHLMQswCQYDVQQGEwJV
# UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE
# ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1l
# cmljYSBPcGVyYXRpb25zMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046RjAwMi0w
# NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Wg
# ghHtMIIHIDCCBQigAwIBAgITMwAAAc4PGPdFl+fG/wABAAABzjANBgkqhkiG9w0B
# AQsFADB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE
# BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD
# VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAeFw0yMzA1MjUxOTEy
# MDhaFw0yNDAyMDExOTEyMDhaMIHLMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2Fz
# aGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENv
# cnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1lcmljYSBPcGVyYXRpb25z
# MScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046RjAwMi0wNUUwLUQ5NDcxJTAjBgNV
# BAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0GCSqGSIb3DQEB
# AQUAA4ICDwAwggIKAoICAQC5CkwZ1yjYx3fnKTw/VnzwGGhKOIjqMDSuHdGg8JoJ
# 2LN2nBUUkAwxhYAR4ZQWg9QbjxZ/DWrD2xeUwLnKOKNDNthX9vaKj+X5Ctxi6ioT
# VU7UB5oQ4wGpkV2kmfnp0RYGdhtc58AaoUZFcvhdBlJ2yETwuCuEV6pk4J7ghGym
# szr0HVqR9B2MJjV8rePL+HGIzIbYLrk0jWmaKRRPsFfxKKw3njFgFlSqaBA4SVuV
# 0FYE/4t0Z9UjXUPLqw+iDeLUv3sp3h9M4oNIZ216VPlVlf3FOFRLlZg8eCeX4xla
# BjWia95nXlXMXQWqaIwkgN4TsRzymgeWuVzMpRPBWk6gOjzxwXnjIcWqx1lPznIS
# v/xtn1HpB+CIF5SPKkCf8lCPiZ1EtB01FzHRj+YhRWJjsRl1gLW1i0ELrrWVAFrD
# PrIshBKoz6SUAyKD7yPx649SyLGBo/vJHxZgMlYirckf9eklprNDeoslhreIYzAJ
# rMJ+YoWn9Dxmg/7hGC/XH8eljmJqBLqyHCmdgS+WArj84ciRGsmqRaUB/4hFGUkL
# v1Ga2vEPtVByUmjHcAppJR1POmi1ATV9FusOQQxkD2nXWSKWfKApD7tGfNZMRvku
# fHFwGf5NnN0Aim0ljBg1O5gs43Fok/uSe12zQL0hSP9Jf+iCL+NPTPAPJPEsbdYa
# vQIDAQABo4IBSTCCAUUwHQYDVR0OBBYEFDD7CEZAo5MMjpl+FWTsUyn54oXFMB8G
# A1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRYMFYwVKBSoFCG
# Tmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY3Jvc29mdCUy
# MFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEFBQcBAQRgMF4w
# XAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2Vy
# dHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3J0MAwG
# A1UdEwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwgwDgYDVR0PAQH/BAQD
# AgeAMA0GCSqGSIb3DQEBCwUAA4ICAQCXIBYW/0UVTDDZO/fQ2XstNC4DZG8RPbrl
# ZHyFt57z/VWqPut6rugayGW1UcvJuxf8REtiTtmf5SQ5N2pu0nTl6O4BtScIvM/K
# 8pe/yj77x8u6vfk8Q6SDOZoFpIpVkFH3y67isf4/SfoN9M2nLb93po/OtlM9AcWT
# JbqunzC+kmeLcxJmCxLcsiBMJ6ZTvSNWQnicgMuv7PF0ip9HYjzFWoNq8qnrs7g+
# +YGPXU7epl1KSBTr9UR7Hn/kNcqCiZf22DhoZPVP7+vZHTY+OXoxoEEOnzAbAlBC
# up/wbXNJissiK8ZyRJXT/R4FVmE22CSvpu+p5MeRlBT42pkIhhMlqXlsdQdT9cWI
# tiW8yWRpaE1ZI1my9FW8JM9DtCQti3ZuGHSNpvm4QAY/61ryrKol4RLf5F+SAl4o
# zVvM8PKMeRdEmo2wOzZK4ME7D7iHzLcYp5ucw0kgsy396faczsXdnLSomXMArstG
# kHvt/F3hq2eESQ2PgrX+gpdBo8uPV16ywmnpAwYqMdZM+yH6B//4MsXEu3Rg5QOo
# OWdjNVB7Qm6MPJg+vDX59XvMmibAzbplxIyp7S1ky7L+g3hq6KxlKQ9abUjYpaOF
# nHtKDFJ+vxzncEMVEV3IHQdjC7urqOBgO7vypeIwjQ689qu2NNuIQ6cZZgMn8EvS
# SWRwDG8giTCCB3EwggVZoAMCAQICEzMAAAAVxedrngKbSZkAAAAAABUwDQYJKoZI
# hvcNAQELBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAw
# DgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x
# MjAwBgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmljYXRlIEF1dGhvcml0eSAy
# MDEwMB4XDTIxMDkzMDE4MjIyNVoXDTMwMDkzMDE4MzIyNVowfDELMAkGA1UEBhMC
# VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNV
# BAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRp
# bWUtU3RhbXAgUENBIDIwMTAwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoIC
# AQDk4aZM57RyIQt5osvXJHm9DtWC0/3unAcH0qlsTnXIyjVX9gF/bErg4r25Phdg
# M/9cT8dm95VTcVrifkpa/rg2Z4VGIwy1jRPPdzLAEBjoYH1qUoNEt6aORmsHFPPF
# dvWGUNzBRMhxXFExN6AKOG6N7dcP2CZTfDlhAnrEqv1yaa8dq6z2Nr41JmTamDu6
# GnszrYBbfowQHJ1S/rboYiXcag/PXfT+jlPP1uyFVk3v3byNpOORj7I5LFGc6XBp
# Dco2LXCOMcg1KL3jtIckw+DJj361VI/c+gVVmG1oO5pGve2krnopN6zL64NF50Zu
# yjLVwIYwXE8s4mKyzbnijYjklqwBSru+cakXW2dg3viSkR4dPf0gz3N9QZpGdc3E
# XzTdEonW/aUgfX782Z5F37ZyL9t9X4C626p+Nuw2TPYrbqgSUei/BQOj0XOmTTd0
# lBw0gg/wEPK3Rxjtp+iZfD9M269ewvPV2HM9Q07BMzlMjgK8QmguEOqEUUbi0b1q
# GFphAXPKZ6Je1yh2AuIzGHLXpyDwwvoSCtdjbwzJNmSLW6CmgyFdXzB0kZSU2LlQ
# +QuJYfM2BjUYhEfb3BvR/bLUHMVr9lxSUV0S2yW6r1AFemzFER1y7435UsSFF5PA
# PBXbGjfHCBUYP3irRbb1Hode2o+eFnJpxq57t7c+auIurQIDAQABo4IB3TCCAdkw
# EgYJKwYBBAGCNxUBBAUCAwEAATAjBgkrBgEEAYI3FQIEFgQUKqdS/mTEmr6CkTxG
# NSnPEP8vBO4wHQYDVR0OBBYEFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMFwGA1UdIARV
# MFMwUQYMKwYBBAGCN0yDfQEBMEEwPwYIKwYBBQUHAgEWM2h0dHA6Ly93d3cubWlj
# cm9zb2Z0LmNvbS9wa2lvcHMvRG9jcy9SZXBvc2l0b3J5Lmh0bTATBgNVHSUEDDAK
# BggrBgEFBQcDCDAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMC
# AYYwDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBTV9lbLj+iiXGJo0T2UkFvX
# zpoYxDBWBgNVHR8ETzBNMEugSaBHhkVodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20v
# cGtpL2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcmwwWgYI
# KwYBBQUHAQEETjBMMEoGCCsGAQUFBzAChj5odHRwOi8vd3d3Lm1pY3Jvc29mdC5j
# b20vcGtpL2NlcnRzL01pY1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNydDANBgkqhkiG
# 9w0BAQsFAAOCAgEAnVV9/Cqt4SwfZwExJFvhnnJL/Klv6lwUtj5OR2R4sQaTlz0x
# M7U518JxNj/aZGx80HU5bbsPMeTCj/ts0aGUGCLu6WZnOlNN3Zi6th542DYunKmC
# VgADsAW+iehp4LoJ7nvfam++Kctu2D9IdQHZGN5tggz1bSNU5HhTdSRXud2f8449
# xvNo32X2pFaq95W2KFUn0CS9QKC/GbYSEhFdPSfgQJY4rPf5KYnDvBewVIVCs/wM
# nosZiefwC2qBwoEZQhlSdYo2wh3DYXMuLGt7bj8sCXgU6ZGyqVvfSaN0DLzskYDS
# PeZKPmY7T7uG+jIa2Zb0j/aRAfbOxnT99kxybxCrdTDFNLB62FD+CljdQDzHVG2d
# Y3RILLFORy3BFARxv2T5JL5zbcqOCb2zAVdJVGTZc9d/HltEAY5aGZFrDZ+kKNxn
# GSgkujhLmm77IVRrakURR6nxt67I6IleT53S0Ex2tVdUCbFpAUR+fKFhbHP+Crvs
# QWY9af3LwUFJfn6Tvsv4O+S3Fb+0zj6lMVGEvL8CwYKiexcdFYmNcP7ntdAoGokL
# jzbaukz5m/8K6TT4JDVnK+ANuOaMmdbhIurwJ0I9JZTmdHRbatGePu1+oDEzfbzL
# 6Xu/OHBE0ZDxyKs6ijoIYn/ZcGNTTY3ugm2lBRDBcQZqELQdVTNYs6FwZvKhggNQ
# MIICOAIBATCB+aGB0aSBzjCByzELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hp
# bmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw
# b3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2EgT3BlcmF0aW9uczEn
# MCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOkYwMDItMDVFMC1EOTQ3MSUwIwYDVQQD
# ExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNloiMKAQEwBwYFKw4DAhoDFQBd
# jZUbFNAyCkVE6DdVWyizTYQHzKCBgzCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1w
# IFBDQSAyMDEwMA0GCSqGSIb3DQEBCwUAAgUA6LdEJzAiGA8yMDIzMDkyMTIyMzcy
# N1oYDzIwMjMwOTIyMjIzNzI3WjB3MD0GCisGAQQBhFkKBAExLzAtMAoCBQDot0Qn
# AgEAMAoCAQACAhi7AgH/MAcCAQACAhOqMAoCBQDouJWnAgEAMDYGCisGAQQBhFkK
# BAIxKDAmMAwGCisGAQQBhFkKAwKgCjAIAgEAAgMHoSChCjAIAgEAAgMBhqAwDQYJ
# KoZIhvcNAQELBQADggEBAFUHpL4W3HDAH59oDkKXCjjD0hvs2b8Nd34ak9ECeWat
# WexSeAZhWEXJO/+u50xna8dNAPVCLrIIu8R7C6Flui+wKuUEUSXHEut6i+4Zs/X5
# IJiqx/irjF5NWf4YA5FVBXogLwKveWlABQ+3LlJHEvi9pJCvZP7+FvUr2BB8oRPg
# thigACANdEjumlKrUvy2/y6C+CPmV2vWk03F/6mcVq5cGK6Z7cbjTuESvoHg2WUx
# eYGsqaowuzg0dN9HrBUYFzIAph2q0lWTEFVzFCeJEFI5t91MDFYZEYNuVLx7K+Tl
# gvwWTCl+SV1HHac2Xu0aBHD6dRqZdBtcHTsJrpVlJLcxggQNMIIECQIBATCBkzB8
# MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVk
# bW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1N
# aWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMAITMwAAAc4PGPdFl+fG/wABAAAB
# zjANBglghkgBZQMEAgEFAKCCAUowGgYJKoZIhvcNAQkDMQ0GCyqGSIb3DQEJEAEE
# MC8GCSqGSIb3DQEJBDEiBCBhQrKGQZ7mG28llZET0DtQ591H7p2pDZzvzZUewloS
# 5DCB+gYLKoZIhvcNAQkQAi8xgeowgecwgeQwgb0EIDJsz0F6L0XDUm53JRBfNMZs
# zKsllLDMiaFZ3PL/LqxnMIGYMIGApH4wfDELMAkGA1UEBhMCVVMxEzARBgNVBAgT
# Cldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m
# dCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENB
# IDIwMTACEzMAAAHODxj3RZfnxv8AAQAAAc4wIgQgE5MyjDVb4H80JW92+s3pyenr
# ggtY7aompQk0stRtQUEwDQYJKoZIhvcNAQELBQAEggIAVrMHrxxKeeZEXqM7WXf7
# RVrG9MM7yLE9KzpEs8wNrNxAtLsw/FlAV6BTB6LInlfY9FMJ5+P5PgjPke1sAKam
# qscPQvyxhNfx+gInV/mtZfvz2CkQ9H47R3nVQxaDWdEni53LDft88skq+LXFrkFq
# w4Q+lmvihbq5oke3eydVwFTqyy4Yg937XJXQaBMegcrX5Z5xXNd9PR+l2apCugB6
# vAKXTH/dnJcZGZRr+FKTpGEAHPXtGc/9kvqLaMyCEEEpI+cklI7WpUEfVHC6ES0l
# v1HwVx8fvAekI2jyYt/PZCxpk2WH/mSIB9ZhyglhanbtNkkYWOold9dzimtczloG
# gBL5gwak1aEyfIO94K6AkVnugYVIw76VnRNq+boRorF8YXtJGzRTpVH1+3rj6B3+
# X1wcP2G62WLsyZqtWsGUa8W5sXw2K2pKza8WEteffBHrclRLBwZ8Q0zX06dML8lG
# 6nHzZjriaeVfZ2ttLu4zUZ9ACbYEU12LQWUtFZIs2ENs2Kb3/ECpIiuzVoXysOg7
# /FTzRqh1KzvRDJHJNUYaCwxk8qoiB3t8dlAZnqMgViQopxSsKxdVS8KPE7+iWJCb
# Z7N/GgAOcwFFTbLPsDhvCQqbW3L5qm4tej01K/uFAQ0Ipv2U8+PW6JgWwu8y3eHS
# MFjC2w2wuQGQT0ZUeyBseD4=
# SIG # End signature block