functions/Get-WUGMonitorTemplate.ps1

<#
.SYNOPSIS
    Retrieves monitor templates or supported monitor types from WhatsUp Gold.
 
.DESCRIPTION
    Get-WUGMonitorTemplate queries the WhatsUp Gold REST API:
    - GET /api/v1/monitors/- to list active, passive, and performance monitor templates.
    - GET /api/v1/monitors/-/config/supported-types to list supported monitor types.
    - GET /api/v1/monitors/{monitorId}/config/template to export a single monitor template.
    - GET /api/v1/monitors/-/config/template to export all monitor templates.
    Supports filtering by type, search string, and various inclusion switches.
 
.PARAMETER MonitorId
    The ID of a specific monitor to retrieve the template for.
 
.PARAMETER MonitorTemplate
    Switch to retrieve the template for a specific monitor by MonitorId.
    Endpoint: GET /api/v1/monitors/{monitorId}/config/template
 
.PARAMETER AllMonitorTemplates
    Switch to retrieve all monitor templates for export.
    Endpoint: GET /api/v1/monitors/-/config/template
 
.PARAMETER Type
    Filter by monitor type. Valid values: all, active, performance, passive.
 
.PARAMETER Search
    A search string to filter monitor templates by name.
 
.PARAMETER View
    The level of detail to return. Valid values: id, basic, info, summary, details. Default: info.
 
.PARAMETER PageId
    The page identifier for paginated results.
 
.PARAMETER Limit
    The maximum number of results to return per page. Default: 250.
 
.PARAMETER IncludeDeviceMonitors
    Include monitors assigned to devices.
 
.PARAMETER IncludeSystemMonitors
    Include system-level monitors.
 
.PARAMETER IncludeCoreMonitors
    Include core monitors.
 
.PARAMETER AllMonitors
    Include all monitors regardless of assignment.
 
.PARAMETER SupportedTypes
    Switch to retrieve the list of supported monitor types instead of templates.
    Endpoint: GET /api/v1/monitors/-/config/supported-types
 
.EXAMPLE
    Get-WUGMonitorTemplate
 
    Returns all monitor templates with default settings.
 
.EXAMPLE
    Get-WUGMonitorTemplate -Type active -Search "HTTP"
 
    Returns active monitor templates matching "HTTP".
 
.EXAMPLE
    Get-WUGMonitorTemplate -SupportedTypes
 
    Returns all supported monitor types (active, passive, performance).
 
.EXAMPLE
    Get-WUGMonitorTemplate -MonitorTemplate -MonitorId "abc-123"
 
    Returns the template for the specified monitor.
 
.EXAMPLE
    Get-WUGMonitorTemplate -AllMonitorTemplates
 
    Returns all monitor templates for export.
 
.NOTES
    Author: Jason Alberino (jason@wug.ninja)
    Reference: https://docs.ipswitch.com/NM/WhatsUpGold2024/02_Guides/rest_api/#tag/Monitor-Templates
#>

function Get-WUGMonitorTemplate {
    [CmdletBinding(DefaultParameterSetName = 'Default')]
    param(
        [Parameter(ParameterSetName = 'Default')]
        [Parameter(ParameterSetName = 'SupportedTypes')]
        [ValidateSet('all', 'active', 'performance', 'passive')]
        [string]$Type,

        [Parameter(ParameterSetName = 'Default')]
        [string]$Search,

        [Parameter(ParameterSetName = 'Default')]
        [ValidateSet('id', 'basic', 'info', 'summary', 'details')]
        [string]$View = 'info',

        [Parameter(ParameterSetName = 'Default')]
        [string]$PageId,

        [Parameter(ParameterSetName = 'Default')]
        [int]$Limit = 250,

        [Parameter(ParameterSetName = 'Default')]
        [switch]$IncludeDeviceMonitors,

        [Parameter(ParameterSetName = 'Default')]
        [switch]$IncludeSystemMonitors,

        [Parameter(ParameterSetName = 'Default')]
        [switch]$IncludeCoreMonitors,

        [Parameter(ParameterSetName = 'Default')]
        [switch]$AllMonitors,

        [Parameter(Mandatory = $true, ParameterSetName = 'SupportedTypes')]
        [switch]$SupportedTypes,

        [Parameter(ParameterSetName = 'SupportedTypes')]
        [switch]$IncludeCoreMonitorTypes,

        [Parameter(Mandatory = $true, ParameterSetName = 'MonitorTemplate')]
        [switch]$MonitorTemplate,

        [Parameter(Mandatory = $true, ParameterSetName = 'MonitorTemplate', ValueFromPipeline = $true, ValueFromPipelineByPropertyName = $true)]
        [Alias('id')]
        [string[]]$MonitorId,

        [Parameter(Mandatory = $true, ParameterSetName = 'AllMonitorTemplates')]
        [switch]$AllMonitorTemplates,

        [Parameter(ParameterSetName = 'MonitorTemplate')]
        [Parameter(ParameterSetName = 'AllMonitorTemplates')]
        [ValidateSet('all', 'clone', 'transfer', 'update')]
        [string]$TemplateOptions
    )

    begin {
        Write-Debug "Initializing Get-WUGMonitorTemplate function."
        Write-Debug "ParameterSetName: $($PSCmdlet.ParameterSetName)"
        Write-Debug "Type: $Type"
        Write-Debug "Search: $Search"
        Write-Debug "View: $View"
        Write-Debug "PageId: $PageId"
        Write-Debug "Limit: $Limit"
        Write-Debug "IncludeDeviceMonitors: $IncludeDeviceMonitors"
        Write-Debug "IncludeSystemMonitors: $IncludeSystemMonitors"
        Write-Debug "IncludeCoreMonitors: $IncludeCoreMonitors"
        Write-Debug "AllMonitors: $AllMonitors"

        # Initialize collection for final output
        $finalOutput = @()
        $baseUri = "$($global:WhatsUpServerBaseURI)/api/v1/monitors/-"
    }

    process {
        # Handle SupportedTypes parameter set
        if ($PSCmdlet.ParameterSetName -eq 'SupportedTypes') {
            $queryParams = @()
            if ($Type) { $queryParams += "type=$Type" }
            if ($IncludeCoreMonitorTypes) { $queryParams += "includeCoreMonitorTypes=true" }
            $query = if ($queryParams.Count -gt 0) { "?" + ($queryParams -join "&") } else { "" }
            $supportedUri = "$($global:WhatsUpServerBaseURI)/api/v1/monitors/-/config/supported-types${query}"
            Write-Verbose "Requesting supported types from URI: $supportedUri"
            try {
                $result = Get-WUGAPIResponse -Uri $supportedUri -Method GET
                if ($result.data) { $finalOutput += $result.data }
            }
            catch {
                Write-Error "Error fetching supported monitor types: $($_.Exception.Message)"
            }
            return
        }

        # Handle MonitorTemplate parameter set
        if ($PSCmdlet.ParameterSetName -eq 'MonitorTemplate') {
            foreach ($mid in $MonitorId) {
                $queryParams = @()
                if ($TemplateOptions) { $queryParams += "options=$TemplateOptions" }
                $query = if ($queryParams.Count -gt 0) { "?" + ($queryParams -join "&") } else { "" }
                $templateUri = "$($global:WhatsUpServerBaseURI)/api/v1/monitors/${mid}/config/template${query}"
                Write-Verbose "Requesting monitor template from URI: $templateUri"
                try {
                    $result = Get-WUGAPIResponse -Uri $templateUri -Method GET
                    if ($result.data) { $finalOutput += $result.data }
                }
                catch {
                    Write-Error "Error fetching template for monitor ${mid}: $($_.Exception.Message)"
                }
            }
            return
        }

        # Handle AllMonitorTemplates parameter set
        if ($PSCmdlet.ParameterSetName -eq 'AllMonitorTemplates') {
            $queryParams = @()
            if ($TemplateOptions) { $queryParams += "options=$TemplateOptions" }
            $query = if ($queryParams.Count -gt 0) { "?" + ($queryParams -join "&") } else { "" }
            $allTemplatesUri = "$($global:WhatsUpServerBaseURI)/api/v1/monitors/-/config/template${query}"
            Write-Verbose "Requesting all monitor templates from URI: $allTemplatesUri"
            try {
                $result = Get-WUGAPIResponse -Uri $allTemplatesUri -Method GET
                if ($result.data) { $finalOutput += $result.data }
            }
            catch {
                Write-Error "Error fetching all monitor templates: $($_.Exception.Message)"
            }
            return
        }

        # Build the query string based on provided parameters
        $queryString = ""
        if ($PSBoundParameters.ContainsKey('Type') -and ![string]::IsNullOrWhiteSpace($Type)) {
            $queryString += "type=$([uri]::EscapeDataString($Type))&"
        }
        if ($PSBoundParameters.ContainsKey('Search') -and ![string]::IsNullOrWhiteSpace($Search)) {
            $queryString += "search=$([uri]::EscapeDataString($Search))&"
        }
        if ($PSBoundParameters.ContainsKey('View') -and ![string]::IsNullOrWhiteSpace($View)) {
            $queryString += "view=$([uri]::EscapeDataString($View))&"
        }
        if ($PSBoundParameters.ContainsKey('PageId') -and ![string]::IsNullOrWhiteSpace($PageId)) {
            $queryString += "pageId=$([uri]::EscapeDataString($PageId))&"
        }
        if ($PSBoundParameters.ContainsKey('Limit') -and $Limit -gt 0) {
            $queryString += "limit=$([uri]::EscapeDataString(${Limit}.ToString()))&"
        }
        if ($IncludeDeviceMonitors) {
            $queryString += "includeDeviceMonitors=true&"
        }
        if ($IncludeSystemMonitors) {
            $queryString += "includeSystemMonitors=true&"
        }
        if ($IncludeCoreMonitors) {
            $queryString += "includeCoreMonitors=true&"
        }
        if ($AllMonitors) {
            $queryString += "allMonitors=true&"
        }

        # Trim the trailing '&' if it exists
        $queryString = $queryString.TrimEnd('&')
        
        # Construct the URI
        $monitorsUri = $baseUri
        if (-not [string]::IsNullOrWhiteSpace($queryString)) {
            $monitorsUri += "?$queryString"
        }

        Write-Verbose "Requesting URI: $monitorsUri"

        try {
            # Make the API call and retrieve the response
            $result = Get-WUGAPIResponse -Uri $monitorsUri -Method GET
            if ($result.data) {
                if ($result.data.activeMonitors) {
                    foreach ($monitor in $result.data.activeMonitors) {
                        $finalOutput += $monitor
                    }
                }
                if ($result.data.passiveMonitors) {
                    foreach ($monitor in $result.data.passiveMonitors) {
                        $finalOutput += $monitor
                    }
                }
                if ($result.data.performanceMonitors) {
                    foreach ($monitor in $result.data.performanceMonitors) {
                        $finalOutput += $monitor
                    }
                }
            }
        }
        catch {
            Write-Error "Error fetching monitor templates: $($_.Exception.Message)"
        }
    }

    end {
        Write-Debug "Get-WUGMonitorTemplate function completed."
        # Output the final data
        return $finalOutput
    }
}
# SIG # Begin signature block
# MIIVlwYJKoZIhvcNAQcCoIIViDCCFYQCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCD1B/bnKELpwCQF
# //3Bn1iSH47HDiAyaz1ISo3D0rfuiaCCEdMwggVvMIIEV6ADAgECAhBI/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
# BAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgQr5YdAkZ/FcQ1SQVZRRqEKivSvKnqD7z
# ISYzwHeyxtIwDQYJKoZIhvcNAQEBBQAEggIAq9A5dIGhmWlssJ8ZXWuSg8qY8Uxu
# w3tubW0ieejhXByS7AVUH0NQGb8K+yOgCwdynJltBjH5+xAj7pONX96r69yaJDOb
# aMi6UrjInYxCZ4KSr3sGHe03/uH0e45LIfqLcDQn0Wac8V8EoEqfmDjbxzStgboV
# hjniF5HlStPu0nEF/Z+oTicYSY5fgGOHnvru7fucgRZ6SoCHtaxL9YbRXZaywVab
# gv6l+RT4KQkF8mD+GmA4Xf5aqF7xml6z+n0984EtiLtlIMy0zDZb+qG+COp1dgEQ
# 6cQyYIWsttMmaMKd2fwO02R9t7BYa1QVhKVaBLVI88JfmjUHwHvIy3GPILJsg3RP
# qpyI7B6HSysQROF4EDE50wAS1JLJqe27HyjF6RUs/xVwAst17Os3lRH3XT3uuNnj
# A6RXZGyW/WGxCP9EiLmxlGde90bfh0s1a9NpyFsLiYwKwNWPQf///e8dAdTlbs8J
# 8vOBe5Kn56K2GNI8V6BiQqZJiV2AzvX+g2uW/Z/PP1hVlVpD813dhOgkNks7Myo5
# 5n/zElsmY+EAflRuJby/LyacAo/BIXRFxpjVqB2pBrR9CaitV6CPPZ0Hy+uYbcAU
# ihGFRfkWfWDLuX0WoDwzeK+ONmTf+giqmsns77Bt6Si0GOAKUh7QBqRwV9wtFWfb
# ENBoeGizUF/E8M0=
# SIG # End signature block