DeprecatedApiTranslator/InputTransformation.ps1

<#
.SYNOPSIS
Creates empty input transformation sctructure
 
.DESCRIPTION
Input transformation structure contains information Path, Header, Query, and Body input of an API operation
 
.EXAMPLE
New-InputTransformationStructure
 
#>

function New-InputTransformationStructure {
    [PSCustomObject]@{
        Path = $null
        Header = $null
        Query = $null
        Body = $null
    }
}

<#
.SYNOPSIS
Adds Path, Header, Query, or Body to an existing InputTransformationStructure
 
.DESCRIPTION
Combines two InputTransformationStructure incrementing the $Base InputTransformationStructure with the content from the $additionInputStructure
 
.PARAMETER Base
The InputTransformationStructure that will be extended with the data from $Addition
 
.PARAMETER Addition
InputTransformationStructure which content will be added to the $Base
 
.EXAMPLE
Join-InputTransformationStructure -Base [ref]$baseInputStructure -Addition $additionInputStructure
#>

function Join-InputTransformationStructure {
    param(
        [Parameter(Mandatory)]
        [ref]
        $Base,

        [Parameter(Mandatory)]
        [PSCustomObject]
        $Addition)

    foreach ($htProperty in 'Path', 'Header', 'Query') {
        if ($null -ne $Addition.$htProperty) {
            if ($null -eq $Base.Value.$htProperty) {
                $Base.Value.$htProperty = $Addition.$htProperty
            } else {
                foreach ($htKeyValue in $Addition.$htProperty.GetEnumerator()) {
                    $Base.Value.$htProperty[$htKeyValue.Key] = $htKeyValue.Value
                }
            }
        }
    }

    if ($null -ne $Addition.Body) {
        if ($null -eq $Base.Value.Body) {
            $Base.Value.Body = $Addition.Body
        } else {
            $propsToAdd = $Addition.Body | Get-Member -MemberType NoteProperty | ForEach-Object { $_.Name }
            foreach ($prop in $propsToAdd) {
                $Base.Value.Body | Add-Member -MemberType NoteProperty -Name $prop -Value $Addition.Body.$prop
            }
        }
    }
}

<#
.SYNOPSIS
Rearranges Path input of an API operation
 
.DESCRIPTION
Rearranges Path API operation input to Header, Query, and Body if needed from the OperationTranslateSchema
 
.PARAMETER OperationTranslateSchema
Translation Schema Object retrieved from Get-OperationTranslationSchema
 
.PARAMETER PathParams
Hashtable with Key name of the path parameter and Value the argument
#>

function Format-PathParams {
    param(
        [Parameter(Mandatory)]
        [PSCustomObject]
        $OperationTranslateSchema,


        [Parameter(Mandatory)]
        [hashtable]
        $PathParams
    )

    $result = New-InputTransformationStructure

    if ( $OperationTranslateSchema.OldInPathParams -ne $OperationTranslateSchema.NewInPathParams ) {
        # Process Path parameter from New API to different place in the Old API
        $OperationTranslateSchema.NewInPathParams | Foreach-Object {
            $newPathParam = $_
            if ($null -eq $OperationTranslateSchema.OldInPathParams -or `
                $OperationTranslateSchema.OldInPathParams -notcontains $newPathParam) {
                # Moving Path parameter from New API to different place in the old API

                # Check Body
                if ($null -ne $OperationTranslateSchema.OldInBodyStruct) {
                    $bodyPropNames = $OperationTranslateSchema.OldInBodyStruct | Get-Member -MemberType NoteProperty | ForEach-Object { $_.Name }
                    if ( $bodyPropNames -contains $newPathParam) {
                        if ($null -eq $result.Body) {
                            # initialize body on first param
                            $result.Body = @{}
                        }

                        # fill the argument value in the body
                        $result.Body[$newPathParam] = $PathParams[$newPathParam]
                    }
                }

                # Check Query
                if ($null -ne $OperationTranslateSchema.OldInQueryParams) {
                    if ( $OperationTranslateSchema.OldInQueryParams -contains $newPathParam) {
                        if ($null -eq $result.Query) {
                            # initialize query on first param
                            $result.Query = @{}
                        }

                        # fill the argument value in the query params
                        $result.Query[$newPathParam] = $PathParams[$newPathParam]
                    }
                }

                # Check Header
                if ($null -ne $OperationTranslateSchema.OldInHeaderParams) {
                    if ( $OperationTranslateSchema.OldInHeaderParams -contains $newPathParam) {
                        if ($null -eq $result.Header) {
                            # initialize query on first param
                            $result.Header = @{}
                        }

                        # fill the argument value in the Header params
                        $result.Header[$newPathParam] = $PathParams[$newPathParam]
                    }
                }
            }
        }
    }

    # return
    $result
}

<#
.SYNOPSIS
Rearranges Headers input of an API operation
 
.DESCRIPTION
Rearranges Headers API operation input to Path, Query, and Body if needed from the OperationTranslateSchema
 
.PARAMETER OperationTranslateSchema
Translation Schema Object retrieved from Get-OperationTranslationSchema
 
.PARAMETER Headers
Heashtable with HTTP Headers of an API. The function modifies the headers.
#>

function Format-Headers {
    param(
        [Parameter(Mandatory)]
        [PSCustomObject]
        $OperationTranslateSchema,

        [Parameter(Mandatory)]
        [hashtable]
        $Headers
    )

    $result = New-InputTransformationStructure

    if ( $OperationTranslateSchema.OldInHeaderParams -ne $OperationTranslateSchema.NewInHeaderParams ) {
        # Process Header parameter from New API to different place in the Old API
        $OperationTranslateSchema.NewInHeaderParams | Foreach-Object {
            $newHeaderParam = $_
            if ($null -eq $OperationTranslateSchema.OldInHeaderParams -or `
                $OperationTranslateSchema.OldInHeaderParams -notcontains $newPathParam) {

                # Moving Header parameter from New API to different place in the old API

                # Check Path
                if ($null -ne $OperationTranslateSchema.OldInPathParams) {
                    if ( $OperationTranslateSchema.OldInPathParams -contains $newHeaderParam) {
                        if ($null -eq $result.Path) {
                            # initialize query on first param
                            $result.Path = @{}
                        }

                        # fill the argument value in the path params
                        $result.Path[$newHeaderParam] = $Headers[$newHeaderParam]
                        # remove the parameter from headers
                        $Headers.Remove($newHeaderParam)
                    }
                }

                # Check Body
                if ($null -ne $OperationTranslateSchema.OldInBodyStruct) {
                    $bodyPropNames = $OperationTranslateSchema.OldInBodyStruct | Get-Member -MemberType NoteProperty | ForEach-Object { $_.Name }
                    if ( $bodyPropNames -contains $newHeaderParam) {
                        if ($null -eq $result.Body) {
                            # initialize body on first param
                            $result.Body = @{}
                        }

                        # fill the argument value in the body
                        $result.Body[$newHeaderParam] = $Headers[$newHeaderParam]
                        # remove the parameter from headers
                        $Headers.Remove($newHeaderParam)
                    }
                }

                # Check Query
                if ($null -ne $OperationTranslateSchema.OldInQueryParams) {
                    if ( $OperationTranslateSchema.OldInQueryParams -contains $newHeaderParam) {
                        if ($null -eq $result.Query) {
                            # initialize query on first param
                            $result.Query = @{}
                        }

                        # fill the argument value in the query params
                        $result.Query[$newHeaderParam] = $Headers[$newHeaderParam]
                        # remove the parameter from headers
                        $Headers.Remove($newHeaderParam)
                    }
                }
            }
        }
    }

    # return
    $result
}

<#
.SYNOPSIS
Rearranges Body input of an API operation
 
.DESCRIPTION
Rearranges Body fields of an API operation input to Path, Query, and Headers if needed from the OperationTranslateSchema
 
.PARAMETER OperationTranslateSchema
Translation Schema Object retrieved from Get-OperationTranslationSchema
 
.PARAMETER Body
PSCustomObject with the HTTP Body of an API. The function modifies the body object.
#>

function Format-Body {
    param(
        [Parameter(Mandatory)]
        [PSCustomObject]
        $OperationTranslateSchema,

        [Parameter(Mandatory)]
        [ref]
        $Body
    )

    $result = New-InputTransformationStructure

    # Only Top Level Body Fields can be moved to Path, Query, or Headers
    $newBodyPropNames = $null
    if ($null -ne $OperationTranslateSchema.NewInBodyStruct) {
        $newBodyPropNames = $OperationTranslateSchema.NewInBodyStruct | Get-Member -MemberType NoteProperty | Foreach-Object { $_.Name }
    }

    $oldBodyPropNames = $null
    if ($null -ne $OperationTranslateSchema.OldInBodyStruct) {
        $oldBodyPropNames = $OperationTranslateSchema.OldInBodyStruct | Get-Member -MemberType NoteProperty | Foreach-Object { $_.Name }
    }

    $bodyFieldsToRemove = @()
    if ( $oldBodyPropNames -ne $newBodyPropNames ) {
        # Process Body properties from the New API to different place in the Old API
        $newBodyPropNames | Foreach-Object {
            $newBodyProp = $_
            if ($null -eq $oldBodyPropNames -or `
                $oldBodyPropNames -notcontains $newBodyProp) {

                # Moving Body field from New API to different place in the old API

                # Check Path
                if ($null -ne $OperationTranslateSchema.OldInPathParams) {
                    if ( $OperationTranslateSchema.OldInPathParams -contains $newBodyProp) {
                        if ($null -eq $result.Path) {
                            # initialize query on first param
                            $result.Path = @{}
                        }

                        if ($null -ne $Body.Value) {
                            # fill the argument value in the path params
                            $result.Path[$newBodyProp] = $Body.Value.$newBodyProp
                            # Collect body fileds that have to be removed from the Body structure
                            $bodyFieldsToRemove += $newBodyProp
                        }
                    }
                }

                # Check Headers
                if ($null -ne $OperationTranslateSchema.OldInHeaderParams) {
                    if ( $OperationTranslateSchema.OldInHeaderParams -contains $newBodyProp) {
                        if ($null -eq $result.Header) {
                            # initialize query on first param
                            $result.Header = @{}
                        }

                        if ($null -ne $Body.Value) {
                            # fill the argument value in the headers
                            $result.Header[$newBodyProp] = $Body.Value.$newBodyProp
                            # Collect body fileds that have to be removed from the Body structure
                            $bodyFieldsToRemove += $newBodyProp
                        }
                    }
                }

                # Check Query
                if ($null -ne $OperationTranslateSchema.OldInQueryParams) {
                    if ( $OperationTranslateSchema.OldInQueryParams -contains $newBodyProp) {
                        if ($null -eq $result.Query) {
                            # initialize query on first param
                            $result.Query = @{}
                        }

                        if ($null -ne $Body) {
                            # fill the argument value in the query params
                            $result.Query[$newBodyProp] = $Body.Value.$newBodyProp
                            # Collect body fileds that have to be removed from the Body structure
                            $bodyFieldsToRemove += $newBodyProp
                        }
                    }
                }
            }
        }
    }

    # Remove transformed properties from the body structure
    if ($bodyFieldsToRemove.Count -gt 0 -and $null -ne $Body.Value) {
        $bodyPropNames = $Body.Value | Get-Member -MemberType NoteProperty | Foreach-Object { $_.Name }
        $transformedBody = @{}
        foreach ($bodyProp in $bodyPropNames) {
            if ($bodyFieldsToRemove -notcontains $bodyProp) {
                $transformedBody[$bodyProp] = $Body.Value.$bodyProp
            }
        }

        $Body.Value = [PSCustomObject]$transformedBody
    }

    # return
    $result
}
# SIG # Begin signature block
# MIIpzwYJKoZIhvcNAQcCoIIpwDCCKbwCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCDOLpl0Jbn+Cr0B
# y/BCCZ7Vuev3KtlSXHw1nv/RxHXm4qCCDrwwggawMIIEmKADAgECAhAIrUCyYNKc
# TJ9ezam9k67ZMA0GCSqGSIb3DQEBDAUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQK
# EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNV
# BAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDAeFw0yMTA0MjkwMDAwMDBaFw0z
# NjA0MjgyMzU5NTlaMGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg
# SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBDb2RlIFNpZ25pbmcg
# UlNBNDA5NiBTSEEzODQgMjAyMSBDQTEwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAw
# ggIKAoICAQDVtC9C0CiteLdd1TlZG7GIQvUzjOs9gZdwxbvEhSYwn6SOaNhc9es0
# JAfhS0/TeEP0F9ce2vnS1WcaUk8OoVf8iJnBkcyBAz5NcCRks43iCH00fUyAVxJr
# Q5qZ8sU7H/Lvy0daE6ZMswEgJfMQ04uy+wjwiuCdCcBlp/qYgEk1hz1RGeiQIXhF
# LqGfLOEYwhrMxe6TSXBCMo/7xuoc82VokaJNTIIRSFJo3hC9FFdd6BgTZcV/sk+F
# LEikVoQ11vkunKoAFdE3/hoGlMJ8yOobMubKwvSnowMOdKWvObarYBLj6Na59zHh
# 3K3kGKDYwSNHR7OhD26jq22YBoMbt2pnLdK9RBqSEIGPsDsJ18ebMlrC/2pgVItJ
# wZPt4bRc4G/rJvmM1bL5OBDm6s6R9b7T+2+TYTRcvJNFKIM2KmYoX7BzzosmJQay
# g9Rc9hUZTO1i4F4z8ujo7AqnsAMrkbI2eb73rQgedaZlzLvjSFDzd5Ea/ttQokbI
# YViY9XwCFjyDKK05huzUtw1T0PhH5nUwjewwk3YUpltLXXRhTT8SkXbev1jLchAp
# QfDVxW0mdmgRQRNYmtwmKwH0iU1Z23jPgUo+QEdfyYFQc4UQIyFZYIpkVMHMIRro
# OBl8ZhzNeDhFMJlP/2NPTLuqDQhTQXxYPUez+rbsjDIJAsxsPAxWEQIDAQABo4IB
# WTCCAVUwEgYDVR0TAQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQUaDfg67Y7+F8Rhvv+
# YXsIiGX0TkIwHwYDVR0jBBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08wDgYDVR0P
# AQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUFBwMDMHcGCCsGAQUFBwEBBGswaTAk
# BggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsGAQUFBzAC
# hjVodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9v
# dEc0LmNydDBDBgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdpY2VydC5j
# b20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNybDAcBgNVHSAEFTATMAcGBWeBDAED
# MAgGBmeBDAEEATANBgkqhkiG9w0BAQwFAAOCAgEAOiNEPY0Idu6PvDqZ01bgAhql
# +Eg08yy25nRm95RysQDKr2wwJxMSnpBEn0v9nqN8JtU3vDpdSG2V1T9J9Ce7FoFF
# UP2cvbaF4HZ+N3HLIvdaqpDP9ZNq4+sg0dVQeYiaiorBtr2hSBh+3NiAGhEZGM1h
# mYFW9snjdufE5BtfQ/g+lP92OT2e1JnPSt0o618moZVYSNUa/tcnP/2Q0XaG3Ryw
# YFzzDaju4ImhvTnhOE7abrs2nfvlIVNaw8rpavGiPttDuDPITzgUkpn13c5Ubdld
# AhQfQDN8A+KVssIhdXNSy0bYxDQcoqVLjc1vdjcshT8azibpGL6QB7BDf5WIIIJw
# 8MzK7/0pNVwfiThV9zeKiwmhywvpMRr/LhlcOXHhvpynCgbWJme3kuZOX956rEnP
# LqR0kq3bPKSchh/jwVYbKyP/j7XqiHtwa+aguv06P0WmxOgWkVKLQcBIhEuWTatE
# QOON8BUozu3xGFYHKi8QxAwIZDwzj64ojDzLj4gLDb879M4ee47vtevLt/B3E+bn
# KD+sEq6lLyJsQfmCXBVmzGwOysWGw/YmMwwHS6DTBwJqakAwSEs0qFEgu60bhQji
# WQ1tygVQK+pKHJ6l/aCnHwZ05/LWUpD9r4VIIflXO7ScA+2GRfS0YW6/aOImYIbq
# yK+p/pQd52MbOoZWeE4wgggEMIIF7KADAgECAhAIV5dCqVO62Q1CN6Pz44xeMA0G
# CSqGSIb3DQEBCwUAMGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg
# SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBDb2RlIFNpZ25pbmcg
# UlNBNDA5NiBTSEEzODQgMjAyMSBDQTEwHhcNMjIwNTA0MDAwMDAwWhcNMjQwNTA0
# MjM1OTU5WjCBxTETMBEGCysGAQQBgjc8AgEDEwJVUzEZMBcGCysGAQQBgjc8AgEC
# EwhEZWxhd2FyZTEdMBsGA1UEDwwUUHJpdmF0ZSBPcmdhbml6YXRpb24xEDAOBgNV
# BAUTBzI4NTM4OTQxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRIw
# EAYDVQQHEwlQYWxvIEFsdG8xFTATBgNVBAoTDFZNd2FyZSwgSW5jLjEVMBMGA1UE
# AxMMVk13YXJlLCBJbmMuMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA
# 6kAACeNtbODhaayXmhKLpjNsCENWh3WTl/NOvObo1Prl949C6y2EU9zMsfYuODNI
# Tv2OoByT3nBCXk3G1gzanGscaUDgr18pzf77Og4ArUHNzil6aX7uAlvKRBPl+6Hv
# wpSMtyK0D6J79UKfYqztvf/9YOFJLGCiq7Dw8932vg0ijcH7sLGgoSjH6+jqBwTT
# suOav1lyiAxE4Q6r8KyDNjg6mYu2uu1TALU4B7/OX/zBIadArVgPHRurud9SSGPq
# itFCgd2OuY0r81Jr0LRxOnwWHLqPOwuSmsnxw7ZaQJCz7FY3oxiES8mbA+VV6zN8
# 18/s2v5htLd3SRrE5n4mhzGyXAVkqIILKiSppibj59dEc5voQ/1MXJ25W1kSy2MX
# B8XjI9j64JzcpjW5YPs2M+BR4mIEC5i2mURO/u3HGG9prRTPOv2MN1WzbLLUG98O
# sqLTrUTQtdvpcWizS1c6m9IFY14fj1OeagnUwY3W0THEloNKC1WZoe+5Oxq/NRxO
# Bz4N4fzCsqc44TBUWmHSqBVOV8PpmQ8xVrfj+aLdk40naf/cf9cdvqslsrKtzkvL
# b+ohLCvqymua9GLNmUbY2EyrOogMseRX8y0awJS4cWRLl9c1O7j7q31GwfX+ozEI
# h1z4pBNATaZSbTTsMi4eKMCzsU/QiDM/iCaZbWL6VgsCAwEAAaOCAkkwggJFMB8G
# A1UdIwQYMBaAFGg34Ou2O/hfEYb7/mF7CIhl9E5CMB0GA1UdDgQWBBRS+9+VGnhS
# 2uj4hRdXRyZkmfv5GzBCBgNVHREEOzA5gRJub3JlcGx5QHZtd2FyZS5jb22gIwYI
# KwYBBQUHCAOgFzAVDBNVUy1ERUxBV0FSRS0yODUzODk0MA4GA1UdDwEB/wQEAwIH
# gDATBgNVHSUEDDAKBggrBgEFBQcDAzCBtQYDVR0fBIGtMIGqMFOgUaBPhk1odHRw
# Oi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkRzRDb2RlU2lnbmlu
# Z1JTQTQwOTZTSEEzODQyMDIxQ0ExLmNybDBToFGgT4ZNaHR0cDovL2NybDQuZGln
# aWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0Q29kZVNpZ25pbmdSU0E0MDk2U0hB
# Mzg0MjAyMUNBMS5jcmwwPQYDVR0gBDYwNDAyBgVngQwBAzApMCcGCCsGAQUFBwIB
# FhtodHRwOi8vd3d3LmRpZ2ljZXJ0LmNvbS9DUFMwgZQGCCsGAQUFBwEBBIGHMIGE
# MCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wXAYIKwYBBQUH
# MAKGUGh0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRH
# NENvZGVTaWduaW5nUlNBNDA5NlNIQTM4NDIwMjFDQTEuY3J0MAwGA1UdEwEB/wQC
# MAAwDQYJKoZIhvcNAQELBQADggIBANPAhp5gl2m66jB8Ft1vCSC2ij4PKMQ3Mwkv
# 1gtZoMKnS/NCi9jl8WpooSMth39ze0LyDrQY8nUO6CPX/LDGkWuf/f+HUoRNw2wS
# seCFw/yvQJBzmmTRzuQqdhkAIM99kD6RCo60WQxskrhOCqx8LNFl3pcsIqonSvok
# /GC1+KmwUKdV5Wuag3jkH1LLOJ5x8CnyTmMWSPsx9aVY+mIzvYuLuZIdAybGYUW7
# sOLRTzhUiGBLlxL56n1dFqzF6IgPePpZHWMwyhIKQLgbkkYNFectTeTehws5v1YZ
# f/tjw2TsVJFEB4UIS1mHnBPiCNMt+/f33jH8vfT6fxVV5FIBXqvr0ruDVseZb/Jm
# xdBfjt2AzMHk3d3l2Ar+06vo4gAFbLFtULTjV9d3jjP4YCGix5Q2UwHvkBjfB46e
# ChIEjkbWnjmkFdZPpvILRVL9lhTa1kKwBzbTnW5KSnxA5CtB3gShY7U8rYmZfPrS
# 4gBBTutdhD8nd4roDdlJZPVMIvcqeRrdb2cx8WGohV8/MCOPeEUvsMiuOwFroreG
# 6ecUwbdjCNkzx2o3hXaHyfolUj4n+iQuealnn4KKfDp7eYDabu9zWbv6UN3D//Ug
# DlYdiQWzsKF0GqJQeqeq0Lga63zr7nOSPzn82AF7QfdYkccoLnD3y2iJObDrAsY0
# KjiW7SylMYIaaTCCGmUCAQEwfTBpMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGln
# aUNlcnQsIEluYy4xQTA/BgNVBAMTOERpZ2lDZXJ0IFRydXN0ZWQgRzQgQ29kZSBT
# aWduaW5nIFJTQTQwOTYgU0hBMzg0IDIwMjEgQ0ExAhAIV5dCqVO62Q1CN6Pz44xe
# MA0GCWCGSAFlAwQCAQUAoHwwEAYKKwYBBAGCNwIBDDECMAAwGQYJKoZIhvcNAQkD
# MQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJ
# KoZIhvcNAQkEMSIEIHnkCrSs5pLed3/n55sP0H9BynfRK9/fUEAKctbQ4s+DMA0G
# CSqGSIb3DQEBAQUABIICAFqddxQ8XY1xXkV3HHrZkixXXPVvwUbK9dBItUrMU79I
# 8qsHmbBGek4P+YynTH371j5y4PSqqHHWFyOquOnw643Dpb1WXvVtyuipyorFLkJL
# UXmRy8hQHg+s7CD09HoUoCPPUCq8tRDW5iEK0gySLt5IPyTA92Zp+nWD0OxX4DIf
# bCrZCPjnlU1rntzgB/IRBrxsoyh/heN2p631Ujk5rLb3kATsvZlRxEuURC70X22a
# JHKMf9Q1Xb5Iy67dMfKdBdLQT4ocp6K9PoB19BpofOq/RQYf4B+JW8mlYOOhDAS9
# ZyWXTG3r+hJyxNeOVRrr1YbnD3bqg1le2EiUQFmK1W3WPZ8eUFKE7KW3Gs4ivMHK
# YV1Z8511sjRGqhbHhcKEvjrjhV7GFvlZPLK+zH2sfdAIfhTdyJ9D+oAHlYv2A5i0
# Bz15JlMsVFMEKyrEsuiesfkedW+r89hjoWVNrgPH6xalXXxXM/YDzjFK4eawCfmA
# BKmxYSXPnO+5ksP6u0v4sp89JsRlhrhQnOknjPiJjuI7VMmORjdKsmUXudhtaOjq
# drdYq/Fw1JU6DTraaF+yq2RQH1dyDvL8LyaKSsW7hmBp49+xdZmQhnil8/gnixp/
# VfffV8lu7jbNaAz/IegeSEpQwbx4rv10pH88WtYxBf4howCCLCIubIolzyWh+Bjp
# oYIXPzCCFzsGCisGAQQBgjcDAwExghcrMIIXJwYJKoZIhvcNAQcCoIIXGDCCFxQC
# AQMxDzANBglghkgBZQMEAgEFADB3BgsqhkiG9w0BCRABBKBoBGYwZAIBAQYJYIZI
# AYb9bAcBMDEwDQYJYIZIAWUDBAIBBQAEIM14+NgFiLLpfk4Fi/Xt+uxregpRinMa
# kmTKO80EcBIuAhAVpd3yB3bp3hY485zIBWw6GA8yMDIzMTAyMTE4MjIxMFqgghMJ
# MIIGwjCCBKqgAwIBAgIQBUSv85SdCDmmv9s/X+VhFjANBgkqhkiG9w0BAQsFADBj
# MQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xOzA5BgNVBAMT
# MkRpZ2lDZXJ0IFRydXN0ZWQgRzQgUlNBNDA5NiBTSEEyNTYgVGltZVN0YW1waW5n
# IENBMB4XDTIzMDcxNDAwMDAwMFoXDTM0MTAxMzIzNTk1OVowSDELMAkGA1UEBhMC
# VVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMSAwHgYDVQQDExdEaWdpQ2VydCBU
# aW1lc3RhbXAgMjAyMzCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAKNT
# RYcdg45brD5UsyPgz5/X5dLnXaEOCdwvSKOXejsqnGfcYhVYwamTEafNqrJq3RAp
# ih5iY2nTWJw1cb86l+uUUI8cIOrHmjsvlmbjaedp/lvD1isgHMGXlLSlUIHyz8sH
# pjBoyoNC2vx/CSSUpIIa2mq62DvKXd4ZGIX7ReoNYWyd/nFexAaaPPDFLnkPG2ZS
# 48jWPl/aQ9OE9dDH9kgtXkV1lnX+3RChG4PBuOZSlbVH13gpOWvgeFmX40QrStWV
# zu8IF+qCZE3/I+PKhu60pCFkcOvV5aDaY7Mu6QXuqvYk9R28mxyyt1/f8O52fTGZ
# ZUdVnUokL6wrl76f5P17cz4y7lI0+9S769SgLDSb495uZBkHNwGRDxy1Uc2qTGaD
# iGhiu7xBG3gZbeTZD+BYQfvYsSzhUa+0rRUGFOpiCBPTaR58ZE2dD9/O0V6MqqtQ
# FcmzyrzXxDtoRKOlO0L9c33u3Qr/eTQQfqZcClhMAD6FaXXHg2TWdc2PEnZWpST6
# 18RrIbroHzSYLzrqawGw9/sqhux7UjipmAmhcbJsca8+uG+W1eEQE/5hRwqM/vC2
# x9XH3mwk8L9CgsqgcT2ckpMEtGlwJw1Pt7U20clfCKRwo+wK8REuZODLIivK8SgT
# IUlRfgZm0zu++uuRONhRB8qUt+JQofM604qDy0B7AgMBAAGjggGLMIIBhzAOBgNV
# HQ8BAf8EBAMCB4AwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcD
# CDAgBgNVHSAEGTAXMAgGBmeBDAEEAjALBglghkgBhv1sBwEwHwYDVR0jBBgwFoAU
# uhbZbU2FL3MpdpovdYxqII+eyG8wHQYDVR0OBBYEFKW27xPn783QZKHVVqllMaPe
# 1eNJMFoGA1UdHwRTMFEwT6BNoEuGSWh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9E
# aWdpQ2VydFRydXN0ZWRHNFJTQTQwOTZTSEEyNTZUaW1lU3RhbXBpbmdDQS5jcmww
# gZAGCCsGAQUFBwEBBIGDMIGAMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdp
# Y2VydC5jb20wWAYIKwYBBQUHMAKGTGh0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNv
# bS9EaWdpQ2VydFRydXN0ZWRHNFJTQTQwOTZTSEEyNTZUaW1lU3RhbXBpbmdDQS5j
# cnQwDQYJKoZIhvcNAQELBQADggIBAIEa1t6gqbWYF7xwjU+KPGic2CX/yyzkzepd
# IpLsjCICqbjPgKjZ5+PF7SaCinEvGN1Ott5s1+FgnCvt7T1IjrhrunxdvcJhN2hJ
# d6PrkKoS1yeF844ektrCQDifXcigLiV4JZ0qBXqEKZi2V3mP2yZWK7Dzp703DNiY
# dk9WuVLCtp04qYHnbUFcjGnRuSvExnvPnPp44pMadqJpddNQ5EQSviANnqlE0Pjl
# SXcIWiHFtM+YlRpUurm8wWkZus8W8oM3NG6wQSbd3lqXTzON1I13fXVFoaVYJmoD
# Rd7ZULVQjK9WvUzF4UbFKNOt50MAcN7MmJ4ZiQPq1JE3701S88lgIcRWR+3aEUuM
# MsOI5ljitts++V+wQtaP4xeR0arAVeOGv6wnLEHQmjNKqDbUuXKWfpd5OEhfysLc
# PTLfddY2Z1qJ+Panx+VPNTwAvb6cKmx5AdzaROY63jg7B145WPR8czFVoIARyxQM
# fq68/qTreWWqaNYiyjvrmoI1VygWy2nyMpqy0tg6uLFGhmu6F/3Ed2wVbK6rr3M6
# 6ElGt9V/zLY4wNjsHPW2obhDLN9OTH0eaHDAdwrUAuBcYLso/zjlUlrWrBciI070
# 7NMX+1Br/wd3H3GXREHJuEbTbDJ8WC9nR2XlG3O2mflrLAZG70Ee8PBf4NvZrZCA
# RK+AEEGKMIIGrjCCBJagAwIBAgIQBzY3tyRUfNhHrP0oZipeWzANBgkqhkiG9w0B
# AQsFADBiMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYD
# VQQLExB3d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQDExhEaWdpQ2VydCBUcnVzdGVk
# IFJvb3QgRzQwHhcNMjIwMzIzMDAwMDAwWhcNMzcwMzIyMjM1OTU5WjBjMQswCQYD
# VQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xOzA5BgNVBAMTMkRpZ2lD
# ZXJ0IFRydXN0ZWQgRzQgUlNBNDA5NiBTSEEyNTYgVGltZVN0YW1waW5nIENBMIIC
# IjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAxoY1BkmzwT1ySVFVxyUDxPKR
# N6mXUaHW0oPRnkyibaCwzIP5WvYRoUQVQl+kiPNo+n3znIkLf50fng8zH1ATCyZz
# lm34V6gCff1DtITaEfFzsbPuK4CEiiIY3+vaPcQXf6sZKz5C3GeO6lE98NZW1Oco
# LevTsbV15x8GZY2UKdPZ7Gnf2ZCHRgB720RBidx8ald68Dd5n12sy+iEZLRS8nZH
# 92GDGd1ftFQLIWhuNyG7QKxfst5Kfc71ORJn7w6lY2zkpsUdzTYNXNXmG6jBZHRA
# p8ByxbpOH7G1WE15/tePc5OsLDnipUjW8LAxE6lXKZYnLvWHpo9OdhVVJnCYJn+g
# GkcgQ+NDY4B7dW4nJZCYOjgRs/b2nuY7W+yB3iIU2YIqx5K/oN7jPqJz+ucfWmyU
# 8lKVEStYdEAoq3NDzt9KoRxrOMUp88qqlnNCaJ+2RrOdOqPVA+C/8KI8ykLcGEh/
# FDTP0kyr75s9/g64ZCr6dSgkQe1CvwWcZklSUPRR8zZJTYsg0ixXNXkrqPNFYLwj
# jVj33GHek/45wPmyMKVM1+mYSlg+0wOI/rOP015LdhJRk8mMDDtbiiKowSYI+RQQ
# EgN9XyO7ZONj4KbhPvbCdLI/Hgl27KtdRnXiYKNYCQEoAA6EVO7O6V3IXjASvUae
# tdN2udIOa5kM0jO0zbECAwEAAaOCAV0wggFZMBIGA1UdEwEB/wQIMAYBAf8CAQAw
# HQYDVR0OBBYEFLoW2W1NhS9zKXaaL3WMaiCPnshvMB8GA1UdIwQYMBaAFOzX44LS
# cV1kTN8uZz/nupiuHA9PMA4GA1UdDwEB/wQEAwIBhjATBgNVHSUEDDAKBggrBgEF
# BQcDCDB3BggrBgEFBQcBAQRrMGkwJAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRp
# Z2ljZXJ0LmNvbTBBBggrBgEFBQcwAoY1aHR0cDovL2NhY2VydHMuZGlnaWNlcnQu
# Y29tL0RpZ2lDZXJ0VHJ1c3RlZFJvb3RHNC5jcnQwQwYDVR0fBDwwOjA4oDagNIYy
# aHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZFJvb3RHNC5j
# cmwwIAYDVR0gBBkwFzAIBgZngQwBBAIwCwYJYIZIAYb9bAcBMA0GCSqGSIb3DQEB
# CwUAA4ICAQB9WY7Ak7ZvmKlEIgF+ZtbYIULhsBguEE0TzzBTzr8Y+8dQXeJLKftw
# ig2qKWn8acHPHQfpPmDI2AvlXFvXbYf6hCAlNDFnzbYSlm/EUExiHQwIgqgWvalW
# zxVzjQEiJc6VaT9Hd/tydBTX/6tPiix6q4XNQ1/tYLaqT5Fmniye4Iqs5f2MvGQm
# h2ySvZ180HAKfO+ovHVPulr3qRCyXen/KFSJ8NWKcXZl2szwcqMj+sAngkSumScb
# qyQeJsG33irr9p6xeZmBo1aGqwpFyd/EjaDnmPv7pp1yr8THwcFqcdnGE4AJxLaf
# zYeHJLtPo0m5d2aR8XKc6UsCUqc3fpNTrDsdCEkPlM05et3/JWOZJyw9P2un8WbD
# Qc1PtkCbISFA0LcTJM3cHXg65J6t5TRxktcma+Q4c6umAU+9Pzt4rUyt+8SVe+0K
# XzM5h0F4ejjpnOHdI/0dKNPH+ejxmF/7K9h+8kaddSweJywm228Vex4Ziza4k9Tm
# 8heZWcpw8De/mADfIBZPJ/tgZxahZrrdVcA6KYawmKAr7ZVBtzrVFZgxtGIJDwq9
# gdkT/r+k0fNX2bwE+oLeMt8EifAAzV3C+dAjfwAL5HYCJtnwZXZCpimHCUcr5n8a
# pIUP/JiW9lVUKx+A+sDyDivl1vupL0QVSucTDh3bNzgaoSv27dZ8/DCCBY0wggR1
# oAMCAQICEA6bGI750C3n79tQ4ghAGFowDQYJKoZIhvcNAQEMBQAwZTELMAkGA1UE
# BhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2lj
# ZXJ0LmNvbTEkMCIGA1UEAxMbRGlnaUNlcnQgQXNzdXJlZCBJRCBSb290IENBMB4X
# DTIyMDgwMTAwMDAwMFoXDTMxMTEwOTIzNTk1OVowYjELMAkGA1UEBhMCVVMxFTAT
# BgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTEh
# MB8GA1UEAxMYRGlnaUNlcnQgVHJ1c3RlZCBSb290IEc0MIICIjANBgkqhkiG9w0B
# AQEFAAOCAg8AMIICCgKCAgEAv+aQc2jeu+RdSjwwIjBpM+zCpyUuySE98orYWcLh
# Kac9WKt2ms2uexuEDcQwH/MbpDgW61bGl20dq7J58soR0uRf1gU8Ug9SH8aeFaV+
# vp+pVxZZVXKvaJNwwrK6dZlqczKU0RBEEC7fgvMHhOZ0O21x4i0MG+4g1ckgHWMp
# Lc7sXk7Ik/ghYZs06wXGXuxbGrzryc/NrDRAX7F6Zu53yEioZldXn1RYjgwrt0+n
# MNlW7sp7XeOtyU9e5TXnMcvak17cjo+A2raRmECQecN4x7axxLVqGDgDEI3Y1Dek
# LgV9iPWCPhCRcKtVgkEy19sEcypukQF8IUzUvK4bA3VdeGbZOjFEmjNAvwjXWkmk
# wuapoGfdpCe8oU85tRFYF/ckXEaPZPfBaYh2mHY9WV1CdoeJl2l6SPDgohIbZpp0
# yt5LHucOY67m1O+SkjqePdwA5EUlibaaRBkrfsCUtNJhbesz2cXfSwQAzH0clcOP
# 9yGyshG3u3/y1YxwLEFgqrFjGESVGnZifvaAsPvoZKYz0YkH4b235kOkGLimdwHh
# D5QMIR2yVCkliWzlDlJRR3S+Jqy2QXXeeqxfjT/JvNNBERJb5RBQ6zHFynIWIgnf
# fEx1P2PsIV/EIFFrb7GrhotPwtZFX50g/KEexcCPorF+CiaZ9eRpL5gdLfXZqbId
# 5RsCAwEAAaOCATowggE2MA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYEFOzX44LS
# cV1kTN8uZz/nupiuHA9PMB8GA1UdIwQYMBaAFEXroq/0ksuCMS1Ri6enIZ3zbcgP
# MA4GA1UdDwEB/wQEAwIBhjB5BggrBgEFBQcBAQRtMGswJAYIKwYBBQUHMAGGGGh0
# dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBDBggrBgEFBQcwAoY3aHR0cDovL2NhY2Vy
# dHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENBLmNydDBFBgNV
# HR8EPjA8MDqgOKA2hjRodHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRB
# c3N1cmVkSURSb290Q0EuY3JsMBEGA1UdIAQKMAgwBgYEVR0gADANBgkqhkiG9w0B
# AQwFAAOCAQEAcKC/Q1xV5zhfoKN0Gz22Ftf3v1cHvZqsoYcs7IVeqRq7IviHGmlU
# Iu2kiHdtvRoU9BNKei8ttzjv9P+Aufih9/Jy3iS8UgPITtAq3votVs/59PesMHqa
# i7Je1M/RQ0SbQyHrlnKhSLSZy51PpwYDE3cnRNTnf+hZqPC/Lwum6fI0POz3A8eH
# qNJMQBk1RmppVLC4oVaO7KTVPeix3P0c2PR3WlxUjG/voVA9/HYJaISfb8rbII01
# YBwCA8sgsKxYoA5AY8WYIsGyWfVVa88nq2x2zm8jLfR+cWojayL/ErhULSd+2DrZ
# 8LaHlv1b0VysGMNNn3O3AamfV6peKOK5lDGCA3YwggNyAgEBMHcwYzELMAkGA1UE
# BhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMTswOQYDVQQDEzJEaWdpQ2Vy
# dCBUcnVzdGVkIEc0IFJTQTQwOTYgU0hBMjU2IFRpbWVTdGFtcGluZyBDQQIQBUSv
# 85SdCDmmv9s/X+VhFjANBglghkgBZQMEAgEFAKCB0TAaBgkqhkiG9w0BCQMxDQYL
# KoZIhvcNAQkQAQQwHAYJKoZIhvcNAQkFMQ8XDTIzMTAyMTE4MjIxMFowKwYLKoZI
# hvcNAQkQAgwxHDAaMBgwFgQUZvArMsLCyQ+CXc6qisnGTxmcz0AwLwYJKoZIhvcN
# AQkEMSIEILpZslTHrVnqCheMC7k0NhdpZwOj18gPL/j+Iw8DUqs4MDcGCyqGSIb3
# DQEJEAIvMSgwJjAkMCIEINL25G3tdCLM0dRAV2hBNm+CitpVmq4zFq9NGprUDHgo
# MA0GCSqGSIb3DQEBAQUABIICAGNUg9oca+chMO1XBAEdB/4IL2t7bB/TESlhAO2P
# qJdCMznWp4ZS0mv7Fw5G14qegOsOOqYzyT535EJWWmZgF2yZlcSRt4xVpwAvi5Ih
# M46jcPzBuLpGnolxpjEK0jmr0BovuyjZcergI1Jbe4DWqk5nbBcp+RxyquR39qnj
# RcZqQxLDNnNe6EV+hgnlzT/NNkO9IVm4BOSXMGrIEN47K2w3ESufhHOCgdtc+l2s
# wyHfRuPM7gOI3+Gn9Nlj1BZPPAXuh+3+qICa0oYeD3lXvMVh+OaN7C83XEqPlzsN
# 6K2CJ6evpfpZiCUDsARIvPOzNfAkZfvR/1qZwrwX55a755+297X2u3do3wAvKxd3
# D2YbYzaQuGMeBDPkdKHapgmj2ldF2O6u/YVT/AbVhIbXJUW7nfxw3o5oo27dyzS1
# gREp+WnVj3jizYqn8YurymI+5dDsMS8tVHBAPnVViyP2wnBEHOmvXnbUOpNHgxtC
# YBxcbe5/lv26wRNHK3wysHyI0pHKDDrBrO0Mbj3U496Dp1O2T72vkh3tJUpZxP2Y
# ia92UcM5W3fwh6FtiTw4/kpg1EQIw2RDuVvbGHI2xvTu4vqaJDjMBFQTDGrHPzEW
# 2yPs3Ph/6y9KQEkNxgy0Qq2c6JmFYiMLjq/xxm3+52Kp+37PV2dPpgpjMnHM1r/y
# c8IU
# SIG # End signature block