CommonUtil.ps1
<# Common Utility functions. #> . $PSScriptRoot/PureStorage.CBS.VVolUtil.ps1 Import-Module Posh-SSH function Get-PCBSHostGroup { <# .SYNOPSIS Get all host groups from a Flash Array .DESCRIPTION Get list of host groups from Flash Array included IP Ranges .PARAMETER FlasharrayAddress Flash Array IP address of FQDN .PARAMETER FlasharrayUsername Flash Array username .PARAMETER FlasharrayPassword Flash Array password #> Param( [Parameter(mandatory=$true)] [String]$FlasharrayAddress, [Parameter(mandatory=$true)] [String]$FlasharrayUsername, [Parameter(mandatory=$true)] [SecureString]$FlasharrayPassword ) $hgroup_result = Invoke-Pfa2CLICommand -Endpoint $FlasharrayAddress -Username $FlasharrayUsername -Password $FlasharrayPassword -CommandText "purehgroup list --csv" $result = ConvertFrom-Csv -InputObject $hgroup_result # TODO: Verify that "IP Range" is in result foreach ($r in $result) { $members = $r | Get-Member | Where {$_.Name -eq "IP Range"} if ($members.Count -le 0) { throw "Could not find 'IP Range'. PLease make sure to use a supported FlashArray" } } return $result } function Get-PCBSHostGroupByIPRange { <# .SYNOPSIS Get host groups from a Flash Array with a specific IP Range .DESCRIPTION Get host groups from a Flash Array with a specific IP Range (inclusive of start and end). Will return an empty list if no host groups are found with the specified IP Range .PARAMETER FlasharrayAddress Flash Array IP address of FQDN .PARAMETER FlasharrayUsername Flash Array username .PARAMETER FlasharrayPassword Flash Array password .PARAMETER IPRangeStart IP Range start .PARAMETER IPRangeEnd IP Range end #> Param( [Parameter(mandatory=$true)] [String]$FlasharrayAddress, [Parameter(mandatory=$true)] [String]$FlasharrayUsername, [Parameter(mandatory=$true)] [SecureString]$FlasharrayPassword, [Parameter(mandatory=$true)] [String] $IPRangeStart, [Parameter(mandatory=$true)] [String] $IPRangeEnd ) $result = @() $hgroups = Get-PCBSHostGroup -FlasharrayAddress $FlasharrayAddress -FlasharrayUsername $FlasharrayUsername -FlasharrayPassword $FlasharrayPassword foreach ($hgroup in $hgroups) { # If more than one IP Range is listed, they will be separated by "/" $ip_ranges = $hgroup."IP Range" -Split "/" | ConvertFrom-Json foreach ($ip_range in $ip_ranges) { if ($ip_range."start" -eq $IPRangeStart -and $ip_range."start" -eq $IPRangeStart) { $result += $hgroup } } } return $result } function New-PCBSHostGroup { <# .SYNOPSIS Create a new host group on Flash Array .DESCRIPTION Create a new host group on with a specified Ip Range Flash Array .PARAMETER FlasharrayAddress Flash Array IP address of FQDN .PARAMETER FlasharrayUsername Flash Array username .PARAMETER FlasharrayPassword Flash Array password .PARAMETER HostGroupName Host group name .PARAMETER HostGroupNameSpace Host group namespace .PARAMETER IPRangeStart IP Range start .PARAMETER IPRangeEnd IP Range end .EXAMPLE An example .NOTES General notes #> Param( [Parameter(mandatory=$true)] [String]$FlasharrayAddress, [Parameter(mandatory=$true)] [String]$FlasharrayUsername, [Parameter(mandatory=$true)] [SecureString]$FlasharrayPassword, [Parameter(mandatory=$true)] [String]$HostGroupName, [Parameter(mandatory=$false)] [String]$HostGroupNameSpace, [Parameter(mandatory=$true)] [String] $IPRangeStart, [Parameter(mandatory=$true)] [String] $IPRangeEnd ) $command = "purehgroup create --iprangelist $IPRangeStart-$IPRangeEnd --personality esxi $HostGroupName" $result = Invoke-Pfa2CLICommand -Endpoint $FlasharrayAddress -Username $FlasharrayUsername -Password $FlasharrayPassword -CommandText $command return $result } function Get-PurityIdentifiefFromVMWareIdentifier { Param ( [String] $name ) $updated_name = $name if (-not $name -match "^[a-zA-Z0-9\-]+$") { $updated_name = $updated_name -replace "[^\w\-]", "" $updated_name = $updated_name -replace "[_]", "" $updated_name = $updated_name -replace " ", "" } return $updated_name } function Convert-EpochToDateTime { Param ( [int64] $EpochMillseconds ) $start_date = New-Object -Type DateTime -ArgumentList 1970, 1, 1, 0, 0, 0, 0 return $start_date.AddMilliseconds($EpochMillseconds) } function Update-ESXCertificates { <# .SYNOPSIS Refresh certificates for all ESXi hosts .DESCRIPTION Refresh certificates for all ESXi hosts #> $service_instance = Get-View ServiceInstance $certMgr = Get-View -Id $service_instance.Content.CertificateManager Get-VMHost | ForEach-Object -Process { Write-Host "Refreshing certificates for $($_.Name).." $certMgr.CertMgrRefreshCACertificatesAndCRLs($_.Id) } } function Remove-PCBSStaticiSCSITargets { <# .SYNOPSIS Remove statis iSCSI Targets .DESCRIPTION Remove statis iSCSI Targets .PARAMETER Esxi ESXi VMware object #> Param( [Parameter(Position=0,mandatory=$true)] [VMware.VimAutomation.ViCore.Types.V1.Inventory.VMHost]$Esxi ) $ESXi | Get-VMHostHba -Type iScsi | Get-IScsiHbaTarget | Where {$_.Type -eq "Static"} | Remove-IScsiHbaTarget -Confirm:$false } function New-PCBSDynamicHostGroup { Param ( [String]$ClusterName, [String]$iSCSIRangeStart, [String]$iSCSIRangeEnd, [String]$FlasharrayAddress, [String]$FlasharrayUsername, [SecureString]$FlasharrayPassword ) # Create dynmaic host group if it does not exist $hgroups = Get-PCBSHostGroupByIPRange -FlasharrayAddress $FlasharrayAddress -FlasharrayUsername $FlasharrayUsername -FlasharrayPassword $FlasharrayPassword -IPRangeStart $iSCSIRangeStart -IPRangeEnd $iSCSIRangeEnd if ($hgroups.Count -eq 0) { # HostGgroup does not exist. We need to create it $HostGroupName = Get-PurityIdentifiefFromVMWareIdentifier -Name $ClusterName Write-Host "Creating host group $HostGroupName ..." $hgroup = New-PCBSHostGroup -FlasharrayAddress $FlasharrayAddress -FlasharrayUsername $FlasharrayUsername -FlasharrayPassword $FlasharrayPassword -IPRangeStart $iSCSIRangeStart -IPRangeEnd $iSCSIRangeEnd -HostGroupName $HostGroupName Write-Host "$HostGroupName was created successfully." } else { $HostGroupName = $hgroups[0].Name Write-Host "HostGroup ($HostGroupName) for iSCSI Range ($iSCSIRangeStart -> $iSCSIRangeEnd) already exists." } # Create software iSCSI adaptor for each ESXi host $fa = Connect-Pfa2Array -EndPoint $FlasharrayAddress -Username $FlasharrayUsername -Password $FlasharrayPassword -IgnoreCertificateError $ESXHosts = $Cluster | Get-VMHost foreach ($ESXHost in $ESXHosts) { Write-Host "Configuring iSCSI target for $ESXHost ..." Set-VmHostPfaiSCSI -esxi $ESXHost -flasharray $fa Write-Host "Removing static iSCSI targets for $ESXHost ..." Remove-StaticiSCSITargets -esxi $ESXHost Write-Host "Rescanning HBA for $ESXHost ..." Get-VMHostStorage -VMHost $ESXHosts -RescanAllHba | Out-Null } } function Get-AvailablePort { Param ( [Parameter(Mandatory=$true)] [int]$StartRange, [Parameter(Mandatory=$true)] [int]$EndRange ) $ipProperties = [Net.NetworkInformation.IPGlobalProperties]::GetIPGlobalProperties() [int[]]$listenerPorts = $ipProperties.GetActiveTcpListeners().Port $port = $StartRange if ($listenerPorts -and $listenerPorts.Count -gt 0) { while ($port -le $EndRange) { if ($listenerPorts.IndexOf($port) -ge 0) { # Port is already used $port = $port +1 } else { break; } } } if ($port -gt $EndRange) { # No port found throw "Could not find available port." } return $port } function New-PCBSConnection { Param ( [Parameter(Mandatory=$true)] [string] $Endpoint ) $port = Get-AvailablePort -StartRange 1500 -EndRange 5000 $sessions = Get-Variable -Name SSH_Sessions -Scope Global $sshSession = $sessions.Value["VC"].Value New-SSHLocalPortForward -SSHSession $sshSession -BoundHost localhost -BoundPort $port -RemoteAddress $Endpoint -RemotePort 443 $newEndpoint = "localhost:$port" return $newEndpoint } # SIG # Begin signature block # MIIjUAYJKoZIhvcNAQcCoIIjQTCCIz0CAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB # gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR # AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUcNOZ46nLeYbOB4YErKezCt9P # g5aggh12MIIFMDCCBBigAwIBAgIQBAkYG1/Vu2Z1U0O1b5VQCDANBgkqhkiG9w0B # AQsFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYD # VQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBBc3N1cmVk # IElEIFJvb3QgQ0EwHhcNMTMxMDIyMTIwMDAwWhcNMjgxMDIyMTIwMDAwWjByMQsw # CQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cu # ZGlnaWNlcnQuY29tMTEwLwYDVQQDEyhEaWdpQ2VydCBTSEEyIEFzc3VyZWQgSUQg # Q29kZSBTaWduaW5nIENBMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA # +NOzHH8OEa9ndwfTCzFJGc/Q+0WZsTrbRPV/5aid2zLXcep2nQUut4/6kkPApfmJ # 1DcZ17aq8JyGpdglrA55KDp+6dFn08b7KSfH03sjlOSRI5aQd4L5oYQjZhJUM1B0 # sSgmuyRpwsJS8hRniolF1C2ho+mILCCVrhxKhwjfDPXiTWAYvqrEsq5wMWYzcT6s # cKKrzn/pfMuSoeU7MRzP6vIK5Fe7SrXpdOYr/mzLfnQ5Ng2Q7+S1TqSp6moKq4Tz # rGdOtcT3jNEgJSPrCGQ+UpbB8g8S9MWOD8Gi6CxR93O8vYWxYoNzQYIH5DiLanMg # 0A9kczyen6Yzqf0Z3yWT0QIDAQABo4IBzTCCAckwEgYDVR0TAQH/BAgwBgEB/wIB # ADAOBgNVHQ8BAf8EBAMCAYYwEwYDVR0lBAwwCgYIKwYBBQUHAwMweQYIKwYBBQUH # AQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wQwYI # KwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFz # c3VyZWRJRFJvb3RDQS5jcnQwgYEGA1UdHwR6MHgwOqA4oDaGNGh0dHA6Ly9jcmw0 # LmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcmwwOqA4oDaG # NGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RD # QS5jcmwwTwYDVR0gBEgwRjA4BgpghkgBhv1sAAIEMCowKAYIKwYBBQUHAgEWHGh0 # dHBzOi8vd3d3LmRpZ2ljZXJ0LmNvbS9DUFMwCgYIYIZIAYb9bAMwHQYDVR0OBBYE # FFrEuXsqCqOl6nEDwGD5LfZldQ5YMB8GA1UdIwQYMBaAFEXroq/0ksuCMS1Ri6en # IZ3zbcgPMA0GCSqGSIb3DQEBCwUAA4IBAQA+7A1aJLPzItEVyCx8JSl2qB1dHC06 # GsTvMGHXfgtg/cM9D8Svi/3vKt8gVTew4fbRknUPUbRupY5a4l4kgU4QpO4/cY5j # DhNLrddfRHnzNhQGivecRk5c/5CxGwcOkRX7uq+1UcKNJK4kxscnKqEpKBo6cSgC # PC6Ro8AlEeKcFEehemhor5unXCBc2XGxDI+7qPjFEmifz0DLQESlE/DmZAwlCEIy # sjaKJAL+L3J+HNdJRZboWR3p+nRka7LrZkPas7CM1ekN3fYBIM6ZMWM9CBoYs4Gb # T8aTEAb8B4H6i9r5gkn3Ym6hU/oSlBiFLpKR6mhsRDKyZqHnGKSaZFHvMIIFNzCC # BB+gAwIBAgIQC4jZOitkx57ksuMgsWXX0jANBgkqhkiG9w0BAQsFADByMQswCQYD # VQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGln # aWNlcnQuY29tMTEwLwYDVQQDEyhEaWdpQ2VydCBTSEEyIEFzc3VyZWQgSUQgQ29k # ZSBTaWduaW5nIENBMB4XDTIwMDczMDAwMDAwMFoXDTIzMTAwNDEyMDAwMFowdDEL # MAkGA1UEBhMCVVMxEzARBgNVBAgTCkNhbGlmb3JuaWExFjAUBgNVBAcTDU1vdW50 # YWluIFZpZXcxGzAZBgNVBAoTElB1cmUgU3RvcmFnZSwgSW5jLjEbMBkGA1UEAxMS # UHVyZSBTdG9yYWdlLCBJbmMuMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKC # AQEA6nefE6+A0nNMY82xtkQb+akwI0oxLqEbRY65bE4re+CVQV2xP89/7FIAXooq # jxNOvrtWicWTGOZjBdAFEXXAUEyu9CkWFOXGLV3/QkcEfY3e3Z3jypa6h1EznSEp # 3wSQEIVbigi6jR2s7NDnDoSDAKnzcGcSZ8Nz7akXFrN8PAmg3gy8a/rwm2Ko7ClR # ZUHj1C/OMPXUqiN0Q4FyAsaeFmvg2PX2twxo192WRdNro1dkKNfmDvym2ss6MXcq # gFEjBcZtHDv3e/i0BjT24Jm1C27IVYZzVf28dqmVfPo7l8bLEHsVKgrRf0VzS2wI # R08gXiMrohf9tv3AbvtPk7ZaawIDAQABo4IBxTCCAcEwHwYDVR0jBBgwFoAUWsS5 # eyoKo6XqcQPAYPkt9mV1DlgwHQYDVR0OBBYEFJOwFn7S4Lj6QdNnH6tpm3FMFWJI # MA4GA1UdDwEB/wQEAwIHgDATBgNVHSUEDDAKBggrBgEFBQcDAzB3BgNVHR8EcDBu # MDWgM6Axhi9odHRwOi8vY3JsMy5kaWdpY2VydC5jb20vc2hhMi1hc3N1cmVkLWNz # LWcxLmNybDA1oDOgMYYvaHR0cDovL2NybDQuZGlnaWNlcnQuY29tL3NoYTItYXNz # dXJlZC1jcy1nMS5jcmwwTAYDVR0gBEUwQzA3BglghkgBhv1sAwEwKjAoBggrBgEF # BQcCARYcaHR0cHM6Ly93d3cuZGlnaWNlcnQuY29tL0NQUzAIBgZngQwBBAEwgYQG # CCsGAQUFBwEBBHgwdjAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQu # Y29tME4GCCsGAQUFBzAChkJodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGln # aUNlcnRTSEEyQXNzdXJlZElEQ29kZVNpZ25pbmdDQS5jcnQwDAYDVR0TAQH/BAIw # ADANBgkqhkiG9w0BAQsFAAOCAQEAMVAa7mXxhBHf0dTzf6LKDncN8KnzB59KBvd0 # KvXZc6FoLHGi2Wg6XBSP+9mdDMMYOkohqstSk7RD+reT8xiptrIkSMcVcTog1Z3e # JjYTK8B7QsSpuu2lo0RWA5rdvqMJ+lVzbbjteTq+uicP4T/EDwv2q+iPAgpXQD8y # r084ExDWJtMfhvy0cxh555xx88rvFWOhJnXYiFtjaO9dp7f2TnZRJ44rmB98jc9E # BR/8GLOi/BhyPiiU4nBv8JIHVP1E5zIt8/9PhfpenmiWBbuuP0YLnzrqRhswtJaq # jJirYNLYojmINrbvdcpEKGK1AitsnuOjFadLI7bc696Y35lUATCCBY0wggR1oAMC # AQICEA6bGI750C3n79tQ4ghAGFowDQYJKoZIhvcNAQEMBQAwZTELMAkGA1UEBhMC # VVMxFTATBgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0 # LmNvbTEkMCIGA1UEAxMbRGlnaUNlcnQgQXNzdXJlZCBJRCBSb290IENBMB4XDTIy # MDgwMTAwMDAwMFoXDTMxMTEwOTIzNTk1OVowYjELMAkGA1UEBhMCVVMxFTATBgNV # BAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTEhMB8G # A1UEAxMYRGlnaUNlcnQgVHJ1c3RlZCBSb290IEc0MIICIjANBgkqhkiG9w0BAQEF # AAOCAg8AMIICCgKCAgEAv+aQc2jeu+RdSjwwIjBpM+zCpyUuySE98orYWcLhKac9 # WKt2ms2uexuEDcQwH/MbpDgW61bGl20dq7J58soR0uRf1gU8Ug9SH8aeFaV+vp+p # VxZZVXKvaJNwwrK6dZlqczKU0RBEEC7fgvMHhOZ0O21x4i0MG+4g1ckgHWMpLc7s # Xk7Ik/ghYZs06wXGXuxbGrzryc/NrDRAX7F6Zu53yEioZldXn1RYjgwrt0+nMNlW # 7sp7XeOtyU9e5TXnMcvak17cjo+A2raRmECQecN4x7axxLVqGDgDEI3Y1DekLgV9 # iPWCPhCRcKtVgkEy19sEcypukQF8IUzUvK4bA3VdeGbZOjFEmjNAvwjXWkmkwuap # oGfdpCe8oU85tRFYF/ckXEaPZPfBaYh2mHY9WV1CdoeJl2l6SPDgohIbZpp0yt5L # HucOY67m1O+SkjqePdwA5EUlibaaRBkrfsCUtNJhbesz2cXfSwQAzH0clcOP9yGy # shG3u3/y1YxwLEFgqrFjGESVGnZifvaAsPvoZKYz0YkH4b235kOkGLimdwHhD5QM # IR2yVCkliWzlDlJRR3S+Jqy2QXXeeqxfjT/JvNNBERJb5RBQ6zHFynIWIgnffEx1 # P2PsIV/EIFFrb7GrhotPwtZFX50g/KEexcCPorF+CiaZ9eRpL5gdLfXZqbId5RsC # AwEAAaOCATowggE2MA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYEFOzX44LScV1k # TN8uZz/nupiuHA9PMB8GA1UdIwQYMBaAFEXroq/0ksuCMS1Ri6enIZ3zbcgPMA4G # A1UdDwEB/wQEAwIBhjB5BggrBgEFBQcBAQRtMGswJAYIKwYBBQUHMAGGGGh0dHA6 # Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBDBggrBgEFBQcwAoY3aHR0cDovL2NhY2VydHMu # ZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENBLmNydDBFBgNVHR8E # PjA8MDqgOKA2hjRodHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRBc3N1 # cmVkSURSb290Q0EuY3JsMBEGA1UdIAQKMAgwBgYEVR0gADANBgkqhkiG9w0BAQwF # AAOCAQEAcKC/Q1xV5zhfoKN0Gz22Ftf3v1cHvZqsoYcs7IVeqRq7IviHGmlUIu2k # iHdtvRoU9BNKei8ttzjv9P+Aufih9/Jy3iS8UgPITtAq3votVs/59PesMHqai7Je # 1M/RQ0SbQyHrlnKhSLSZy51PpwYDE3cnRNTnf+hZqPC/Lwum6fI0POz3A8eHqNJM # QBk1RmppVLC4oVaO7KTVPeix3P0c2PR3WlxUjG/voVA9/HYJaISfb8rbII01YBwC # A8sgsKxYoA5AY8WYIsGyWfVVa88nq2x2zm8jLfR+cWojayL/ErhULSd+2DrZ8LaH # lv1b0VysGMNNn3O3AamfV6peKOK5lDCCBq4wggSWoAMCAQICEAc2N7ckVHzYR6z9 # KGYqXlswDQYJKoZIhvcNAQELBQAwYjELMAkGA1UEBhMCVVMxFTATBgNVBAoTDERp # Z2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTEhMB8GA1UEAxMY # RGlnaUNlcnQgVHJ1c3RlZCBSb290IEc0MB4XDTIyMDMyMzAwMDAwMFoXDTM3MDMy # MjIzNTk1OVowYzELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMu # MTswOQYDVQQDEzJEaWdpQ2VydCBUcnVzdGVkIEc0IFJTQTQwOTYgU0hBMjU2IFRp # bWVTdGFtcGluZyBDQTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAMaG # NQZJs8E9cklRVcclA8TykTepl1Gh1tKD0Z5Mom2gsMyD+Vr2EaFEFUJfpIjzaPp9 # 85yJC3+dH54PMx9QEwsmc5Zt+FeoAn39Q7SE2hHxc7Gz7iuAhIoiGN/r2j3EF3+r # GSs+QtxnjupRPfDWVtTnKC3r07G1decfBmWNlCnT2exp39mQh0YAe9tEQYncfGpX # evA3eZ9drMvohGS0UvJ2R/dhgxndX7RUCyFobjchu0CsX7LeSn3O9TkSZ+8OpWNs # 5KbFHc02DVzV5huowWR0QKfAcsW6Th+xtVhNef7Xj3OTrCw54qVI1vCwMROpVymW # Jy71h6aPTnYVVSZwmCZ/oBpHIEPjQ2OAe3VuJyWQmDo4EbP29p7mO1vsgd4iFNmC # KseSv6De4z6ic/rnH1pslPJSlRErWHRAKKtzQ87fSqEcazjFKfPKqpZzQmiftkaz # nTqj1QPgv/CiPMpC3BhIfxQ0z9JMq++bPf4OuGQq+nUoJEHtQr8FnGZJUlD0UfM2 # SU2LINIsVzV5K6jzRWC8I41Y99xh3pP+OcD5sjClTNfpmEpYPtMDiP6zj9NeS3YS # UZPJjAw7W4oiqMEmCPkUEBIDfV8ju2TjY+Cm4T72wnSyPx4JduyrXUZ14mCjWAkB # KAAOhFTuzuldyF4wEr1GnrXTdrnSDmuZDNIztM2xAgMBAAGjggFdMIIBWTASBgNV # HRMBAf8ECDAGAQH/AgEAMB0GA1UdDgQWBBS6FtltTYUvcyl2mi91jGogj57IbzAf # BgNVHSMEGDAWgBTs1+OC0nFdZEzfLmc/57qYrhwPTzAOBgNVHQ8BAf8EBAMCAYYw # EwYDVR0lBAwwCgYIKwYBBQUHAwgwdwYIKwYBBQUHAQEEazBpMCQGCCsGAQUFBzAB # hhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wQQYIKwYBBQUHMAKGNWh0dHA6Ly9j # YWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRSb290RzQuY3J0MEMG # A1UdHwQ8MDowOKA2oDSGMmh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2Vy # dFRydXN0ZWRSb290RzQuY3JsMCAGA1UdIAQZMBcwCAYGZ4EMAQQCMAsGCWCGSAGG # /WwHATANBgkqhkiG9w0BAQsFAAOCAgEAfVmOwJO2b5ipRCIBfmbW2CFC4bAYLhBN # E88wU86/GPvHUF3iSyn7cIoNqilp/GnBzx0H6T5gyNgL5Vxb122H+oQgJTQxZ822 # EpZvxFBMYh0MCIKoFr2pVs8Vc40BIiXOlWk/R3f7cnQU1/+rT4osequFzUNf7WC2 # qk+RZp4snuCKrOX9jLxkJodskr2dfNBwCnzvqLx1T7pa96kQsl3p/yhUifDVinF2 # ZdrM8HKjI/rAJ4JErpknG6skHibBt94q6/aesXmZgaNWhqsKRcnfxI2g55j7+6ad # cq/Ex8HBanHZxhOACcS2n82HhyS7T6NJuXdmkfFynOlLAlKnN36TU6w7HQhJD5TN # OXrd/yVjmScsPT9rp/Fmw0HNT7ZAmyEhQNC3EyTN3B14OuSereU0cZLXJmvkOHOr # pgFPvT87eK1MrfvElXvtCl8zOYdBeHo46Zzh3SP9HSjTx/no8Zhf+yvYfvJGnXUs # HicsJttvFXseGYs2uJPU5vIXmVnKcPA3v5gA3yAWTyf7YGcWoWa63VXAOimGsJig # K+2VQbc61RWYMbRiCQ8KvYHZE/6/pNHzV9m8BPqC3jLfBInwAM1dwvnQI38AC+R2 # AibZ8GV2QqYphwlHK+Z/GqSFD/yYlvZVVCsfgPrA8g4r5db7qS9EFUrnEw4d2zc4 # GqEr9u3WfPwwggbAMIIEqKADAgECAhAMTWlyS5T6PCpKPSkHgD1aMA0GCSqGSIb3 # DQEBCwUAMGMxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjE7 # MDkGA1UEAxMyRGlnaUNlcnQgVHJ1c3RlZCBHNCBSU0E0MDk2IFNIQTI1NiBUaW1l # U3RhbXBpbmcgQ0EwHhcNMjIwOTIxMDAwMDAwWhcNMzMxMTIxMjM1OTU5WjBGMQsw # CQYDVQQGEwJVUzERMA8GA1UEChMIRGlnaUNlcnQxJDAiBgNVBAMTG0RpZ2lDZXJ0 # IFRpbWVzdGFtcCAyMDIyIC0gMjCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoC # ggIBAM/spSY6xqnya7uNwQ2a26HoFIV0MxomrNAcVR4eNm28klUMYfSdCXc9FZYI # L2tkpP0GgxbXkZI4HDEClvtysZc6Va8z7GGK6aYo25BjXL2JU+A6LYyHQq4mpOS7 # eHi5ehbhVsbAumRTuyoW51BIu4hpDIjG8b7gL307scpTjUCDHufLckkoHkyAHoVW # 54Xt8mG8qjoHffarbuVm3eJc9S/tjdRNlYRo44DLannR0hCRRinrPibytIzNTLlm # yLuqUDgN5YyUXRlav/V7QG5vFqianJVHhoV5PgxeZowaCiS+nKrSnLb3T254xCg/ # oxwPUAY3ugjZNaa1Htp4WB056PhMkRCWfk3h3cKtpX74LRsf7CtGGKMZ9jn39cFP # cS6JAxGiS7uYv/pP5Hs27wZE5FX/NurlfDHn88JSxOYWe1p+pSVz28BqmSEtY+VZ # 9U0vkB8nt9KrFOU4ZodRCGv7U0M50GT6Vs/g9ArmFG1keLuY/ZTDcyHzL8IuINeB # rNPxB9ThvdldS24xlCmL5kGkZZTAWOXlLimQprdhZPrZIGwYUWC6poEPCSVT8b87 # 6asHDmoHOWIZydaFfxPZjXnPYsXs4Xu5zGcTB5rBeO3GiMiwbjJ5xwtZg43G7vUs # fHuOy2SJ8bHEuOdTXl9V0n0ZKVkDTvpd6kVzHIR+187i1Dp3AgMBAAGjggGLMIIB # hzAOBgNVHQ8BAf8EBAMCB4AwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggr # BgEFBQcDCDAgBgNVHSAEGTAXMAgGBmeBDAEEAjALBglghkgBhv1sBwEwHwYDVR0j # BBgwFoAUuhbZbU2FL3MpdpovdYxqII+eyG8wHQYDVR0OBBYEFGKK3tBh/I8xFO2X # C809KpQU31KcMFoGA1UdHwRTMFEwT6BNoEuGSWh0dHA6Ly9jcmwzLmRpZ2ljZXJ0 # LmNvbS9EaWdpQ2VydFRydXN0ZWRHNFJTQTQwOTZTSEEyNTZUaW1lU3RhbXBpbmdD # QS5jcmwwgZAGCCsGAQUFBwEBBIGDMIGAMCQGCCsGAQUFBzABhhhodHRwOi8vb2Nz # cC5kaWdpY2VydC5jb20wWAYIKwYBBQUHMAKGTGh0dHA6Ly9jYWNlcnRzLmRpZ2lj # ZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNFJTQTQwOTZTSEEyNTZUaW1lU3RhbXBp # bmdDQS5jcnQwDQYJKoZIhvcNAQELBQADggIBAFWqKhrzRvN4Vzcw/HXjT9aFI/H8 # +ZU5myXm93KKmMN31GT8Ffs2wklRLHiIY1UJRjkA/GnUypsp+6M/wMkAmxMdsJiJ # 3HjyzXyFzVOdr2LiYWajFCpFh0qYQitQ/Bu1nggwCfrkLdcJiXn5CeaIzn0buGqi # m8FTYAnoo7id160fHLjsmEHw9g6A++T/350Qp+sAul9Kjxo6UrTqvwlJFTU2WZoP # VNKyG39+XgmtdlSKdG3K0gVnK3br/5iyJpU4GYhEFOUKWaJr5yI+RCHSPxzAm+18 # SLLYkgyRTzxmlK9dAlPrnuKe5NMfhgFknADC6Vp0dQ094XmIvxwBl8kZI4DXNlpf # lhaxYwzGRkA7zl011Fk+Q5oYrsPJy8P7mxNfarXH4PMFw1nfJ2Ir3kHJU7n/NBBn # 9iYymHv+XEKUgZSCnawKi8ZLFUrTmJBFYDOA4CPe+AOk9kVH5c64A0JH6EE2cXet # /aLol3ROLtoeHYxayB6a1cLwxiKoT5u92ByaUcQvmvZfpyeXupYuhVfAYOd4Vn9q # 78KVmksRAsiCnMkaBXy6cbVOepls9Oie1FqYyJ+/jbsYXEP10Cro4mLueATbvdH7 # WwqocH7wl4R44wgDXUcsY6glOJcB0j862uXl9uab3H4szP8XTE0AotjWAQ64i+7m # 4HJViSwnGWH2dwGMMYIFRDCCBUACAQEwgYYwcjELMAkGA1UEBhMCVVMxFTATBgNV # BAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTExMC8G # A1UEAxMoRGlnaUNlcnQgU0hBMiBBc3N1cmVkIElEIENvZGUgU2lnbmluZyBDQQIQ # C4jZOitkx57ksuMgsWXX0jAJBgUrDgMCGgUAoHAwEAYKKwYBBAGCNwIBDDECMAAw # GQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwGCisG # AQQBgjcCARUwIwYJKoZIhvcNAQkEMRYEFNJie786rmpylLS9LPPKZN9rGISvMA0G # CSqGSIb3DQEBAQUABIIBAHaX9tEoWfULZ9wou8KPZm/oSxRHTGNSKmwEbtgIeRvz # oOcp+Dwd8TUqscnQF/JDu7ys1KLV1Outgkh5qbELdbd31gqbhVSIuiXyHCBOSQHA # yDmgCbVrcPbiOlh9nIYDgiLCgO0yovx4jgYsqGFOv1SvPtcxVa5CE+BCyR9nu/NY # 8OKm9M4idxb3X166wMIpFMNLBqKcU5asqRrwWXKAnT41SwOgkAKCt8T5skWFG0BD # RdyVIK1NA2x0Dv1+LUJ3o0NmJx5SuRQTKHMQrJJdHqmR0AEYkX4nqsd/fw0lwdpO # Y7lUgj7aRvKdueLT86vyly9/6kwX7FDXEasCSNYBsduhggMgMIIDHAYJKoZIhvcN # AQkGMYIDDTCCAwkCAQEwdzBjMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNl # cnQsIEluYy4xOzA5BgNVBAMTMkRpZ2lDZXJ0IFRydXN0ZWQgRzQgUlNBNDA5NiBT # SEEyNTYgVGltZVN0YW1waW5nIENBAhAMTWlyS5T6PCpKPSkHgD1aMA0GCWCGSAFl # AwQCAQUAoGkwGAYJKoZIhvcNAQkDMQsGCSqGSIb3DQEHATAcBgkqhkiG9w0BCQUx # DxcNMjMwMjAzMDExMzU0WjAvBgkqhkiG9w0BCQQxIgQg++9AySXVS5trIP+8xLH2 # kDxnreZ8bCAz8CoeFyzF4I0wDQYJKoZIhvcNAQEBBQAEggIAjmWy5j2GnJu5Zz/4 # 1hWPtQ7pPgK/EvREjZHeEiuJxz6w+yrKubHto9tQH5zlQ+yC7JP925OcivyzYJIL # AJ0RfR8BX7OeaFqhncxdmm8B/olmO70deRhV52GcXDLfDWniaQC1x+lXstrrHDPj # kfl5mBkNiBP4FKDE0SJk1Emazn+Atx+4yamVlGc8o/gxnW8yH6OwvlJ1SBmYQbvn # gJ5XpHxxuxvFkPAVTNENUHT90oidYLDeeFWjZQ2zm9sBiZQR6KtRc2z+SK47QKtT # LheYeCXMFLNOC5CRnavaE/RFIkdOaL3WBbCAKee3TspD6LxHRmL07lNmMK+/6UID # 5u2nac5LSjp9Jn8S/xS8PS4rfJ5ug9GyDrkwR+Sv4IRtHxlfmalDEzl+rBcnJ2Ka # 6JcQuF2DKEqHmaWGPutL9dgqZemTRXUmJ4NSz61QLLFWZOTFTXQQE+T32ygwF5+9 # HYiVJLZjGCpceEmFMMlKJJF43ofozGjCXGgdd/JWWchZosIpyq3wkQLB2gGXvkBS # V52LQ/hln9CZBeYm2VTbrhpMuvWUrvm8qlmqNB5admQ/V33W4lMrbU8DcLanYM9I # Adze1x+XivCKb+5IbiKX66s6vkM5LFFeIySGIO1d4xxNWwD1c7x4Al8MVuzpzN/8 # pkAv7vIPQI9quJh3j9jycqE8myU= # SIG # End signature block |