helpers/proxmox/Rename_and_update_devices_based_on_Proxmox_data.ps1
|
# Specify the hostname or IP and port of your Proxmox server $ProxmoxHost = "192.168.1.39" $ProxmoxPort = "8006" $ProxmoxUri = "https://${ProxmoxHost}:${ProxmoxPort}" # Specify the WhatsUp Gold IP Address or Hostname $WUGServer = "192.168.74.74" # Set your Proxmox Cred if (!$ProxmoxCred) { $ProxmoxCred = Get-Credential -Message "Enter credentials for Proxmox server" } $ProxmoxUsername = $ProxmoxCred.UserName $ProxmoxPassword = $ProxmoxCred.GetNetworkCredential().Password # Set your WhatsUp Gold Cred if (!$WUGCred) { $WUGCred = Get-Credential -Message "Enter credentials for WUG server" } # Check if the WhatsUpGoldPS module is loaded, and if not, import it if (-not (Get-Module -Name WhatsUpGoldPS)) { Import-Module WhatsUpGoldPS } # Load helper functions . "$PSScriptRoot\ProxmoxHelpers.ps1" # Ignore SSL cert validation (self-signed) Initialize-SSLBypass # Authenticate to Proxmox $cookie = Connect-ProxmoxServer -Server $ProxmoxUri -Username $ProxmoxUsername -Password $ProxmoxPassword # Connect to WUG Connect-WUGServer -serverUri $WUGServer -Credential $WUGCred -Protocol https -IgnoreSSLErrors # Get list of nodes and gather host + VM data $nodes = Get-ProxmoxNodes -Server $ProxmoxUri -Cookie $cookie $hostResults = foreach ($node in $nodes) { Get-ProxmoxNodeDetail -Server $ProxmoxUri -Cookie $cookie -Node $node.node } $vmResults = foreach ($node in $nodes) { $vms = Get-ProxmoxVMs -Server $ProxmoxUri -Cookie $cookie -Node $node.node foreach ($vm in $vms) { Get-ProxmoxVMDetail -Server $ProxmoxUri -Cookie $cookie -Node $node.node -VMID $vm.vmid } } # Output gathered data Write-Host "`n=== Proxmox Hosts ===" -ForegroundColor Cyan $hostResults | Format-Table NodeName, IPAddress, Status, CPUCores, RAM_Total, PVEVersion -AutoSize Write-Host "`n=== Proxmox VMs ===" -ForegroundColor Cyan $vmResults | Sort-Object Node, VMID | Format-Table Name, IPAddress, Status, Node, CPUs, RAM_Total -AutoSize # Update each Proxmox host in WUG Write-Host "`n=== Updating existing WUG devices (Hosts) ===" -ForegroundColor Cyan $updated = 0; $skipped = 0 foreach ($host_ in $hostResults) { if ($host_.IPAddress -notmatch '^\d{1,3}(\.\d{1,3}){3}$') { Write-Warning "Skipping host $($host_.NodeName) - no valid IPv4 address" $skipped++ continue } # Look up existing device in WUG by IP $device = Get-WUGDevice -SearchValue $host_.IPAddress -View id if (-not $device) { Write-Warning "Skipping $($host_.NodeName) ($($host_.IPAddress)) - not found in WUG" $skipped++ continue } $DeviceID = $device.id | Select-Object -First 1 Write-Host "Updating host $($host_.NodeName) ($($host_.IPAddress)) - Device ID: $DeviceID" -ForegroundColor Yellow # Update display name and note $note = "Proxmox sync $((Get-Date).ToString('yyyy-MM-dd HH:mm')) | " + "PVE $($host_.PVEVersion) | Kernel: $($host_.KernelVersion) | " + "CPU: $($host_.CPUModel) ($($host_.CPUSockets)S/$($host_.CPUCores)C/$($host_.CPUThreads)T) | " + "RAM: $($host_.RAM_Used) / $($host_.RAM_Total) | " + "RootFS: $($host_.RootFS_Used) / $($host_.RootFS_Total) | " + "Load: $($host_.LoadAvg1) $($host_.LoadAvg5) $($host_.LoadAvg15)" Set-WUGDeviceProperties -DeviceId $DeviceID -DisplayName $host_.NodeName -note $note # Update attributes $attributes = @{ "Proxmox_Type" = "Host" "Proxmox_NodeName" = "$($host_.NodeName)" "Proxmox_NodeID" = "$($host_.NodeID)" "Proxmox_Status" = "$($host_.Status)" "Proxmox_Uptime" = "$($host_.Uptime)" "Proxmox_PVEVersion" = "$($host_.PVEVersion)" "Proxmox_KernelVersion" = "$($host_.KernelVersion)" "Proxmox_CPUModel" = "$($host_.CPUModel)" "Proxmox_CPUSockets" = "$($host_.CPUSockets)" "Proxmox_CPUCores" = "$($host_.CPUCores)" "Proxmox_CPUThreads" = "$($host_.CPUThreads)" "Proxmox_CPUPercent" = "$($host_.CPUPercent)" "Proxmox_RAM_Used" = "$($host_.RAM_Used)" "Proxmox_RAM_Total" = "$($host_.RAM_Total)" "Proxmox_RAM_Free" = "$($host_.RAM_Free)" "Proxmox_Swap_Used" = "$($host_.Swap_Used)" "Proxmox_Swap_Total" = "$($host_.Swap_Total)" "Proxmox_RootFS_Used" = "$($host_.RootFS_Used)" "Proxmox_RootFS_Total" = "$($host_.RootFS_Total)" "Proxmox_RootFS_Free" = "$($host_.RootFS_Free)" "Proxmox_LoadAvg1" = "$($host_.LoadAvg1)" "Proxmox_LoadAvg5" = "$($host_.LoadAvg5)" "Proxmox_LoadAvg15" = "$($host_.LoadAvg15)" "Proxmox_LastSync" = "$((Get-Date).ToString('yyyy-MM-dd HH:mm:ss'))" } foreach ($attr in $attributes.GetEnumerator()) { if ($attr.Value -and $attr.Value -ne "" -and $attr.Value -ne "N/A") { Set-WUGDeviceAttribute -DeviceId $DeviceID -Name $attr.Key -Value $attr.Value } } Write-Host " Updated with $($attributes.Count) attributes" -ForegroundColor Green $updated++ } # Update each Proxmox VM in WUG Write-Host "`n=== Updating existing WUG devices (VMs) ===" -ForegroundColor Cyan foreach ($vm in $vmResults) { if ($vm.IPAddress -notmatch '^\d{1,3}(\.\d{1,3}){3}$') { Write-Warning "Skipping VM $($vm.Name) - no valid IPv4 address" $skipped++ continue } # Look up existing device in WUG by IP $device = Get-WUGDevice -SearchValue $vm.IPAddress -View id if (-not $device) { Write-Warning "Skipping $($vm.Name) ($($vm.IPAddress)) - not found in WUG" $skipped++ continue } $DeviceID = $device.id | Select-Object -First 1 Write-Host "Updating VM $($vm.Name) ($($vm.IPAddress)) - Device ID: $DeviceID" -ForegroundColor Yellow # Update display name and note $note = "Proxmox sync $((Get-Date).ToString('yyyy-MM-dd HH:mm')) | " + "VMID: $($vm.VMID) | Node: $($vm.Node) | Status: $($vm.Status) | " + "CPU: $($vm.CPUs) vCPUs ($($vm.CPUPercent)) | " + "RAM: $($vm.RAM_Used) / $($vm.RAM_Total) | " + "Disk: $($vm.Disk_Used) / $($vm.Disk_Total) | " + "Net In: $($vm.NetIn_KB), Out: $($vm.NetOut_KB)" Set-WUGDeviceProperties -DeviceId $DeviceID -DisplayName $vm.Name -note $note # Update attributes $attributes = @{ "Proxmox_Type" = "VM" "Proxmox_VMID" = "$($vm.VMID)" "Proxmox_Name" = "$($vm.Name)" "Proxmox_Node" = "$($vm.Node)" "Proxmox_Status" = "$($vm.Status)" "Proxmox_QMPStatus" = "$($vm.QMPStatus)" "Proxmox_Uptime" = "$($vm.Uptime)" "Proxmox_CPUPercent" = "$($vm.CPUPercent)" "Proxmox_CPUs" = "$($vm.CPUs)" "Proxmox_RAM_Used" = "$($vm.RAM_Used)" "Proxmox_RAM_Total" = "$($vm.RAM_Total)" "Proxmox_Disk_Used" = "$($vm.Disk_Used)" "Proxmox_Disk_Total" = "$($vm.Disk_Total)" "Proxmox_Disk_Read" = "$($vm.Disk_Read)" "Proxmox_Disk_Write" = "$($vm.Disk_Write)" "Proxmox_NetIn_KB" = "$($vm.NetIn_KB)" "Proxmox_NetOut_KB" = "$($vm.NetOut_KB)" "Proxmox_Tags" = "$($vm.Tags)" "Proxmox_HAGroup" = "$($vm.HAGroup)" "Proxmox_HAState" = "$($vm.HAState)" "Proxmox_HAManaged" = "$($vm.HAManaged)" "Proxmox_LastSync" = "$((Get-Date).ToString('yyyy-MM-dd HH:mm:ss'))" } foreach ($attr in $attributes.GetEnumerator()) { if ($attr.Value -and $attr.Value -ne "" -and $attr.Value -ne "N/A") { Set-WUGDeviceAttribute -DeviceId $DeviceID -Name $attr.Key -Value $attr.Value } } Write-Host " Updated with $($attributes.Count) attributes" -ForegroundColor Green $updated++ } Write-Host "`n=== Summary ===" -ForegroundColor Cyan Write-Host " Updated: $updated devices" Write-Host " Skipped: $skipped devices" # Cleanup if ($Global:WUGConnection) { Disconnect-WUGServer } # SIG # Begin signature block # MIIVlwYJKoZIhvcNAQcCoIIViDCCFYQCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCDZyOcUqVtdVtbX # PILYQvT/mYXGy7nwm1L3IswBVn3yFaCCEdMwggVvMIIEV6ADAgECAhBI/JO0YFWU # jTanyYqJ1pQWMA0GCSqGSIb3DQEBDAUAMHsxCzAJBgNVBAYTAkdCMRswGQYDVQQI # DBJHcmVhdGVyIE1hbmNoZXN0ZXIxEDAOBgNVBAcMB1NhbGZvcmQxGjAYBgNVBAoM # EUNvbW9kbyBDQSBMaW1pdGVkMSEwHwYDVQQDDBhBQUEgQ2VydGlmaWNhdGUgU2Vy # dmljZXMwHhcNMjEwNTI1MDAwMDAwWhcNMjgxMjMxMjM1OTU5WjBWMQswCQYDVQQG # EwJHQjEYMBYGA1UEChMPU2VjdGlnbyBMaW1pdGVkMS0wKwYDVQQDEyRTZWN0aWdv # IFB1YmxpYyBDb2RlIFNpZ25pbmcgUm9vdCBSNDYwggIiMA0GCSqGSIb3DQEBAQUA # A4ICDwAwggIKAoICAQCN55QSIgQkdC7/FiMCkoq2rjaFrEfUI5ErPtx94jGgUW+s # hJHjUoq14pbe0IdjJImK/+8Skzt9u7aKvb0Ffyeba2XTpQxpsbxJOZrxbW6q5KCD # J9qaDStQ6Utbs7hkNqR+Sj2pcaths3OzPAsM79szV+W+NDfjlxtd/R8SPYIDdub7 # P2bSlDFp+m2zNKzBenjcklDyZMeqLQSrw2rq4C+np9xu1+j/2iGrQL+57g2extme # me/G3h+pDHazJyCh1rr9gOcB0u/rgimVcI3/uxXP/tEPNqIuTzKQdEZrRzUTdwUz # T2MuuC3hv2WnBGsY2HH6zAjybYmZELGt2z4s5KoYsMYHAXVn3m3pY2MeNn9pib6q # RT5uWl+PoVvLnTCGMOgDs0DGDQ84zWeoU4j6uDBl+m/H5x2xg3RpPqzEaDux5mcz # mrYI4IAFSEDu9oJkRqj1c7AGlfJsZZ+/VVscnFcax3hGfHCqlBuCF6yH6bbJDoEc # QNYWFyn8XJwYK+pF9e+91WdPKF4F7pBMeufG9ND8+s0+MkYTIDaKBOq3qgdGnA2T # OglmmVhcKaO5DKYwODzQRjY1fJy67sPV+Qp2+n4FG0DKkjXp1XrRtX8ArqmQqsV/ # AZwQsRb8zG4Y3G9i/qZQp7h7uJ0VP/4gDHXIIloTlRmQAOka1cKG8eOO7F/05QID # AQABo4IBEjCCAQ4wHwYDVR0jBBgwFoAUoBEKIz6W8Qfs4q8p74Klf9AwpLQwHQYD # VR0OBBYEFDLrkpr/NZZILyhAQnAgNpFcF4XmMA4GA1UdDwEB/wQEAwIBhjAPBgNV # HRMBAf8EBTADAQH/MBMGA1UdJQQMMAoGCCsGAQUFBwMDMBsGA1UdIAQUMBIwBgYE # VR0gADAIBgZngQwBBAEwQwYDVR0fBDwwOjA4oDagNIYyaHR0cDovL2NybC5jb21v # ZG9jYS5jb20vQUFBQ2VydGlmaWNhdGVTZXJ2aWNlcy5jcmwwNAYIKwYBBQUHAQEE # KDAmMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5jb21vZG9jYS5jb20wDQYJKoZI # hvcNAQEMBQADggEBABK/oe+LdJqYRLhpRrWrJAoMpIpnuDqBv0WKfVIHqI0fTiGF # OaNrXi0ghr8QuK55O1PNtPvYRL4G2VxjZ9RAFodEhnIq1jIV9RKDwvnhXRFAZ/ZC # J3LFI+ICOBpMIOLbAffNRk8monxmwFE2tokCVMf8WPtsAO7+mKYulaEMUykfb9gZ # pk+e96wJ6l2CxouvgKe9gUhShDHaMuwV5KZMPWw5c9QLhTkg4IUaaOGnSDip0TYl # d8GNGRbFiExmfS9jzpjoad+sPKhdnckcW67Y8y90z7h+9teDnRGWYpquRRPaf9xH # +9/DUp/mBlXpnYzyOmJRvOwkDynUWICE5EV7WtgwggYaMIIEAqADAgECAhBiHW0M # UgGeO5B5FSCJIRwKMA0GCSqGSIb3DQEBDAUAMFYxCzAJBgNVBAYTAkdCMRgwFgYD # VQQKEw9TZWN0aWdvIExpbWl0ZWQxLTArBgNVBAMTJFNlY3RpZ28gUHVibGljIENv # ZGUgU2lnbmluZyBSb290IFI0NjAeFw0yMTAzMjIwMDAwMDBaFw0zNjAzMjEyMzU5 # NTlaMFQxCzAJBgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0ZWQxKzAp # BgNVBAMTIlNlY3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYwggGiMA0G # CSqGSIb3DQEBAQUAA4IBjwAwggGKAoIBgQCbK51T+jU/jmAGQ2rAz/V/9shTUxjI # ztNsfvxYB5UXeWUzCxEeAEZGbEN4QMgCsJLZUKhWThj/yPqy0iSZhXkZ6Pg2A2NV # DgFigOMYzB2OKhdqfWGVoYW3haT29PSTahYkwmMv0b/83nbeECbiMXhSOtbam+/3 # 6F09fy1tsB8je/RV0mIk8XL/tfCK6cPuYHE215wzrK0h1SWHTxPbPuYkRdkP05Zw # mRmTnAO5/arnY83jeNzhP06ShdnRqtZlV59+8yv+KIhE5ILMqgOZYAENHNX9SJDm # +qxp4VqpB3MV/h53yl41aHU5pledi9lCBbH9JeIkNFICiVHNkRmq4TpxtwfvjsUe # dyz8rNyfQJy/aOs5b4s+ac7IH60B+Ja7TVM+EKv1WuTGwcLmoU3FpOFMbmPj8pz4 # 4MPZ1f9+YEQIQty/NQd/2yGgW+ufflcZ/ZE9o1M7a5Jnqf2i2/uMSWymR8r2oQBM # dlyh2n5HirY4jKnFH/9gRvd+QOfdRrJZb1sCAwEAAaOCAWQwggFgMB8GA1UdIwQY # MBaAFDLrkpr/NZZILyhAQnAgNpFcF4XmMB0GA1UdDgQWBBQPKssghyi47G9IritU # pimqF6TNDDAOBgNVHQ8BAf8EBAMCAYYwEgYDVR0TAQH/BAgwBgEB/wIBADATBgNV # HSUEDDAKBggrBgEFBQcDAzAbBgNVHSAEFDASMAYGBFUdIAAwCAYGZ4EMAQQBMEsG # A1UdHwREMEIwQKA+oDyGOmh0dHA6Ly9jcmwuc2VjdGlnby5jb20vU2VjdGlnb1B1 # YmxpY0NvZGVTaWduaW5nUm9vdFI0Ni5jcmwwewYIKwYBBQUHAQEEbzBtMEYGCCsG # AQUFBzAChjpodHRwOi8vY3J0LnNlY3RpZ28uY29tL1NlY3RpZ29QdWJsaWNDb2Rl # U2lnbmluZ1Jvb3RSNDYucDdjMCMGCCsGAQUFBzABhhdodHRwOi8vb2NzcC5zZWN0 # aWdvLmNvbTANBgkqhkiG9w0BAQwFAAOCAgEABv+C4XdjNm57oRUgmxP/BP6YdURh # w1aVcdGRP4Wh60BAscjW4HL9hcpkOTz5jUug2oeunbYAowbFC2AKK+cMcXIBD0Zd # OaWTsyNyBBsMLHqafvIhrCymlaS98+QpoBCyKppP0OcxYEdU0hpsaqBBIZOtBajj # cw5+w/KeFvPYfLF/ldYpmlG+vd0xqlqd099iChnyIMvY5HexjO2AmtsbpVn0OhNc # WbWDRF/3sBp6fWXhz7DcML4iTAWS+MVXeNLj1lJziVKEoroGs9Mlizg0bUMbOalO # hOfCipnx8CaLZeVme5yELg09Jlo8BMe80jO37PU8ejfkP9/uPak7VLwELKxAMcJs # zkyeiaerlphwoKx1uHRzNyE6bxuSKcutisqmKL5OTunAvtONEoteSiabkPVSZ2z7 # 6mKnzAfZxCl/3dq3dUNw4rg3sTCggkHSRqTqlLMS7gjrhTqBmzu1L90Y1KWN/Y5J # KdGvspbOrTfOXyXvmPL6E52z1NZJ6ctuMFBQZH3pwWvqURR8AgQdULUvrxjUYbHH # j95Ejza63zdrEcxWLDX6xWls/GDnVNueKjWUH3fTv1Y8Wdho698YADR7TNx8X8z2 # Bev6SivBBOHY+uqiirZtg0y9ShQoPzmCcn63Syatatvx157YK9hlcPmVoa1oDE5/ # L9Uo2bC5a4CH2RwwggY+MIIEpqADAgECAhAHnODk0RR/hc05c892LTfrMA0GCSqG # SIb3DQEBDAUAMFQxCzAJBgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0 # ZWQxKzApBgNVBAMTIlNlY3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYw # HhcNMjYwMjA5MDAwMDAwWhcNMjkwNDIxMjM1OTU5WjBVMQswCQYDVQQGEwJVUzEU # MBIGA1UECAwLQ29ubmVjdGljdXQxFzAVBgNVBAoMDkphc29uIEFsYmVyaW5vMRcw # FQYDVQQDDA5KYXNvbiBBbGJlcmlubzCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCC # AgoCggIBAPN6aN4B1yYWkI5b5TBj3I0VV/peETrHb6EY4BHGxt8Ap+eT+WpEpJyE # tRYPxEmNJL3A38Bkg7mwzPE3/1NK570ZBCuBjSAn4mSDIgIuXZnvyBO9W1OQs5d6 # 7MlJLUAEufl18tOr3ST1DeO9gSjQSAE5Nql0QDxPnm93OZBon+Fz3CmE+z3MwAe2 # h4KdtRAnCqwM+/V7iBdbw+JOxolpx+7RVjGyProTENIG3pe/hKvPb501lf8uBAAD # LdjZr5ip8vIWbf857Yw1Bu10nVI7HW3eE8Cl5//d1ribHlzTzQLfttW+k+DaFsKZ # BBL56l4YAlIVRsrOiE1kdHYYx6IGrEA809R7+TZA9DzGqyFiv9qmJAbL4fDwetDe # yIq+Oztz1LvEdy8Rcd0JBY+J4S0eDEFIA3X0N8VcLeAwabKb9AjulKXwUeqCJLvN # 79CJ90UTZb2+I+tamj0dn+IKMEsJ4v4Ggx72sxFr9+6XziodtTg5Luf2xd6+Phha # mOxF2px9LObhBLLEMyRsCHZIzVZOFKu9BpHQH7ufGB+Sa80Tli0/6LEyn9+bMYWi # 2ttn6lLOPThXMiQaooRUq6q2u3+F4SaPlxVFLI7OJVMhar6nW6joBvELTJPmANSM # jDSRFDfHRCdGbZsL/keELJNy+jZctF6VvxQEjFM8/bazu6qYhrA7AgMBAAGjggGJ # MIIBhTAfBgNVHSMEGDAWgBQPKssghyi47G9IritUpimqF6TNDDAdBgNVHQ4EFgQU # 6YF0o0D5AVhKHbVocr8GaSIBibAwDgYDVR0PAQH/BAQDAgeAMAwGA1UdEwEB/wQC # MAAwEwYDVR0lBAwwCgYIKwYBBQUHAwMwSgYDVR0gBEMwQTA1BgwrBgEEAbIxAQIB # AwIwJTAjBggrBgEFBQcCARYXaHR0cHM6Ly9zZWN0aWdvLmNvbS9DUFMwCAYGZ4EM # AQQBMEkGA1UdHwRCMEAwPqA8oDqGOGh0dHA6Ly9jcmwuc2VjdGlnby5jb20vU2Vj # dGlnb1B1YmxpY0NvZGVTaWduaW5nQ0FSMzYuY3JsMHkGCCsGAQUFBwEBBG0wazBE # BggrBgEFBQcwAoY4aHR0cDovL2NydC5zZWN0aWdvLmNvbS9TZWN0aWdvUHVibGlj # Q29kZVNpZ25pbmdDQVIzNi5jcnQwIwYIKwYBBQUHMAGGF2h0dHA6Ly9vY3NwLnNl # Y3RpZ28uY29tMA0GCSqGSIb3DQEBDAUAA4IBgQAEIsm4xnOd/tZMVrKwi3doAXvC # wOA/RYQnFJD7R/bSQRu3wXEK4o9SIefye18B/q4fhBkhNAJuEvTQAGfqbbpxow03 # J5PrDTp1WPCWbXKX8Oz9vGWJFyJxRGftkdzZ57JE00synEMS8XCwLO9P32MyR9Z9 # URrpiLPJ9rQjfHMb1BUdvaNayomm7aWLAnD+X7jm6o8sNT5An1cwEAob7obWDM6s # X93wphwJNBJAstH9Ozs6LwISOX6sKS7CKm9N3Kp8hOUue0ZHAtZdFl6o5u12wy+z # zieGEI50fKnN77FfNKFOWKlS6OJwlArcbFegB5K89LcE5iNSmaM3VMB2ADV1FEcj # GSHw4lTg1Wx+WMAMdl/7nbvfFxJ9uu5tNiT54B0s+lZO/HztwXYQUczdsFon3pjs # Nrsk9ZlalBi5SHkIu+F6g7tWiEv3rtVApmJRnLkUr2Xq2a4nbslUCt4jKs5UX4V1 # nSX8OM++AXoyVGO+iTj7z+pl6XE9Gw/Td6WKKKsxggMaMIIDFgIBATBoMFQxCzAJ # BgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0ZWQxKzApBgNVBAMTIlNl # Y3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYCEAec4OTRFH+FzTlzz3Yt # N+swDQYJYIZIAWUDBAIBBQCggYQwGAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZ # BgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYB # BAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgtmcu5Ln+k0zATECDlrMn5l6FFYoxgqNs # 4ZzeKu5rCaEwDQYJKoZIhvcNAQEBBQAEggIA8hnwbEtSDEafYYngxp++rTLnC+R4 # C1W5kuIU2lxWiSGz/fJ/b9UCyM1XxJKzzJLjxMWCyV3bZaXXxABm5BiqvjUCY2jA # nIob2Pt184DpY/1i8he++HBtp0ke3a7zph+yQhtDAC9sWEHGwipvI9YEpLFBXt/A # xuaSKFI20BwqVJnKC3yoMJ6zjLU1jo4GHhMsr54vqXGRc2IR0njE65cXLEDR76MW # VwMIdaY4hqhOfRuK3fF8FpjeSbnPKaypPWF2PVRSweKCLlDZkLynYEoG1lMKStIZ # yOCcO+wHEXVZm/eJEsymHCh7xCx0I0OYxsFumEpk5uTUVjM9I7r8Y+1YizVFav0R # uXe8Z63UAA8wFmrskaUyC/+C7wxzZq3woyFMK0vUhomTkpDeEWErH5WDmG0Y6tff # 9UWLIcnVNPp2iKxO6iklDclRR2Ke4AQoICqSe4i+Tk4icgDGlwnWBklDkzcJormP # Sqpl0DmlePO2WrgevzbBjGlVBaCvqa3ardZzf7OpWSle3FySUquLuXwSoiR6BtxP # vzC+SUr9W2GHpBGEcnP/7hTKoTz6fgTyoeiiMDCE8kUw7qAliEZgUzKzlzGiPtam # pJfWzKP5AGM/7rn874DWnY1FxfEFNQBe6VtSmNgnvaUQcrTX8P2wssIdXfwhXsX9 # jEtjKR8asGd6UFc= # SIG # End signature block |