Policy.Autorest/custom/Helpers.ps1
|
# ---------------------------------------------------------------------------------- # Copyright (c) Microsoft Corporation. All rights reserved. # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # http://www.apache.org/licenses/LICENSE-2.0 # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # Code generated by Microsoft (R) AutoRest Code Generator.Changes may cause incorrect behavior and will be lost if the code # is regenerated. # ---------------------------------------------------------------------------------- using namespace System.Management.Automation.Language # split policy ids into usable parts function ParsePolicyId { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] # the resource Id of a policy definition param($resourceId, $policyType) # validate args if (!$resourceId) { throw 'ParsePolicyId(resourceId, policyType) argument error: resourceId must be provided.' } if (!$policyType) { # extract policyType $temp = $resourceId -split '/providers/Microsoft.Authorization/' if ($temp.Length -lt 2) { throw 'parsePolicy(resourceId, policyType) argument error: resourceId is not a Microsoft.Authorization resource type' } $policyType = ($temp[1] -split '/')[0] } if (!$policyType) { throw 'ParsePolicyId(resourceId, policyType) argument error: unable to find type name.' } $mark = "/providers/Microsoft.Authorization/$($policyType)/" $parts = $resourceId -split $mark $scope = $parts[0] $name = '' if ($parts.Length -gt 1) { $parts = $parts[1] -split '/' $name = $parts[0] if (($parts.Length -gt 2) -and ($parts[1] -eq 'versions')) { $parsedVersion = ParsePolicyVersion $parts[2] } } $scopeType = 'none' $subId = '' $mgName = '' $rgName = '' $resource = '' $resNamespace = '' $resType = '' $resName = '' if (!$scope) { $scopeType = 'builtin' } elseif ($scope -like '/providers/Microsoft.Management/managementGroups/*') { $scopeType = 'mgname' $mgName = ($scope -split '/providers/Microsoft.Management/managementGroups/')[1] } elseif ($scope -like '/subscriptions/*/resourceGroups/*/*') { $scopeType = 'resource' $temp = ($scope -split '/subscriptions/')[1] $temp = ($temp -split '/resourceGroups/') $subId = $temp[0] $temp = ($temp[1] -split '/providers/') $rgName = $temp[0] $temp = $temp[1] -split '/' if ($temp.Length -gt 2) { $resNamespace = $temp[0] $resType = $temp[1..($temp.Length-2)] -join '/' $resName = $temp[$temp.Length-1] } $resource = $scope } elseif ($scope -like '/subscriptions/*/resourceGroups/*') { $scopeType = 'rgname' $temp = ($scope -split '/subscriptions/')[1] $temp = ($temp -split '/resourceGroups/') $subId = $temp[0] $rgName = $temp[1] } elseif ($scope -like '/subscriptions/*') { $scopeType = 'subId' $subId = ($scope -split '/subscriptions/')[1] } $artifactRef = '' $artifact = $scope + $mark + $name if ($parsedVersion.VersionRef) { $artifactRef = "$artifact/versions/$($parsedVersion.VersionRef)" } return @{ PolicyType = $policyType Scope = $scope ScopeType = $scopeType SubscriptionId = $subId ManagementGroupName = $mgName ResourceGroupName = $rgName Resource = $resource ResourceNamespace = $resNamespace ResourceType = $resType ResourceName = $resName Name = $name Artifact = $artifact ArtifactRef = $artifactRef Version = $parsedVersion.Version Major = $parsedVersion.Major Minor = $parsedVersion.Minor Patch = $parsedVersion.Patch Suffix = $parsedVersion.Suffix VersionRef = $parsedVersion.VersionRef VersionMajorRef = $parsedVersion.VersionMajorRef VersionMinorRef = $parsedVersion.VersionMinorRef } } # parse policy version with format: (ddd|*).(ddd|*).(ddd|*)[-suffix] function ParsePolicyVersion { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] # the resource Id of a policy definition param($version) $parts = $version -split '\.' $major = $parts[0] $minor = '' if ($parts.Length -gt 1) { $minor = $parts[1] } $patch = '' $suffix = '' if ($parts.Length -gt 2) { $parts = $parts[2] -split '-' $patch = $parts[0] if ($parts.Length -gt 1) { $suffix = $parts[1] } } $versionMinorRef = '' $versionMajorRef = @($major,'*','*') -join '.' if ($minor -ne '*') { $versionMinorRef = @($major,$minor,'*') -join '.' } if ($suffix) { if ($versionMinorRef) { $versionMinorRef = $versionMinorRef + '-' + $suffix } $versionMajorRef = $versionMajorRef + '-' + $suffix } $versionRef = '' if ($versionMinorRef) { $versionRef = $versionMinorRef } else { $versionRef = $versionMajorRef } return @{ Version = $version Major = $major Minor = $minor Patch = $patch Suffix = $suffix VersionRef = $versionRef VersionMajorRef = $versionMajorRef VersionMinorRef = $versionMinorRef } } # split policy definition resourceId into its parts (used externally) function ParsePolicyDefinitionId { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] # the resource Id of a policy definition param($ResourceId) ParsePolicyId $ResourceId 'policyDefinitions' } # split policy set definition resourceId into its parts function ParsePolicySetDefinitionId { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] # the resource Id of a policy set definition param($ResourceId) ParsePolicyId $ResourceId 'policySetDefinitions' } # split policy assignment resourceId into its parts function ParsePolicyAssignmentId { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] # the resource Id of a policy set definition param($ResourceId) ParsePolicyId $ResourceId 'policyAssignments' } # split policy assignment resourceId into its parts function ParsePolicyExemptionId { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] # the resource Id of a policy set definition param($resourceId) ParsePolicyId $ResourceId 'policyExemptions' } # Wrapper for JSON -> PSObject conversion that works on both Core and Desktop editions function ConvertFrom-JsonSafe { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [Parameter(ValueFromPipeline)] $InputObject, [switch]$AsHashtable = $false ) if ($PSVersionTable.PSEdition -eq 'Core') { ConvertFrom-Json $InputObject -AsHashtable:$AsHashtable } elseif ($AsHashtable) { # ConvertFrom-Json on Windows Powershell doesn't support -AsHashtable parameter $converted = ConvertParameterInput ($InputObject | ConvertFrom-Json) if (($converted -is [array]) -and ($converted.Count -eq 1)) { return $converted[0] } else { return $converted } } else { ConvertFrom-Json $InputObject } } # convert the parameter object (could be either hashtable or PSCustomObject) to policy-formatted hashtable suitable for autorest serializers function ConvertParameterObject { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param ($InputObject) $returnValue = @{} if ($InputObject -is [hashtable]) { foreach ($key in $InputObject.Keys) { $returnValue[$key] = @{ value = (ConvertParameterInput -InputObject $InputObject[$key]) } } } else { foreach ($property in $InputObject.PSObject.Properties) { $returnValue[$property.Name] = @{ value = (ConvertParameterInput -InputObject $InputObject.PSObject.Properties[$property.Name]) } } } return $returnValue } # Convert input parameter value to hashtable type expected by the autorest serializers function ConvertParameterArray { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param ($InputObject) if ($InputObject -is [array]) { $collection = @( foreach ($object in $InputObject) { ConvertParameterArray $object } ) Write-Output -NoEnumerate $collection } elseif ($InputObject -is [hashtable]) { return $InputObject } elseif ($InputObject -is [PSObject]) { $hash = @{} foreach ($property in $InputObject.PSObject.Properties) { $hash[$property.Name] = (ConvertParameterArray $property.Value).PSObject.BaseObject } $hash } else { return $InputObject } } # convert various parameter input formats to policy-formatted hashtable suitable for autorest serializers function ConvertParameterInput { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param ($InputObject) # traverse collections to ensure nested values are all processed if ($InputObject -is [hashtable]) { $returnValue = @{} foreach ($key in $InputObject.Keys) { $returnValue[$key] = (ConvertParameterInput $InputObject[$key]) } return $returnValue } elseif ($InputObject -is [array]) { $returnValue = @() foreach ($object in $InputObject) { $returnValue += (ConvertParameterInput $object) } return ,$returnValue } elseif ($InputObject -is [PSObject]) { $returnValue = @{} foreach ($property in $InputObject.PSObject.Properties) { $returnValue[$property.Name] = (ConvertParameterInput $property.Value) } return $returnValue } else { return $InputObject } } # Convert output hashtable object output by autorest serializers to PSCustomObject format for legacy support function ConvertObjectToPSObject { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param($InputObject) if ($null -eq $InputObject) { return [PSCustomObject]$null } if ($InputObject -is [array]) { return ,@(foreach ($obj in $InputObject) { ConvertObjectToPSObject $obj }) } if (!$InputObject.ToJsonString) { return [PSCustomObject]$InputObject } $jsonString = $InputObject.ToJsonString() if ($jsonString -is [array]) { $jsonString = "[$([System.String]::Join(',', $jsonString))]" } ConvertFrom-JsonSafe $jsonString } # Recursive conversion function for common AST blocks from parsing function Convert-AstLiteral { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param([Ast] $Node) # ensure Node isn't null before checking type in switch if ($null -eq $Node) { return $null } switch ($Node) { # Numbers, $true/$false/$null { $_.GetType() -eq [ConstantExpressionAst] } { return $_.Value } # Strings like "text" or 'text' { $_.GetType() -eq [StringConstantExpressionAst] } { return $_.Value } { $_.GetType() -eq [ExpandableStringExpressionAst] } { return $_.Value } # This node type is essentially a wrapper node for an array { $_.GetType() -eq [ArrayExpressionAst] } { $arr = @() foreach ($e in $_.SubExpression.Statements) { $arr += Convert-AstLiteral $e } return $arr } # Arrays: @( ... ) { $_.GetType() -eq [ArrayLiteralAst] } { $arr = @() foreach ($e in $_.Elements) { $arr += Convert-AstLiteral $e } return $arr } # Nested hashtables { $_.GetType() -eq [HashtableAst] } { $h = @{} foreach ($kv in $_.KeyValuePairs) { $k = $kv.Item1.Value $h[$k] = Convert-AstLiteral $kv.Item2 } return $h } # Case where a literal is wrapped in a pipeline { $_.GetType() -eq [PipelineAst] } { if ($_.PipelineElements.Count -eq 1) { return Convert-AstLiteral $_.PipelineElements[0].Expression } else { throw "Pipeline contains multiple elements, expected a single literal expression when parsing PSCustomObject." } } # Handles variables such as $null, $true, $false { $_.GetType() -eq [VariableExpressionAst] } { if ($_.VariablePath.IsVariable) { return $_.VariablePath.UserPath } else { throw "Unsupported variable path for safe conversion: $($_.VariablePath.ToString()). Unable to parse PSCustomObject." } } # Handles the case where a literal gets tokenized as a MemberExpressionAst - returns the string representation of the token { $_.GetType() -eq [MemberExpressionAst] } { return $_.Extent.Text } default { # Anything else is not allowed throw "Unsupported AST node for safe conversion: $($_.GetType().Name). Value: $($_.ToString()). Unable to parse PSCustomObject." } } } # Safely converts a string representation of a hashtable or PSCustomObject into a hashtable function ConvertTo-HashtableSafely { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [Parameter(Mandatory)] [string] $InputObject ) # Adding quotes to unquoted strings, datetimes, etc. to avoid parsing errors $fixedInput = $InputObject -replace '(?m)(=\s*)([a-zA-Z_][\w\-\.:/]*|\d{1,2}:\d{2}:\d{2}|\d{4}-\d{2}-\d{2}(?:[T ]\d{2}:\d{2}:\d{2}Z?)?|\d{1,2}/\d{1,2}/\d{4}(?:\s+\d{1,2}:\d{2}:\d{2})?|\d{4}/\d{2}/\d{2}(?:\s+\d{1,2}:\d{2}:\d{2})?)(\s*[\r\n;}]|$)', '$1"$2"$3' $tokens = $null; $errors = $null $ast = [Parser]::ParseInput($fixedInput, [ref]$tokens, [ref]$errors) if ($errors -and $errors.Count -gt 0) { throw "Invalid PSCustomObject or hashtable literal: $($errors[0].Message)" } # Find the first expression in the script $expr = $ast.EndBlock.Statements | ForEach-Object { $_.PipelineElements | ForEach-Object { $_.Expression } } | Select-Object -First 1 if (-not ($expr -is [HashtableAst])) { throw "Top-level expression is not a hashtable. It is of type: $($expr.GetType())" } return Convert-AstLiteral $expr } function GetPSObjectProperty { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param ( [PSObject]$PropertyObject, [string]$PropertyPath ) $propertyNames = $PropertyPath.Split('.') $tmpObject = $PropertyObject foreach ($propertyName in $propertyNames) { $propertyInfo = $tmpObject.PSObject.Properties[$propertyName] if ($propertyInfo) { if ($propertyInfo.Value -is [PSObject]) { $tmpObject = [PSObject]$propertyInfo.Value continue } return $propertyInfo.Value } } return $tmpObject } # tests whether the given string is a Uri function Test-Uri { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param([string]$Value) $uri = '' [System.Uri]::TryCreate($Value, [System.UriKind]::Absolute, [ref]$uri) } # issues a GET to the given address and returns the contents function Get-UriContent { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param([string]$UriAddress) $response = Invoke-WebRequest $UriAddress -DisableKeepAlive -Method Get if ($response.StatusCode -eq 200) { return $response.Content } } # if the given string is a file path or URI, returns the contents of the file or web page # otherwise returns the original string function GetFileUriOrStringParameterValue { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param([string]$parameterValue) # Test-Path can throw on bad input, but we just want to move to the next check if it does try { if (Test-Path $parameterValue) { return Get-Content $parameterValue | Out-String } } catch { # if error, want to handle exactly same as path not being valid by continuing } if (Test-Uri $parameterValue) { return Get-UriContent $parameterValue } else { return $parameterValue } } function ResolvePolicyParameter { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [string]$ParameterName, [string]$ParameterValue, [bool]$Debug = $false ) $policy = GetFileUriOrStringParameterValue $ParameterValue if ($debug) { Write-Host -ForegroundColor Cyan "Parameter ${ParameterName}:" $policy } $policyParameter = ConvertFrom-JsonSafe -AsHashtable $policy if ($policyParameter.properties) { return $policyParameter.properties } else { return $policyParameter } } function ResolvePolicyMetadataParameter { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( $MetadataValue, [bool]$Debug = $false ) if ($MetadataValue -is [hashtable]) { return $MetadataValue } if ([System.String]::IsNullOrEmpty($MetadataValue)) { return $MetadataValue } # This function will usually be passed a string, but can have an issue if passed a PSCustomObject that isn't converted to string if ($MetadataValue -isnot [string]) { $MetadataValue = $MetadataValue | ConvertTo-Json -Depth 30 } $metadata = (GetFileUriOrStringParameterValue $MetadataValue).Trim() if ($debug) { Write-Host -ForegroundColor Cyan Metadata: $metadata } if ($metadata -like '@{*') { # probably a PSCustomObject, try converting to hashtable return ConvertTo-HashtableSafely $metadata } # otherwise it should be a JSON string if ($metadata -like '{*}') { return $metadata | ConvertFrom-JsonSafe -AsHashtable } throw "Unrecognized metadata format - value: [$($metadataValue)], type: [$($metadataValue.GetType())]" } # construct the full Id of a resource given the various parts (only used internally in this file) function resolvePolicyArtifact { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [string]$name, [string]$subscriptionId, [string]$managementGroupName, [string]$id, [string]$policyType ) $scope = '' $scopeType = 'none' $scopeName = '' $resourceId = '<invalid>' if ($id -and !$subscriptionId -and !$managementGroupName) { $resolved = ParsePolicyId $id $policyType $scope = $resolved.Scope $scopeType = $resolved.ScopeType switch ($scopeType) { 'subId' { $subscriptionId = $resolved.SubscriptionId $scopeName = "subscription $($subscriptionId)" } 'mgName' { $managementGroupName = $resolved.ManagementGroupName $scopeName = "management group $($managementGroupName)" } 'rgname' { $subscriptionId = $resolved.SubscriptionId $scopeName = "resource group $($resolved.ResourceGroupName) (subId: $($subscriptionId))" } 'resource' { $subscriptionId = $resolved.SubscriptionId $scopeName = "resource id $($resolved.Resource)" } 'none' { $scopeName = "scope $($scope)" } } $name = $resolved.Name $resourceId = $id } else { if ($name) { if ($managementGroupName) { $scopeType = 'mgName' $scopeName = "management group $($managementGroupName)" $scope = "/providers/Microsoft.Management/managementGroups/$($managementGroupName)" } else { if (!$subscriptionId) { $subscriptionId = (Get-SubscriptionId) } $scopeType = 'subId' $scopeName = "subscription $($subscriptionId)" $scope = "/subscriptions/$($subscriptionId)" } $resourceId = "$($scope)/providers/Microsoft.Authorization/$($policyType)/$($name)" } } return @{ Scope = $scope ScopeType = $scopeType ScopeName = $scopeName Name = $name SubscriptionId = $subscriptionId ManagementGroupName = $managementGroupName ResourceId = $resourceId ResourceGroupName = $resolved.ResourceGroupName ResourceNamespace = $resolved.ResourceNamespace ResourceType = $resolved.ResourceType ResourceName = $resolved.ResourceName Artifact = $resolved.Artifact Version = $resolved.Version } } function ResolvePolicyDefinition { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [string]$Name, [string]$SubscriptionId, [string]$ManagementGroupName, [string]$Id ) resolvePolicyArtifact $Name $SubscriptionId $ManagementGroupName $Id 'policyDefinitions' } function ResolvePolicySetDefinition { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [string]$Name, [string]$SubscriptionId, [string]$ManagementGroupName, [string]$Id ) resolvePolicyArtifact $Name $SubscriptionId $ManagementGroupName $Id 'policySetDefinitions' } function ResolvePolicyAssignment { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [string]$Name, [string]$Scope, [string]$Id ) if ($Id) { $resourceId = $Id } elseif ($Scope) { $resourceId = "$($Scope)/providers/Microsoft.Authorization/policyAssignments/$($Name)" } else { $resourceId = "/subscriptions/$($(Get-SubscriptionId))/providers/Microsoft.Authorization/policyAssignments/$($Name)" } resolvePolicyArtifact $null $null $null $resourceId 'policyAssignments' } function ResolvePolicyExemption { [Microsoft.Azure.PowerShell.Cmdlets.Policy.DoNotExportAttribute()] param( [string]$Name, [string]$Scope, [string]$Id ) if ($Id) { $resourceId = $Id } elseif ($Scope) { $resourceId = "$($Scope)/providers/Microsoft.Authorization/policyExemptions/$($Name)" } else { $resourceId = "/subscriptions/$($(Get-SubscriptionId))/providers/Microsoft.Authorization/policyExemptions/$($Name)" } resolvePolicyArtifact $null $null $null $resourceId 'policyExemptions' } function LocationCompleter( $commandName, $parameterName, $wordToComplete, $commandAst, $fakeBoundParameter ) { if ($global:AzPSPolicyCachedLocations.Count -le 0) { $response = Invoke-AzRestMethod -Uri "https://management.azure.com/subscriptions/$($(Get-SubscriptionId))/locations?api-version=2022-12-01" -Method GET $global:AzPSPolicyCachedLocations = ($response.Content | ConvertFrom-JsonSafe).value | Sort-Object -Property name | Select-Object -ExpandProperty name } # If you see the following error, it means your context access has expired # The given key 'AzureAttestationServiceEndpointSuffix' was not present in the dictionary. $global:AzPSPolicyCachedLocations | Where-Object { $_ -like "$wordToComplete*" } } function Get-SubscriptionId { $script = Resolve-Path "$PSScriptRoot/../utils/Get-SubscriptionIdTestSafe.ps1" return . $script } function Get-ExtraParameters ( $DefaultProfile, $Break, $HttpPipelineAppend, $HttpPipelinePrepend, $Proxy, $ProxyCredential, $ProxyUseDefaultCredentials ) { $parms = @{} if ($PSBoundParameters['DefaultProfile']) { $parms['DefaultProfile'] = $PSBoundParameters['DefaultProfile'] } if ($PSBoundParameters['Break']) { $parms['Break'] = $PSBoundParameters['Break'] } if ($PSBoundParameters['HttpPipelineAppend']) { $parms['HttpPipelineAppend'] = $PSBoundParameters['HttpPipelineAppend'] } if ($PSBoundParameters['HttpPipelinePrepend']) { $parms['HttpPipelinePrepend'] = $PSBoundParameters['HttpPipelinePrepend'] } if ($PSBoundParameters['Proxy']) { $parms['Proxy'] = $PSBoundParameters['Proxy'] } if ($PSBoundParameters['ProxyCredential']) { $parms['ProxyCredential'] = $PSBoundParameters['ProxyCredential'] } if ($PSBoundParameters['ProxyUseDefaultCredentials']) { $parms['ProxyUseDefaultCredentials'] = $PSBoundParameters['ProxyUseDefaultCredentials'] } return $parms } # register the location completer for New-AzPolicyAssignment Register-ArgumentCompleter -CommandName New-AzPolicyAssignment -ParameterName Location -ScriptBlock ${function:LocationCompleter} # cache Azure locations to be used by the location completer (Get-AzLocation is not available in this context, need to use REST) $global:AzPSPolicyCachedLocations = @() # SIG # Begin signature block # MIIoLQYJKoZIhvcNAQcCoIIoHjCCKBoCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCB97cRrvC0xIYWR # KS2GaOXaXAEjWZeDD28ZRO/nU9zM9KCCDXYwggX0MIID3KADAgECAhMzAAAEhV6Z # 7A5ZL83XAAAAAASFMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjUwNjE5MTgyMTM3WhcNMjYwNjE3MTgyMTM3WjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQDASkh1cpvuUqfbqxele7LCSHEamVNBfFE4uY1FkGsAdUF/vnjpE1dnAD9vMOqy # 5ZO49ILhP4jiP/P2Pn9ao+5TDtKmcQ+pZdzbG7t43yRXJC3nXvTGQroodPi9USQi # 9rI+0gwuXRKBII7L+k3kMkKLmFrsWUjzgXVCLYa6ZH7BCALAcJWZTwWPoiT4HpqQ # hJcYLB7pfetAVCeBEVZD8itKQ6QA5/LQR+9X6dlSj4Vxta4JnpxvgSrkjXCz+tlJ # 67ABZ551lw23RWU1uyfgCfEFhBfiyPR2WSjskPl9ap6qrf8fNQ1sGYun2p4JdXxe # UAKf1hVa/3TQXjvPTiRXCnJPAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUuCZyGiCuLYE0aU7j5TFqY05kko0w # RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW # MBQGA1UEBRMNMjMwMDEyKzUwNTM1OTAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci # tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG # CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu # Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0 # MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBACjmqAp2Ci4sTHZci+qk # tEAKsFk5HNVGKyWR2rFGXsd7cggZ04H5U4SV0fAL6fOE9dLvt4I7HBHLhpGdE5Uj # Ly4NxLTG2bDAkeAVmxmd2uKWVGKym1aarDxXfv3GCN4mRX+Pn4c+py3S/6Kkt5eS # DAIIsrzKw3Kh2SW1hCwXX/k1v4b+NH1Fjl+i/xPJspXCFuZB4aC5FLT5fgbRKqns # WeAdn8DsrYQhT3QXLt6Nv3/dMzv7G/Cdpbdcoul8FYl+t3dmXM+SIClC3l2ae0wO # lNrQ42yQEycuPU5OoqLT85jsZ7+4CaScfFINlO7l7Y7r/xauqHbSPQ1r3oIC+e71 # 5s2G3ClZa3y99aYx2lnXYe1srcrIx8NAXTViiypXVn9ZGmEkfNcfDiqGQwkml5z9 # nm3pWiBZ69adaBBbAFEjyJG4y0a76bel/4sDCVvaZzLM3TFbxVO9BQrjZRtbJZbk # C3XArpLqZSfx53SuYdddxPX8pvcqFuEu8wcUeD05t9xNbJ4TtdAECJlEi0vvBxlm # M5tzFXy2qZeqPMXHSQYqPgZ9jvScZ6NwznFD0+33kbzyhOSz/WuGbAu4cHZG8gKn # lQVT4uA2Diex9DMs2WHiokNknYlLoUeWXW1QrJLpqO82TLyKTbBM/oZHAdIc0kzo # STro9b3+vjn2809D0+SOOCVZMIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq # 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 # Z25pbmcgUENBIDIwMTECEzMAAASFXpnsDlkvzdcAAAAABIUwDQYJYIZIAWUDBAIB # BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO # MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIEhpObXBLjQ7QUagQMJxLzSL # z++xWqZ1eAdIPlEbxeBsMEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A # cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB # BQAEggEAGng1Hg/ZEmW8zjePYcYIqi1stHCkjBmaqPEHUxo9u25jYflgRVMHtzgK # +wDXNW2qZb5ON+Fwf59Y1D0YttMDk0xcLw7Dtjo7b61ZeaUHilKLHDDwNC7FDC65 # NDcBiFigNYer5B2TT4sySTCuywRn5Y7t1ug2y0WWSa/ZEiGz7pk+BU0lyWV/zyzg # f8Dj/A5h4Cw52sRPQrAkkZzdkrRHUBXO8vL4yENSz6HfYxHNyAfLTKivXs13cIh3 # EaSbSgc+zQbma+ZTjRDOtovwjlHizc0a00UbsT2NOX5LHFZz3UHy+m/pHecmddRd # ReBuhKWDaK3yBnZ4XjFnvz92pyxbN6GCF5cwgheTBgorBgEEAYI3AwMBMYIXgzCC # F38GCSqGSIb3DQEHAqCCF3AwghdsAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFSBgsq # hkiG9w0BCRABBKCCAUEEggE9MIIBOQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl # AwQCAQUABCBb6MdlnDwhtyfYlCtCYZ5+8nY5k8uB0P+5KZHcH6oKAwIGaZRIde1r # GBMyMDI2MDIxOTAzMzIwNi4xMjJaMASAAgH0oIHRpIHOMIHLMQswCQYDVQQGEwJV # UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE # ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1l # cmljYSBPcGVyYXRpb25zMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046QTQwMC0w # NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Wg # ghHtMIIHIDCCBQigAwIBAgITMwAAAgJ5UHQhFH24oQABAAACAjANBgkqhkiG9w0B # AQsFADB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD # VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAeFw0yNTAxMzAxOTQy # NDRaFw0yNjA0MjIxOTQyNDRaMIHLMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2Fz # aGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENv # cnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1lcmljYSBPcGVyYXRpb25z # MScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046QTQwMC0wNUUwLUQ5NDcxJTAjBgNV # BAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQC3eSp6cucUGOkcPg4vKWKJfEQeshK2ZBsYU1tDWvQu # 6L9lp+dnqrajIdNeH1HN3oz3iiGoJWuN2HVNZkcOt38aWGebM0gUUOtPjuLhuO5d # 67YpQsHBJAWhcve/MVdoQPj1njiAjSiOrL8xFarFLI46RH8NeDhAPXcJpWn7AIzC # yIjZOaJ2DWA+6QwNzwqjBgIpf1hWFwqHvPEedy0notXbtWfT9vCSL9sdDK6K/HH9 # HsaY5wLmUUB7SfuLGo1OWEm6MJyG2jixqi9NyRoypdF8dRyjWxKRl2JxwvbetlDT # io66XliTOckq2RgM+ZocZEb6EoOdtd0XKh3Lzx29AhHxlk+6eIwavlHYuOLZDKod # POVN6j1IJ9brolY6mZboQ51Oqe5nEM5h/WJX28GLZioEkJN8qOe5P5P2Yx9HoOqL # ugX00qCzxq4BDm8xH85HKxvKCO5KikopaRGGtQlXjDyusMWlrHcySt56DhL4dcVn # n7dFvL50zvQlFZMhVoehWSQkkWuUlCCqIOrTe7RbmnbdJosH+7lC+n53gnKy4OoZ # zuUeqzCnSB1JNXPKnJojP3De5xwspi5tUvQFNflfGTsjZgQAgDBdg/DO0TGgLRDK # vZQCZ5qIuXpQRyg37yc51e95z8U2mysU0XnSpWeigHqkyOAtDfcIpq5Gv7HV+da2 # RwIDAQABo4IBSTCCAUUwHQYDVR0OBBYEFNoGubUPjP2f8ifkIKvwy1rlSHTZMB8G # A1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRYMFYwVKBSoFCG # Tmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY3Jvc29mdCUy # MFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEFBQcBAQRgMF4w # XAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2Vy # dHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3J0MAwG # A1UdEwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwgwDgYDVR0PAQH/BAQD # AgeAMA0GCSqGSIb3DQEBCwUAA4ICAQCD83aFQUxN37HkOoJDM1maHFZVUGcqTQcP # nOD6UoYRMmDKv0GabHlE82AYgLPuVlukn7HtJPF2z0jnTgAfRMn26JFLPG7O/XbK # K25hrBPJ30lBuwjATVt58UA1BWo7lsmnyrur/6h8AFzrXyrXtlvzQYqaRYY9k0UF # Y5GM+n9YaEEK2D268e+a+HDmWe+tYL2H+9O4Q1MQLag+ciNwLkj/+QlxpXiWou9K # vAP0tIk+fH8F3ww5VOTi9aZ9+qPjszw31H4ndtivBZaH5s5boJmH2JbtMuf2y7hS # dJdE0UW2B0FEZPLImemlKhslJNVqEO7RPgl7c81QuVSO58ffpmbwtSxhYrES3VsP # glXn9ODF7DqmPMG/GysB4o/QkpNUq+wS7bORTNzqHMtH+ord2YSma+1byWBr/izI # KggOCdEzaZDfym12GM6a4S+Iy6AUIp7/KIpAmfWfXrcMK7V7EBzxoezkLREEWI4X # tPwpEBntOa1oDH3Z/+dRxsxL0vgya7jNfrO7oizTAln/2ZBYB9ioUeobj5AGL45m # 2mcKSk7HE5zUReVkILpYKBQ5+X/8jFO1/pZyqzQeI1/oJ/RLoic1SieLXfET9EWZ # IBjZMZ846mDbp1ynK9UbNiCjSwmTF509Yn9M47VQsxsv1olQu51rVVHkSNm+rTrL # wK1tvhv0mTCCB3EwggVZoAMCAQICEzMAAAAVxedrngKbSZkAAAAAABUwDQYJKoZI # 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 # MCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOkE0MDAtMDVFMC1EOTQ3MSUwIwYDVQQD # ExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNloiMKAQEwBwYFKw4DAhoDFQBJ # iUhpCWA/3X/jZyIy0ye6RJwLzqCBgzCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1w # IFBDQSAyMDEwMA0GCSqGSIb3DQEBCwUAAgUA7UDA7DAiGA8yMDI2MDIxODIyNTEy # NFoYDzIwMjYwMjE5MjI1MTI0WjB3MD0GCisGAQQBhFkKBAExLzAtMAoCBQDtQMDs # AgEAMAoCAQACAgO4AgH/MAcCAQACAhP4MAoCBQDtQhJsAgEAMDYGCisGAQQBhFkK # BAIxKDAmMAwGCisGAQQBhFkKAwKgCjAIAgEAAgMHoSChCjAIAgEAAgMBhqAwDQYJ # KoZIhvcNAQELBQADggEBAFfEi8iu+gv3w4bViBZdPZdzXn+/25m4QOaTys/MYtQD # p2niT0MTR3hZD8Lc7qC8HGatHcoUBXeqz7HHICMZLcXCuvGuKJG4b1U8XDge+EJr # YSJoM50U0yoBGhHVfZwR3S4cPlOwguAjGkZDnr/PWo3KPRcBrhqUBSRAXDAIREt3 # v20VHxBbsk2d/aoAk/kKNYaIJL/jIgLM7JJl0hj1Bfd6BmN+IGCN/PN47Ppyz6G7 # Kgyio4FV3VkZX9MDJ0pw20LmSIP1mjJEkBxwTIFi86okHcYHKABQp9AY8uZtfIVM # 4HGKYqj27Xm6z1JHIbqBIac/K7FP8dkcZ2Dk+OfRZ+UxggQNMIIECQIBATCBkzB8 # MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVk # bW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1N # aWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMAITMwAAAgJ5UHQhFH24oQABAAAC # AjANBglghkgBZQMEAgEFAKCCAUowGgYJKoZIhvcNAQkDMQ0GCyqGSIb3DQEJEAEE # MC8GCSqGSIb3DQEJBDEiBCBHt35URPcED/cggxf1KZEBxX2Bbl1gkMTOFdYZ/uCL # sDCB+gYLKoZIhvcNAQkQAi8xgeowgecwgeQwgb0EIPON6gEYB5bLzXLWuUmL8Zd8 # xXAsqXksedFyolfMlF/sMIGYMIGApH4wfDELMAkGA1UEBhMCVVMxEzARBgNVBAgT # Cldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m # dCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENB # IDIwMTACEzMAAAICeVB0IRR9uKEAAQAAAgIwIgQgT/th+Kt8L02XOqvUN6NL6msn # dIL1dRN5NEfKfOEoIygwDQYJKoZIhvcNAQELBQAEggIAJvFJE1vNCM0uWwtttP8c # Tc76ug9+1LeMl5u0D+46B0Oxb2i8WC14+Qqc0g3cFlYynE8Q2ncDLi2HJBM9HdNe # Y97k428MVTg2W2lbIjkQ6yIF8n5aPcmjNpP2tq5xWMXHeNBpizw+c/zGXYvPBxk6 # P6OtS3Mdfy2H2IXCRJsSWCM9WU33G5C/RLBDbc1ZfLVEhuLSH5TT1GDfCndjA5GG # 243OcXASfxXfNTuN2tDM8aVzJjhyWJc0mbjFm17FyqJ0FcAuJY2KdMXiVVqjMCjx # POyjGiecLwzXD2vZ4fhnYitr7NPcdbz4K91dTxa0BQiDVv8oYvP4cUA3td+I5Nd/ # +2BUdnHv+xOqbF1a5SzCJERhVULt6qyNGNidXwrEETyGLmxWM/d6nD7NKd4cd7wK # dJxfivOrMekKUkVqflXoLloILyZWkr47ou8amFNCW+xzH09t6hhLdSeN0qmrgfCc # 9UxNyAuuvxBIFJARMKNGCMbfWHncbMcT1/DshxpTfMegh0kFQ9uWFW6Bc9DJ/3Uk # XaHeCFoIifg6qSnxHJgIFj1maS1LwdmCBCX13mEaDqUbVvmqFKblbZa8PrsgnOmU # /7IWJkZFsd2Iyt6/wQtmvEF73YFYtactcEb24eopGr4i/Lqx9bNCaDa268eDVgi3 # yKfz1uCzg4OWbhU9t8xlqzc= # SIG # End signature block |