Private/Shared/Write-ToConsoleLog.ps1
|
function Write-ToConsoleLog { <# .SYNOPSIS Writes formatted log messages to the console with timestamps and log levels. .DESCRIPTION This function provides consistent console logging with timestamps, log levels, and color coding. It supports error, warning, success, and plan modes with appropriate coloring. Can also write to a file when in plan mode. .PARAMETER Messages One or more messages to write to the console. .PARAMETER Level The log level (INFO, ERROR, WARNING, SUCCESS, PLAN). Defaults to INFO. .PARAMETER Color The console color to use. Defaults to Blue for INFO, or determined by Level. .PARAMETER NewLine Adds the newline prefix before the message. .PARAMETER Overwrite Uses carriage return to overwrite the current line (for progress indicators). .PARAMETER IsError Sets the level to ERROR and uses red coloring. .PARAMETER IsWarning Sets the level to WARNING and uses yellow coloring. .PARAMETER IsSuccess Sets the level to SUCCESS and uses green coloring. .PARAMETER IsPlan Sets the level to PLAN and uses gray coloring. Also enables file writing. .PARAMETER WriteToFile Enables writing the message to a log file. .PARAMETER LogFilePath The path to the log file when WriteToFile is enabled. .EXAMPLE Write-ToConsoleLog "Starting process..." .EXAMPLE Write-ToConsoleLog "Operation completed successfully" -IsSuccess .EXAMPLE Write-ToConsoleLog "Something went wrong" -IsError #> [CmdletBinding()] param ( [Parameter(Mandatory = $true, Position = 0)] [string[]]$Messages, [Parameter(Mandatory = $false)] [string]$Level = "INFO", [Parameter(Mandatory = $false)] [System.ConsoleColor]$Color = [System.ConsoleColor]::White, [Parameter(Mandatory = $false)] [switch]$NewLine, [Parameter(Mandatory = $false)] [switch]$Overwrite, [Parameter(Mandatory = $false)] [switch]$IsError, [Parameter(Mandatory = $false)] [switch]$IsWarning, [Parameter(Mandatory = $false)] [switch]$IsSuccess, [Parameter(Mandatory = $false)] [switch]$IsPlan, [Parameter(Mandatory = $false)] [switch]$IsPrompt, [Parameter(Mandatory = $false)] [switch]$IsSelection, [Parameter(Mandatory = $false)] [switch]$WriteToFile, [Parameter(Mandatory = $false)] [string]$LogFilePath = $null, [Parameter(Mandatory = $false)] [switch]$ShowDateTime, [Parameter(Mandatory = $false)] [switch]$ShowType, [Parameter(Mandatory = $false)] [string]$IndentTemplate = " ", [Parameter(Mandatory = $false)] [int]$IndentLevel = 0, [Parameter(Mandatory = $false)] [array]$Defaults = @( @{ Level = "INFO" Color = [System.ConsoleColor]::Blue NewLine = $false ShowDateTime = $true ShowType = $true WriteToFile = $false }, @{ Level = "ERROR" Color = [System.ConsoleColor]::Red NewLine = $true ShowDateTime = $true ShowType = $true WriteToFile = $false }, @{ Level = "WARNING" Color = [System.ConsoleColor]::Yellow NewLine = $true ShowDateTime = $true ShowType = $true WriteToFile = $false }, @{ Level = "SUCCESS" Color = [System.ConsoleColor]::Green NewLine = $true ShowDateTime = $true ShowType = $true WriteToFile = $false }, @{ Level = "PLAN" Color = [System.ConsoleColor]::Gray NewLine = $false ShowDateTime = $true ShowType = $true WriteToFile = $true }, @{ Level = "INPUT REQUIRED" Color = [System.ConsoleColor]::Magenta NewLine = $true ShowDateTime = $true ShowType = $true WriteToFile = $false }, @{ Level = "SELECTION" Color = [System.ConsoleColor]::White NewLine = $false ShowDateTime = $false ShowType = $false WriteToFile = $false } ) ) if ($IsError) { $Level = "ERROR" } elseif ($IsWarning) { $Level = "WARNING" } elseif ($IsSuccess) { $Level = "SUCCESS" } elseif ($IsPlan) { $Level = "PLAN" } elseif ($IsPrompt) { $Level = "INPUT REQUIRED" } elseif ($IsSelection) { $Level = "SELECTION" } $defaultSettings = $Defaults | Where-Object { $_.Level -eq $Level } | Select-Object -First 1 $timestamp = Get-Date -Format "yyyy-MM-dd HH:mm:ss.fff" if ($Color -eq [System.ConsoleColor]::White) { if ($defaultSettings) { $Color = $defaultSettings.Color } } $prefix = "" if ($Overwrite) { $prefix = "`r" } else { if ($AddNewLine -or ($defaultSettings -and $defaultSettings.NewLine)) { $prefix = [System.Environment]::NewLine } } if ($ShowDateTime -or ($defaultSettings -and $defaultSettings.ShowDateTime)) { $prefix += "[$timestamp] " } if ($ShowType -or ($defaultSettings -and $defaultSettings.ShowType)) { $prefix += "[$Level] " } if ($IndentLevel -gt 0) { $indentString = $IndentTemplate * $IndentLevel $prefix = $indentString + $prefix } $finalMessages = @() foreach ($Message in $Messages) { $finalMessages += "$prefix$Message" } if ($finalMessages.Count -gt 1) { $finalMessages = $finalMessages -join "`n" } Write-Host $finalMessages -ForegroundColor $Color -NoNewline:$Overwrite.IsPresent if (($WriteToFile -or ($defaultSettings -and $defaultSettings.WriteToFile)) -and $LogFilePath) { Add-Content -Path $LogFilePath -Value $finalMessages } } # SIG # Begin signature block # MIInSgYJKoZIhvcNAQcCoIInOzCCJzcCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCDSJycjxJQKDglq # 6qk69DTUVlm7IRLPuW4SVUtAb8jxwKCCDLowggX1MIID3aADAgECAhMzAAACHU0Z # yE7XD1dIAAAAAAIdMA0GCSqGSIb3DQEBCwUAMFcxCzAJBgNVBAYTAlVTMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBD # b2RlIFNpZ25pbmcgUENBIDIwMjQwHhcNMjYwNDE2MTg1OTQzWhcNMjcwNDE1MTg1 # OTQzWjB0MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYD # VQQDExVNaWNyb3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IB # DwAwggEKAoIBAQDQvewXxx9gZZFC6Ys1WBay8BJ8kGA4JQnH5CMafqOASlTpK9H8 # o5ZXTXt0caVQTNMUPt445wXYD+dFtaKWTwDn1I52oUSrC9vJin1Gsqt+zyKJL5Dg # 3eQXbQNR61DmMy20GLTIO3SFed9Rfi/ophgCLGFLDR3r0KvHjwMb/jYWS0celV/4 # Lz27LfAekm8v9E5IXaeiXbAUYZKK090n4CVl3JBtbN+9DtI9SNu/yjvozW52/u7R # X/Ttpa/KDlpuokZ+Zcbvmtd9ur9gFLvZzh41o9MsE/clQtdaFWGvuo6Jua/ntpgk # ey3E5/vBFe+MJPG6phdnuo6r57ZudCudiI1bAgMBAAGjggGbMIIBlzAOBgNVHQ8B # Af8EBAMCB4AwHwYDVR0lBBgwFgYKKwYBBAGCN0wIAQYIKwYBBQUHAwMwHQYDVR0O # BBYEFH6QuMwqcPG0hQlQ6c5jCtTTLrVeMEUGA1UdEQQ+MDykOjA4MR4wHAYDVQQL # ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xFjAUBgNVBAUTDTIzMDAxMis1MDc1NTkw # HwYDVR0jBBgwFoAUf1k/VCHarU/vBeXmo9ctBpQSCDEwYAYDVR0fBFkwVzBVoFOg # UYZPaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9jcmwvTWljcm9zb2Z0 # JTIwQ29kZSUyMFNpZ25pbmclMjBQQ0ElMjAyMDI0LmNybDBtBggrBgEFBQcBAQRh # MF8wXQYIKwYBBQUHMAKGUWh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMv # Y2VydHMvTWljcm9zb2Z0JTIwQ29kZSUyMFNpZ25pbmclMjBQQ0ElMjAyMDI0LmNy # dDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4ICAQBKTbYOjzwTG/DXGaz9 # s6+fQeaTtDcFmMY+5UyVFCyj7Pv+5i37qfX8lSL/tBIfYQfWsMuBQlfZurJD6r4H # VJ2CeH+1fgiq8dcHdVKoZ3Sa2qXoX3cq9iS8cVb06B7+5/XJ7I0OxHH9fDsvJ3T3 # w5V/ZtAIFmLrl+P0CtG+92uzRsn0nTbdFjOkLMLWPLAU3THohKRlSEMgFJpPkm5n # 5UAZ35xX6FWCrDLsSKb555bTifwa8mJBwdlof0bmfYidH+dxZ1FdDxvLnNl9zeKs # A4kejaaIqqIPguhwAti5Ql7BlTNoJNwxCvBmqW2MQLnCkYN/VVUsR3V2x/rcTNzo # Bf/Z/SpROvdaA2ZOOd1uioXJt3tdLQ7vHpqpib0KfWr/FWXW10q38VxfCnRQBqzb # SuztR7nEMuzX7Ck+B/XaPDXd1qh72+QYyB0Z2VzWmO9zsnb9Uq/dwu8LGeQqnyu6 # 7SDGACvnXii2fb9+US492VTnXSnFKyqwgzUyFMtZK1/sHYTv6bG4TtQUygQxTN+Z # V+aJIlKO2MqZ7bKrAnOzS9m6NgoTdWOq11bTOZwKlIEV/EhV9SWkDmdpR/hPPT2v # 6TEj4F8PT/zHjRezIU5c/DGlt/VhY/pK0XkJtEyMmmS1BMtjU/rqBZVMIm3dnxQs # /TBByr+Cf8Z1r7aifQVQ+WSqzjCCBr0wggSloAMCAQICEzMAAAA5O7Y3Gb8GHWcA # AAAAADkwDQYJKoZIhvcNAQEMBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpX # YXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQg # Q29ycG9yYXRpb24xMjAwBgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmljYXRl # IEF1dGhvcml0eSAyMDExMB4XDTI0MDgwODIwNTQxOFoXDTM2MDMyMjIyMTMwNFow # VzELMAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEo # MCYGA1UEAxMfTWljcm9zb2Z0IENvZGUgU2lnbmluZyBQQ0EgMjAyNDCCAiIwDQYJ # KoZIhvcNAQEBBQADggIPADCCAgoCggIBANgBnB7jOMeqlRYHNa265v4IY9fH8TKh # emHfPINe1gpLaV3dhg324WwH06LcHbpnsBukCDNitryo0dtS/EW6I/yEL/bLSY8h # KpbfQuWusBPr9qazYcDxCW/qnjb5JsI1s8bNOg3bVATvQVL4tcf03aTycsz8QeCd # M0l/yHRObJ9QqazM1r6VPEOJ7LL+uEEb73w6QCuhs89a1uv1zerOYMnsneRRwCbp # yW11IcggU0cRKDDq1pjVJzIbIF6+oiXXbReOsgeI8zu1FyQfK0fVkaya8SmVHQ/t # Of23mZ4W9k0Ri22QW9p3UgSC5OUDktKxxcCmGL6tXLfOGSWHIIV4YrTJTT6PNty5 # REojHJuZHArkF9VnHTERWoTjAzfI3kP+5b4alUdhgAZ7ttOu1bVnXfHaqPYl2rPs # 20ji03LOVWsh/radgE17es5hL+t6lV0eVHrVhsssROWJuz2MXMCt7iw7lFPG9LXK # Gjsmonn2gotGdHIuEg5JnJMJVmixd5LRlkmgYRZKzhxSCwyoGIq0PhaA7Y+VPct5 # pCHkijcIIDm0nlkK+0KyepolcqGm0T/GYQRMhHJlGOOmVQop36wUVUYklUy++vDW # eEgEo4s7hxN6mIbf2MSIQ/iIfMZgJxC69oukMUXCrOC3SkE/xIkgpfl22MM1itkZ # 35nNXkMolU1lAgMBAAGjggFOMIIBSjAOBgNVHQ8BAf8EBAMCAYYwEAYJKwYBBAGC # NxUBBAMCAQAwHQYDVR0OBBYEFH9ZP1Qh2q1P7wXl5qPXLQaUEggxMBkGCSsGAQQB # gjcUAgQMHgoAUwB1AGIAQwBBMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU # ci06AjGQQ7kUBU7h6qfHMdEjiTQwWgYDVR0fBFMwUTBPoE2gS4ZJaHR0cDovL2Ny # bC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0MjAx # MV8yMDExXzAzXzIyLmNybDBeBggrBgEFBQcBAQRSMFAwTgYIKwYBBQUHMAKGQmh0 # dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2VydHMvTWljUm9vQ2VyQXV0MjAx # MV8yMDExXzAzXzIyLmNydDANBgkqhkiG9w0BAQwFAAOCAgEAFJQfOChP7onn6fLI # MKrSlN1WYKwDFgAddymOUO3FrM8d7B/W/iQ6DxXsDn7D5W4wMwYeLystcEqfkjz4 # NURRgazyMu5yRzQh4LqjA4tStTcJh1opExo7nn5PuPBYnbu0+THSuVHTe0VTTPVh # ily/piFrDo3axQ9P4C+Ol5yet+2gTfekICS5xS+cYfSIvgn0JksVBVMYVI5QFu/q # hnLhsEFEUzG8fvv0hjgkO+lkpV9ty6GkN4vdnd7ya6Q6aR9y34aiM1qmxaxBi6OU # nyNl6fkuun/diTFnYDLTppOkr/mg5WSfCiDVMNCxtj4wPKC5OmHm1DQIt/MNokbb # H3UGsFP1QbzsLocuSqLCvH09Io3fDPTmscR9Y75G4qX7RTX8AdBPo0I6OEojf39z # uFZt0qOHm65YWQE69cZM2ueE1MB05dNNgHK9gTE7zKvK/fg8B2qjW88MT/WF5V5u # vZGtqa9FSL2RazArA+rDPuf6JGYz4HpgMZHB4S6szWSKYBv0VisCzfxgeU+dquXW # 9bd0auYlOB58DPcOYKdc3Se94g+xL4pcEhbB54JOgAkwYTu/9dLeH2pDqeJZAABV # DWRQCaXfO5LgyKwKCLYXpigrZYCjUSBcr+Ve8PFWMhVTQl0v4q8J/AUmQN5W4n10 # 1cY2L4A7GTQG1h32HHAvfQESWP0xghnmMIIZ4gIBATBuMFcxCzAJBgNVBAYTAlVT # MR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jv # c29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMjQCEzMAAAIdTRnITtcPV0gAAAAAAh0w # DQYJYIZIAWUDBAIBBQCggbAwGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYK # KwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIMAVdUOn # zzqnjSwqk26EJ6XZT2KmMKP3aLBO5CMgIfgYMEQGCisGAQQBgjcCAQwxNjA0oBSA # EgBNAGkAYwByAG8AcwBvAGYAdKEcgBpodHRwczovL3d3dy5taWNyb3NvZnQuY29t # IDANBgkqhkiG9w0BAQEFAASCAQC75bP9uaCLgch4uWaCY5TOY+oULkXGfFTJLJ62 # /N7RutuvZ3CZyqxvJb8a/C1nUJH3V5O/R3QyKBRdGJmgvTqAlCNwUQjPBwGN5VSu # lOtFGE2lS+xDrb+ejbpwenmQ6mUYtK1KpeZIoIBPJemHao5xuSXOZUJBWhZLf4if # o1252FBX87gP+bbRcSAP1Y7IWt4sKG6KFZytf/KSDXp3Zlir+tLbZBvCjtuo3iZd # vJZ6aucsetmzEFhH3g8KHRT1+p/dX0ZD0+5nIVJ+wzQK/jnNNUSnBqsIwQrnYk+W # HaF4KE/5U7y6WHvtVojUgN1x6ekO3Pys2VmMYkH6o3kCpi0coYIXljCCF5IGCisG # AQQBgjcDAwExgheCMIIXfgYJKoZIhvcNAQcCoIIXbzCCF2sCAQMxDzANBglghkgB # ZQMEAgEFADCCAVEGCyqGSIb3DQEJEAEEoIIBQASCATwwggE4AgEBBgorBgEEAYRZ # CgMBMDEwDQYJYIZIAWUDBAIBBQAEIOvDt6/ARc5cMWg7TU5ifdtn7GI5JyFj1Akv # GPB9mOfPAgZqMbX4bjkYEjIwMjYwNjI1MDkwMjQ4LjU3WjAEgAIB9KCB0aSBzjCB # yzELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjElMCMGA1UECxMc # TWljcm9zb2Z0IEFtZXJpY2EgT3BlcmF0aW9uczEnMCUGA1UECxMeblNoaWVsZCBU # U1MgRVNOOjk2MDAtMDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1T # dGFtcCBTZXJ2aWNloIIR7TCCByAwggUIoAMCAQICEzMAAAImNbQ+Z0OT9h8AAQAA # AiYwDQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hp # bmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw # b3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAw # HhcNMjYwMjE5MTk0MDAyWhcNMjcwNTE3MTk0MDAyWjCByzELMAkGA1UEBhMCVVMx # EzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoT # FU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJp # Y2EgT3BlcmF0aW9uczEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjk2MDAtMDVF # MC1EOTQ3MSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIIC # IjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAv/8PmWSC+URRaVSPA92crjbv # CM3ABDg99Di2I4d9+6143KyJTks6SXOGXlMRMpYusehes5uDUfHWz2/XYg6f4TlM # x1hsYgNWd0VygW/+Bf6I8rzeN0hBlqkn1XNyNwZbGE9+XnFQCfZFX/dGDr9vbRYy # QQWZGL60/w8MMkS7HDsj+Kfvf5ciYw/lC7N2FwYVa32fjG0AfgWCi6kbwSm4/8Ef # WpDazMUCDaBmg+Y2Tvzf50rJhtj9c7/3L+IEqDAZ1nkCAZI3diOcNQ3l7qadr0oj # ZkebANmZASt8okFmib0cTg5ZH4sh5PTJJP062E45ozwm4XvrqoRzUcvFJWzr1rAq # tLUbVMpB79/Q4KgHyTaxPDkrfN+awcw6a63AIcJoH4aFYuACnqsIpGNJ1GuPCtEM # vNF0+H4hANxMRGTWkRaPTavxsG34KiqhpnPcYSrkRXI7bmuCh2b5wL6VoBBUjgpK # L17PwXC55BBzPcWfIlGoOQsoTal2GBP89KXF3WGjSJEOvBFprry5u4PaKH1drzpu # RlTaIIZ5Fhupxj6PycBIJgPaEktBH7xUTFjwP4KUrxd7IQKyh52hKVUkotwzW8or # mIhLPSqNCJGXPMShXkEWqhkcawftq/+wd1/NU6bPfSoRcIPsA/O6HdthdjFINce7 # cLd4GlZxZNCGSMFDwhsCAwEAAaOCAUkwggFFMB0GA1UdDgQWBBRd4Z/X2CTdmP2f # S1WwTyMQM1x7EDAfBgNVHSMEGDAWgBSfpxVdAF5iXYP05dJlpxtTNRnpcjBfBgNV # HR8EWDBWMFSgUqBQhk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2Ny # bC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcmwwbAYI # KwYBBQUHAQEEYDBeMFwGCCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAy # MDEwKDEpLmNydDAMBgNVHRMBAf8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMI # MA4GA1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEANlMdLa/bGAo7KtBU # colacdeTrqzV7+kJO4l/gK1T5c98OLao3EzZtXbd7U7mtsCTKSYTx/J+1F4Zk/fH # eQ7uOC5eGnZ7HuEkH1YN0C+mkZFx5yVJ/MvYn7Qfz8ttp/cZ8DxX490JQzkoC7h5 # Xrve1NIh9mWpbmGm1lyYoXYmKSvHDxxZamTgIqLdPhg/HT/kTLr6cbfxdi5mHpaa # GcK4ohTlJpcRFFe8SR8mWZQ6rrJHsrqtNQ07/dTeKjPHHzz4Zf9qXOB3jH/53dh6 # LonyWw0BTXeFj4S859Est7tWYd/kt7Y0HajfncsOtEWlfkVwqDi4BWdzAIPxqA5f # n72+Ycv9Wu9XLV/MBGndT3+nXn9qan5d8BXPvoZBfA1102aHyTeGFWjdyJ8GwBkA # w5DSuJannAoscZkisInT8pxnhOTrbzaCtaZC7Jv4vBdUYxnM2f5EhdlQ5KhvOuok # Po1WxjkOg7t3sshXBSmaAyeAeZhAEvWk9kiqpZS3smzm6B+q+u7IkeQf9h57u/im # LmGctARGnbQOp4BuHJ0kfSbK5IYbCpm9dbo6sICcjcfdjLKBhsDYDdWiNECA/ArI # whcKX5jM5QOOMRymL94Zo8XqgUuLlOa0HogJ1YKSXmlRipv5czw7Z6aRlfu14Uae # W9WcU/2e18XPiOx+/4wrUA2aGmUwggdxMIIFWaADAgECAhMzAAAAFcXna54Cm0mZ # AAAAAAAVMA0GCSqGSIb3DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMK # V2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0 # IENvcnBvcmF0aW9uMTIwMAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0 # ZSBBdXRob3JpdHkgMjAxMDAeFw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMyMjVa # MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdS # ZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMT # HU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0BAQEF # AAOCAg8AMIICCgKCAgEA5OGmTOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51yMo1 # V/YBf2xK4OK9uT4XYDP/XE/HZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY6GB9 # alKDRLemjkZrBxTzxXb1hlDcwUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9cmmv # Haus9ja+NSZk2pg7uhp7M62AW36MEBydUv626GIl3GoPz130/o5Tz9bshVZN7928 # jaTjkY+yOSxRnOlwaQ3KNi1wjjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDuaRr3t # pK56KTesy+uDRedGbsoy1cCGMFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74kpEe # HT39IM9zfUGaRnXNxF803RKJ1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2K26o # ElHovwUDo9Fzpk03dJQcNIIP8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5TI4C # vEJoLhDqhFFG4tG9ahhaYQFzymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZki1ug # poMhXV8wdJGUlNi5UPkLiWHzNgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9QBXps # xREdcu+N+VLEhReTwDwV2xo3xwgVGD94q0W29R6HXtqPnhZyacaue7e3PmriLq0C # AwEAAaOCAd0wggHZMBIGCSsGAQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUCBBYE # FCqnUv5kxJq+gpE8RjUpzxD/LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJlpxtT # NRnpcjBcBgNVHSAEVTBTMFEGDCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIBFjNo # dHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9yeS5o # dG0wEwYDVR0lBAwwCgYIKwYBBQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBD # AEEwCwYDVR0PBAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU1fZW # y4/oolxiaNE9lJBb186aGMQwVgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2NybC5t # aWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIwMTAt # MDYtMjMuY3JsMFoGCCsGAQUFBwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDovL3d3 # dy5taWNyb3NvZnQuY29tL3BraS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0y # My5jcnQwDQYJKoZIhvcNAQELBQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/ypb+pc # FLY+TkdkeLEGk5c9MTO1OdfCcTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulmZzpT # Td2YurYeeNg2LpypglYAA7AFvonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM9W0j # VOR4U3UkV7ndn/OOPcbzaN9l9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECWOKz3 # +SmJw7wXsFSFQrP8DJ6LGYnn8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4FOmR # sqlb30mjdAy87JGA0j3mSj5mO0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3UwxTSw # ethQ/gpY3UA8x1RtnWN0SCyxTkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPXfx5b # RAGOWhmRaw2fpCjcZxkoJLo4S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVXVAmx # aQFEfnyhYWxz/gq77EFmPWn9y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGConsX # HRWJjXD+57XQKBqJC4822rpM+Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU5nR0 # W2rRnj7tfqAxM328y+l7vzhwRNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEGahC0 # HVUzWLOhcGbyoYIDUDCCAjgCAQEwgfmhgdGkgc4wgcsxCzAJBgNVBAYTAlVTMRMw # EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN # aWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNVBAsTHE1pY3Jvc29mdCBBbWVyaWNh # IE9wZXJhdGlvbnMxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjo5NjAwLTA1RTAt # RDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaIjCgEB # MAcGBSsOAwIaAxUAov3zMRbZKq+1zF3bEcllNJUih2eggYMwgYCkfjB8MQswCQYD # VQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEe # MBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3Nv # ZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAO3naAgwIhgP # MjAyNjA2MjUwODQwNDBaGA8yMDI2MDYyNjA4NDA0MFowdzA9BgorBgEEAYRZCgQB # MS8wLTAKAgUA7edoCAIBADAKAgEAAgIfJwIB/zAHAgEAAgIULDAKAgUA7ei5iAIB # ADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMCoAowCAIBAAIDB6EgoQow # CAIBAAIDAYagMA0GCSqGSIb3DQEBCwUAA4IBAQB/o7dKHcBKj79xH7Ed/wloNqXe # ZhSIw3GXF6sTFOLXSh/2vu/Pto+EZlyyVOmXCmPrZJF84AOsMAuDeusxz9Jj+uYx # ggxhxwo1bKCItjsn24VPTTUzQi2aFQ0G1BCZUG+1VujR3OaSlD0+o8EZKvbcYbnq # 1zHKl/ts/s68RgjU9PtB+Xs2n9yg+Q5Ic12zu1Zw04pSr+Ew3A8Eoz1e+03kknyh # z0Pio2EQswpbUSWLxzeZL0DG0QUd8nSRRQOHODnGTDBhgjLPTEX39uPG7XIdJMzn # 7M3aUe9Hyx6EAuI/aKwnOqNcDzqKGMwMECEhrlS2vkiEzDnN8tfHwdosPianMYIE # DTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x # EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv # bjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAIm # NbQ+Z0OT9h8AAQAAAiYwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzEN # BgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQgKmzeow3j+qmhCzBNPoHtl0Lf # X80WdQtsIiy/BoTX3WUwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCDMMlxh # Z0zbGUQa7OhjfwW1dKJGjT91QWKixY1LBZ24KjCBmDCBgKR+MHwxCzAJBgNVBAYT # AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBU # aW1lLVN0YW1wIFBDQSAyMDEwAhMzAAACJjW0PmdDk/YfAAEAAAImMCIEIK8aVowO # xh55O0f704pKbzpDH6IZ2Z+XfuVNaxR6HuYEMA0GCSqGSIb3DQEBCwUABIICALWl # 7wmzHIc2M3DploHELSfQXa09rjb90wCOpiaIrw6i+7BVI07WOFFBR1//aLHzttoW # snk82gbkdVUmOChA42Ot5sWhqo0KVqE8QUSCtChEyolHzztzJvbq7Cq+Lb7hKnDf # 3YC+5F4AbufMk3PGC8+7hr9m+UyLaDFfY25BB0ULdSkw5AQrjeH7BzgeZWZ65AUt # gAjEFYygHhtMJYA+jSWhrqgppffkT7JNxTJptruu5lhNc1LOrowB+AlMarWFoNLS # Q5aDLomBLPKp8hh29Y9JNJKNeERlkPqmL5HOdriO001QpqcI/D7cNwSfQNndubAm # UJwZqHNhmtunSSifT2/cTt+cbUm3KpDyQz7kO9PU0loJlJ6gIagCr5fbDvml6IbT # cF53Q+PAn9DuLDd+yTWrtcZMLlF3rkjhHH3bt3ohg5damqZdzE7ECJWb3IMvYpEM # tY//9z+NV+5kE6jKXDa9dfbiBx7f2ltchTKRcOdv36t/cdJdIM8UBVA0dEizyAzG # kTzjx+3KdUuAAKdLETOji92U8YhDN18hy1HOa7BHzd2MvnGQ8Uqgfy7+rANRRc14 # p879QD8kVc+f81i8X/Apbj+YkH02d8OR/2upKilBb2wc3X2hg54IaZBSyh4/etxj # tdKrjmpVkcAIfwtrKh+JfO59OjKOSsZq39iCzhnB # SIG # End signature block |