functions/Invoke-Installer.ps1

function Invoke-Installer {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName)]
        [string]$File,
        [Parameter(Mandatory = $True, ValueFromPipelineByPropertyName)]
        [string]$OutDir
    )
    begin {
        $versionMinimum = [Version]'5.0'
        if ($versionMinimum -gt $PSVersionTable.PSVersion)
        { throw "This script requires PowerShell $versionMinimum" }

        # Get function definition files.
        $Functions = @( Get-ChildItem -Path "$PSScriptRoot\installer" -Filter *.ps1 -ErrorAction SilentlyContinue )

        # Dot source the files
        foreach ($Import in @($Functions)) {
            try {
                . $Import.fullname
            }
            catch {
                Write-Error -Message "Failed to import function $($Import.fullname): $_"
            }
        }

        function True {return "1"}
        function False {return "0"}
        function NullableString {
            param([string] $Text)
            if ([string]::IsNullOrEmpty($Text)) {
                return [nullstring]::value
            }
            else {
                return $Text -replace "‘‘", """" -replace "’’", """" -replace "’", "'"
            }
        }        
    }
    process {
        try {
            Test-Path $File | Out-Null;
            $InputFile = Get-Item $File
            $FileDirectory = Split-Path $File -Parent
            $Msg = "INSTALLER - $(Split-Path $File -Leaf)"; Write-Host $Msg -ForegroundColor Magenta; Write-Verbose $Msg; Write-Log $Msg;
        }
        catch {
            $Msg = "$(" " * 8)Unable to find any hcx files."; Write-Host $Msg -ForegroundColor Red; Write-Verbose $Msg; Write-Log $Msg 'error';
        }

        if ($InputFile.Extension -eq '.hcx') {
            try {
                $Msg = "$(" " * 4)Unzipping hcx file..."; Write-Host $Msg -ForegroundColor Gray; Write-Verbose $Msg; Write-Log $Msg;
                
                Copy-Item -Path $File -Destination $File.Replace('.hcx', '.zip') -Force | Out-Null
                $ZipFile = $File.Replace('.hcx', '.zip')
                
                $OutBin = "$($FileDirectory)\$((Split-Path $File -Leaf).Replace('.hcx', '_bin'))"
                $Zipoutdir = "$($OutBin)\$((Split-Path $File -Leaf).Replace('.hcx', '_zip'))"
                if (Test-Path $OutBin) {
                    Remove-Item $OutBin -Force -Recurse | Out-Null
                }
                If (!(Test-Path $Zipoutdir)) {
                    New-Item -ItemType Directory -Force -Path $Zipoutdir -ErrorAction Stop | Out-Null
                }
                Unzip -file $ZipFile -destination $Zipoutdir
                Remove-Item $ZipFile -Force | Out-Null
            }
            catch {
                $Msg = "$(" " * 8)Unable to unzip file."; Write-Host $Msg -ForegroundColor Red; Write-Verbose $Msg; Write-Log $Msg 'error';
            }
            $Msg = "$(" " * 4)Getting sam json object..."; Write-Host $Msg -ForegroundColor Gray; Write-Verbose $Msg; Write-Log $Msg;
            $DataFile = Get-ChildItem $Zipoutdir -Recurse | Where-Object { $_.Extension -eq ".sam" }
        }
        elseif ($InputFile.Extension -eq '.sm') {
            $Msg = "$(" " * 4)Getting sm json object..."; Write-Host $Msg -ForegroundColor Gray; Write-Verbose $Msg; Write-Log $Msg;
            $DataFile = $InputFile
        }

        $RawContent = (Get-Content $DataFile.FullName | Select-Object -Skip 1);
        try {
            $jsonSettings = New-Object Newtonsoft.Json.JsonSerializerSettings
            $jsonSettings.TypeNameHandling = 'Objects'
            $jsonSettings.PreserveReferencesHandling = 'Objects'
            $RawData = [Newtonsoft.Json.JsonConvert]::DeserializeObject($RawContent, $jsonSettings)
            if (Test-Path $OutBin) {
                Remove-Item $OutBin -Recurse -Force
            }
        }
        catch {
            $ErrorMessage = $_.Exception.Message
            $Msg = "$(" " * 8)Unable to deserialize json object :( --> $ErrorMessage"; Write-Host $Msg -ForegroundColor Red; Write-Verbose $Msg; Write-Log $Msg 'error';
            Exit
        }

        $Msg = "$(" " * 4)Formatting raw data to MDS format..."; Write-Host $Msg -ForegroundColor Gray; Write-Verbose $Msg; Write-Log $Msg;

        $id = [Id]::new()
        $ids = @{}
        function GetId($contentId) {
            if (!$ids.ContainsKey($contentId)) {
                $ids.Add($contentId, $id.GetNewId())
            }
            return $ids.Item($contentId)
        }            

        $DataMart = [DataMart]::new()
        $DataMart.Id = GetId($RawData.ContentId)
        $DataMart.ContentId = $RawData.ContentId
        $DataMart.Name = NullableString($RawData.DataMartNM)
        $DataMart.DataMartType = ($RawData.DataMartTypeDSC)
        $DataMart.Description = NullableString($RawData.DescriptionTXT)
        $DataMart.SqlAgentProxyName = NullableString($RawData.SqlAgentProxyName)
        $DataMart.SqlCredentialName = NullableString($RawData.SqlCredentialName)
        $DataMart.DefaultEngineVersion = $RawData.DefaultEngineVersionNumber
        $DataMart.SystemName = NullableString($RawData.SystemNM)
        $DataMart.DataStewardFullName = NullableString($RawData.DataStewardFullNM)
        $DataMart.DataStewardEmail = NullableString($RawData.DataStewardEmailTXT)
        $DataMart.Version = NullableString($RawData.VersionText)
        $DataMart.IsHidden = NullableString($RawData.IsHidden)

        foreach ($RawConnection in $RawData.Connections) {
            $Connection = [Connection]::new()
            $Connection.Id = GetId($RawConnection.ContentId)
            $Connection.ContentId = $RawConnection.ContentId
            $Connection.SystemName = NullableString($RawConnection.SystemName)
            $Connection.Description = NullableString($RawConnection.Description)
            $Connection.DataSystemTypeCode = NullableString($RawConnection.DataSystemTypeCode)
            $Connection.DataSystemVersion = NullableString($RawConnection.DataSystemTypeVersion)
            $Connection.SystemVendorName = NullableString($RawConnection.SystemVendorName)
            $Connection.SystemVersion = NullableString($RawConnection.SystemVersion)

            foreach ($RawAttributeValue in $RawConnection.AttributeValues) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = $RawAttributeValue.AttributeName
                $AttributeValue.AttributeValue = $RawAttributeValue.LongTextValue + $RawAttributeValue.TextValue + $RawAttributeValue.NumberValue
                $Connection.AttributeValues += $AttributeValue
            }
            if (!$Connection.AttributeValues) {$Connection.AttributeValues = @()}
            $DataMart.Connections += $Connection
        }
        if (!$DataMart.Connections) {$DataMart.Connections = @()}

        foreach ($RawAttributeValue in $RawData.AttributeValues) {
            $AttributeValue = [ObjectAttributeValue]::new()
            $AttributeValue.AttributeName = $RawAttributeValue.AttributeName
            $AttributeValue.AttributeValue = $RawAttributeValue.LongTextValue + $RawAttributeValue.TextValue + $RawAttributeValue.NumberValue
            $DataMart.AttributeValues += $AttributeValue
        }
        if (!$DataMart.AttributeValues) {$DataMart.AttributeValues = @()}

        foreach ($RawEntity in $RawData.Tables) {
            $Entity = [Entity]::new()
            $Entity.Id = GetId($RawEntity.ContentId)
            $Entity.ContentId = $RawEntity.ContentId
            $Entity.ConnectionId = GetId($RawEntity.DestinationConnection.ContentId)
            $Entity.BusinessDescription = NullableString($RawEntity.DescriptionTXT)
            $Entity.EntityName = NullableString($RawEntity.ViewName)
            $Entity.PersistenceType = "Database"
            $Entity.IsPublic = $RawEntity.IsPublic
            $Entity.AllowsDataEntry = $RawEntity.AllowsDataEntry
            $Entity.RecordCountMismatchThreshold = $RawEntity.RowCountMismatchThreshold
            $Entity.LastSuccessfulLoadTimestamp = $RawEntity.SuccessfulLastRunDate
            $Entity.LastModifiedTimestamp = $null
            $Entity.LastDeployedTimestamp = $null

            if ($RawEntity.DatabaseNM) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = "DatabaseName"
                $AttributeValue.AttributeValue = $RawEntity.DatabaseNM
                $Entity.AttributeValues += $AttributeValue
            }
            if ($RawEntity.SchemaNM) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = "SchemaName"
                $AttributeValue.AttributeValue = $RawEntity.SchemaNM
                $Entity.AttributeValues += $AttributeValue
            }
            if ($RawEntity.TableNM) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = "TableName"
                $AttributeValue.AttributeValue = $RawEntity.TableNM
                $Entity.AttributeValues += $AttributeValue
            }
            if ($RawEntity.ViewName) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = "ViewName"
                $AttributeValue.AttributeValue = $RawEntity.ViewName
                $Entity.AttributeValues += $AttributeValue
            }
            if ($RawEntity.FileGroup) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = "FileGroupNumber"
                $AttributeValue.AttributeValue = $RawEntity.FileGroup
                $Entity.AttributeValues += $AttributeValue
            }
            if (($RawEntity.IsPersisted | Measure-Object).Count) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = "PersistedFlag"
                $AttributeValue.AttributeValue = (. $RawEntity.IsPersisted) <# takes value True/False and runs the True/False function; which converts to 1/0 #>
                $Entity.AttributeValues += $AttributeValue
            }

            foreach ($RawAttributeValue in $RawEntity.AttributeValues) {
                $AttributeValue = [ObjectAttributeValue]::new()
                $AttributeValue.AttributeName = $RawAttributeValue.AttributeName
                $AttributeValue.AttributeValue = $RawAttributeValue.LongTextValue + $RawAttributeValue.TextValue + $RawAttributeValue.NumberValue
                $Entity.AttributeValues += $AttributeValue
            }
            if (!$Entity.AttributeValues) {$Entity.AttributeValues = @()}

            foreach ($RawBinding in $RawEntity.FedByBindings) {
                $Binding = [Binding]::new()
                $Binding.Id = GetId($RawBinding.ContentId)
                $Binding.ContentId = $RawBinding.ContentId
                $Binding.Name = NullableString($RawBinding.BindingName)
                $Binding.DestinationEntityId = $Entity.Id
                $Binding.SourceConnectionId = GetId($RawBinding.SourceConnection.ContentId)
                $Binding.BindingType = ($RawBinding.GetType().Name -replace "Binding", "")
                $Binding.Classification = NullableString($RawBinding.ClassificationCode)
                $Binding.Description = NullableString($RawBinding.BindingDescription)
                $Binding.LoadTypeCode = NullableString($RawBinding.LoadType)
                $Binding.Status = NullableString($RawBinding.BindingStatus)
                $Binding.GroupingColumn = NullableString($RawBinding.GroupingColumn)
                $Binding.GroupingFormat = NullableString($RawBinding.GroupingFormat)
                $Binding.GrainName = NullableString($RawBinding.GrainName)

                foreach ($RawAttributeValue in $RawBinding.AttributeValues) {
                    $AttributeValue = [ObjectAttributeValue]::new()
                    $AttributeValue.AttributeName = $RawAttributeValue.AttributeName
                    $AttributeValue.AttributeValue = $RawAttributeValue.LongTextValue + $RawAttributeValue.TextValue + $RawAttributeValue.NumberValue
                    $Binding.AttributeValues += $AttributeValue
                }
                if (!$Binding.AttributeValues) {$Binding.AttributeValues = @()}

                $DataMart.Bindings += $Binding
            }
            if (!$DataMart.Bindings) {$DataMart.Bindings = @()}
    
            foreach ($RawField in $RawEntity.Columns) {
                $Field = [Field]::new()
                $Field.Id = GetId($RawField.ContentId)
                $Field.ContentId = $RawField.ContentId
                $Field.FieldName = NullableString($RawField.ColumnNM)
                $Field.BusinessDescription = NullableString($RawField.DescriptionTXT)
                $Field.DataType = NullableString($RawField.DataTypeDSC)
                $Field.DefaultValue = NullableString($RawField.DefaultValueTXT)
                $Field.DataSensitivity = NullableString($RawField.DataSensitivityCD)
                $Field.Ordinal = $RawField.Ordinal
                $Field.Status = NullableString($RawField.Status)
                $Field.ExampleData = NullableString($RawField.ExampleDataTXT)
                # $Field.ExampleDataUpdatetimestamp = $RawField.ExampleDataUpdateDate
                $Field.IsPrimaryKey = $RawField.IsPrimaryKeyValue
                $Field.IsNullable = $RawField.IsNullableValue
                $Field.IsAutoIncrement = $RawField.AutoIncrement
                $Field.ExcludeFromBaseView = $RawField.ExcludeFromBaseViewValue
                $Field.IsSystemField = $RawField.IsSystemColumnValue

                foreach ($RawAttributeValue in $RawField.AttributeValues) {
                    $AttributeValue = [ObjectAttributeValue]::new()
                    $AttributeValue.AttributeName = $RawAttributeValue.AttributeName
                    $AttributeValue.AttributeValue = $RawAttributeValue.LongTextValue + $RawAttributeValue.TextValue + $RawAttributeValue.NumberValue
                    $Field.AttributeValues += $AttributeValue
                }
                if (!$Field.AttributeValues) {$Field.AttributeValues = @()}

                $Entity.Fields += $Field
            }
            if (!$Entity.Fields) {$Entity.Fields = @()}

            foreach ($RawIndex in $RawEntity.Indexes) {
                $Index = [Index]::new()
                $Index.Id = GetId($RawIndex.ContentId)
                $Index.ContentId = $RawIndex.ContentId
                $Index.IndexName = NullableString($RawIndex.IndexName)
                $Index.IsUnique = $RawIndex.IsUnique
                $Index.IsActive = $RawIndex.IsActive
                $Index.IndexTypeCode = NullableString($RawIndex.IndexTypeCode)
                $Index.IsColumnStore = $RawIndex.IsColumnStore
                $Index.IsCapSystem = $RawIndex.IsCapSystem
                $Index.LastModifiedTimestamp = $RawIndex.LastModifiedTimestamp
                $Index.LastDeployedTimestamp = $RawIndex.LastDeployedTimestamp

                foreach ($RawIndexField in $RawIndex.IndexColumns) {
                    $IndexField = [IndexField]::new()
                    $IndexField.Id = GetId("$($RawIndexField.Index.ContentId)_$($RawIndexField.Column.ContentId)")
                    $IndexField.IndexId = GetId($RawIndexField.Index.ContentId)
                    $IndexField.FieldId = GetId($RawIndexField.Column.ContentId)
                    $IndexField.Ordinal = $RawIndexField.Ordinal
                    $IndexField.IsDescending = $RawIndexField.IsDescending
                    $IndexField.IsCovering = $RawIndexField.IsCovering
                    $Index.IndexFields += $IndexField
                }
                if (!$Index.IndexFields) {$Index.IndexFields = @()}
                $Entity.Indexes += $Index
            }
            if (!$Entity.Indexes) {$Entity.Indexes = @()}
            $DataMart.Entities += $Entity
        }
        if (!$DataMart.Entities) {$DataMart.Entities = @()}

        if (!$OutVar) {
            New-Directory -Dir (Split-Path $OutDir -Parent);
            [Newtonsoft.Json.JsonConvert]::SerializeObject($DataMart, [Newtonsoft.Json.Formatting]::Indented) | Out-File "$($OutDir).json" -Force -Encoding default
            $Msg = "$(" " * 4)Output to file $("$($OutDir).json")"; Write-Host $Msg -ForegroundColor Cyan; Write-Verbose $Msg; Write-Log $Msg;
        }
        else {
            $Msg = "$(" " * 4)Output to variable"; Write-Host $Msg -ForegroundColor Cyan; Write-Verbose $Msg; Write-Log $Msg;
        }

        $Msg = "Success!`r`n"; Write-Host $Msg -ForegroundColor Green; Write-Verbose $Msg; Write-Log $Msg;
        $Output = New-Object PSObject
        $Output | Add-Member -Type NoteProperty -Name RawData -Value $DataMart
        $Output | Add-Member -Type NoteProperty -Name Outdir -Value $OutDir
        return $Output
    }
}

# SIG # Begin signature block
# MIIaxQYJKoZIhvcNAQcCoIIatjCCGrICAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCtqIlzJYbKLiJU
# NJZo6DXmG1i9L+awhsQEFhWuCmQPTqCCCqMwggUwMIIEGKADAgECAhAECRgbX9W7
# ZnVTQ7VvlVAIMA0GCSqGSIb3DQEBCwUAMGUxCzAJBgNVBAYTAlVTMRUwEwYDVQQK
# EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xJDAiBgNV
# BAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAeFw0xMzEwMjIxMjAwMDBa
# Fw0yODEwMjIxMjAwMDBaMHIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2Vy
# dCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xMTAvBgNVBAMTKERpZ2lD
# ZXJ0IFNIQTIgQXNzdXJlZCBJRCBDb2RlIFNpZ25pbmcgQ0EwggEiMA0GCSqGSIb3
# DQEBAQUAA4IBDwAwggEKAoIBAQD407Mcfw4Rr2d3B9MLMUkZz9D7RZmxOttE9X/l
# qJ3bMtdx6nadBS63j/qSQ8Cl+YnUNxnXtqrwnIal2CWsDnkoOn7p0WfTxvspJ8fT
# eyOU5JEjlpB3gvmhhCNmElQzUHSxKCa7JGnCwlLyFGeKiUXULaGj6YgsIJWuHEqH
# CN8M9eJNYBi+qsSyrnAxZjNxPqxwoqvOf+l8y5Kh5TsxHM/q8grkV7tKtel05iv+
# bMt+dDk2DZDv5LVOpKnqagqrhPOsZ061xPeM0SAlI+sIZD5SlsHyDxL0xY4PwaLo
# LFH3c7y9hbFig3NBggfkOItqcyDQD2RzPJ6fpjOp/RnfJZPRAgMBAAGjggHNMIIB
# yTASBgNVHRMBAf8ECDAGAQH/AgEAMA4GA1UdDwEB/wQEAwIBhjATBgNVHSUEDDAK
# BggrBgEFBQcDAzB5BggrBgEFBQcBAQRtMGswJAYIKwYBBQUHMAGGGGh0dHA6Ly9v
# Y3NwLmRpZ2ljZXJ0LmNvbTBDBggrBgEFBQcwAoY3aHR0cDovL2NhY2VydHMuZGln
# aWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENBLmNydDCBgQYDVR0fBHow
# eDA6oDigNoY0aHR0cDovL2NybDQuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJl
# ZElEUm9vdENBLmNybDA6oDigNoY0aHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0Rp
# Z2lDZXJ0QXNzdXJlZElEUm9vdENBLmNybDBPBgNVHSAESDBGMDgGCmCGSAGG/WwA
# AgQwKjAoBggrBgEFBQcCARYcaHR0cHM6Ly93d3cuZGlnaWNlcnQuY29tL0NQUzAK
# BghghkgBhv1sAzAdBgNVHQ4EFgQUWsS5eyoKo6XqcQPAYPkt9mV1DlgwHwYDVR0j
# BBgwFoAUReuir/SSy4IxLVGLp6chnfNtyA8wDQYJKoZIhvcNAQELBQADggEBAD7s
# DVoks/Mi0RXILHwlKXaoHV0cLToaxO8wYdd+C2D9wz0PxK+L/e8q3yBVN7Dh9tGS
# dQ9RtG6ljlriXiSBThCk7j9xjmMOE0ut119EefM2FAaK95xGTlz/kLEbBw6RFfu6
# r7VRwo0kriTGxycqoSkoGjpxKAI8LpGjwCUR4pwUR6F6aGivm6dcIFzZcbEMj7uo
# +MUSaJ/PQMtARKUT8OZkDCUIQjKyNookAv4vcn4c10lFluhZHen6dGRrsutmQ9qz
# sIzV6Q3d9gEgzpkxYz0IGhizgZtPxpMQBvwHgfqL2vmCSfdibqFT+hKUGIUukpHq
# aGxEMrJmoecYpJpkUe8wggVrMIIEU6ADAgECAhAMMCpTLsjxo9FR9hag8ePUMA0G
# CSqGSIb3DQEBCwUAMHIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJ
# bmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xMTAvBgNVBAMTKERpZ2lDZXJ0
# IFNIQTIgQXNzdXJlZCBJRCBDb2RlIFNpZ25pbmcgQ0EwHhcNMjAwMzMxMDAwMDAw
# WhcNMjMwNTEwMTIwMDAwWjCBpzELMAkGA1UEBhMCVVMxDTALBgNVBAgTBFV0YWgx
# FzAVBgNVBAcTDlNhbHQgTGFrZSBDaXR5MR4wHAYDVQQKExVIZWFsdGggQ2F0YWx5
# c3QsIEluYy4xHjAcBgNVBAMTFUhlYWx0aCBDYXRhbHlzdCwgSW5jLjEwMC4GCSqG
# SIb3DQEJARYhYWRtaW5uaXN0cmF0b3JAaGVhbHRoY2F0YWx5c3QuY29tMIIBIjAN
# BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA2fY0HWdxDJezDOsbHp7f9u/lrrD5
# nuZ1mENMgvixlrtC/KXgBRXlcWH7ajIOKljKnWCSAZwlZy4nFGbMagKmMzohXUXg
# xo94u5nCdiBa/kgPazNGpL0AyGgX2VARMbcpm8Gdy+/uH3Kc7L91lcoGZVVBnVIt
# 1oj5iXURqmhL83TrMyYqyj3XOH0So8Y10FVLPSukocMzMqBIRgvn/7EP0iWtOjXx
# +o1wB5Ql+z9G3NCqF6CKE/Pn355XYbbmjF7BPzKoOjocHO6VU2uEflJWq1ZFb0QY
# /tAosyyLYi9kFfO1damtJfRbbsVqavwg2UeQkzhg9CpB6eSsmBXPlFHudQIDAQAB
# o4IBxTCCAcEwHwYDVR0jBBgwFoAUWsS5eyoKo6XqcQPAYPkt9mV1DlgwHQYDVR0O
# BBYEFFjfHOOIre2C4m9NCk8TFJlDwMxUMA4GA1UdDwEB/wQEAwIHgDATBgNVHSUE
# DDAKBggrBgEFBQcDAzB3BgNVHR8EcDBuMDWgM6Axhi9odHRwOi8vY3JsMy5kaWdp
# Y2VydC5jb20vc2hhMi1hc3N1cmVkLWNzLWcxLmNybDA1oDOgMYYvaHR0cDovL2Ny
# bDQuZGlnaWNlcnQuY29tL3NoYTItYXNzdXJlZC1jcy1nMS5jcmwwTAYDVR0gBEUw
# QzA3BglghkgBhv1sAwEwKjAoBggrBgEFBQcCARYcaHR0cHM6Ly93d3cuZGlnaWNl
# cnQuY29tL0NQUzAIBgZngQwBBAEwgYQGCCsGAQUFBwEBBHgwdjAkBggrBgEFBQcw
# AYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tME4GCCsGAQUFBzAChkJodHRwOi8v
# Y2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRTSEEyQXNzdXJlZElEQ29kZVNp
# Z25pbmdDQS5jcnQwDAYDVR0TAQH/BAIwADANBgkqhkiG9w0BAQsFAAOCAQEAsBxn
# 9yJAQi+9cJPZpJvOEV6iHaOBGv8898wNJCc4eB5g8WPziEY70GZVeqEdx3z0wS8U
# QQIr19Hkju2NFZjDtzB9z1jAc/9EgqFGoCZbPijv1EYAa2oOVAp1BPbLjqBSdXqu
# 2mzqo14CJ30oNom9ep9F6LGZ5zEoPsMrJejSbJGr4EacrksX8C8qeFklc7FzwiGk
# GX7IQxidrrhOm2fOvGGAAxnvNYAR0FqJK0LiWWPSt5R/j63H/6HQtqD2sLevI3+O
# bRP74TPchDobFmWlSogX9oB63E7fsbDAqecY0cRPQ6tVWK53Ke2sB514nahFjZDa
# mxsa3/acZWL659ly3jGCD3gwgg90AgEBMIGGMHIxCzAJBgNVBAYTAlVTMRUwEwYD
# VQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xMTAv
# BgNVBAMTKERpZ2lDZXJ0IFNIQTIgQXNzdXJlZCBJRCBDb2RlIFNpZ25pbmcgQ0EC
# EAwwKlMuyPGj0VH2FqDx49QwDQYJYIZIAWUDBAIBBQCgfDAQBgorBgEEAYI3AgEM
# MQIwADAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4w
# DAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgjCiCemPZ9oGdLIpi/Q3QOdhs
# De4b87THnIAdtWZgZ0UwDQYJKoZIhvcNAQEBBQAEggEAqD0ZwaY9IaQEE3SXxKdP
# Fa77t6SGQkabHe+QMwDRbe2bkFUjpfVTjHtTfVvgZNpDDN3axFwzN7DCBKIicZ7s
# uMVWtkt17gqb2k+A+/atZdubPNUSx7cbUINUhUcsk6V1MnMInKDbrrWSB9yUS95u
# uZSJPD/jCWpCYII3DzsoWwE7nlvvTDvBaHq5YZ9LIA83PBgJxdXc+jE/ypdMGtWJ
# EkJ3FlsGelvC5GrUQN6FkdjITqmEgBJIDeeyeZqPCQT9NNi6xU5ZfN9UzuL9JNWS
# lsaJTKCMfVyCw+2EDzfvrzk/Xppeh92fxA9KSIF3nDUsFCpmHD+zzncKRcDojHQ/
# 9aGCDUQwgg1ABgorBgEEAYI3AwMBMYINMDCCDSwGCSqGSIb3DQEHAqCCDR0wgg0Z
# AgEDMQ8wDQYJYIZIAWUDBAIBBQAwdwYLKoZIhvcNAQkQAQSgaARmMGQCAQEGCWCG
# SAGG/WwHATAxMA0GCWCGSAFlAwQCAQUABCBJMzeUhpPT3HFqhs1x3zEaaEER6Q9I
# RuNlg3WXcJN/uAIQRRFyr3fkUwopJywOtemZ0hgPMjAyMTA1MDcyMTM3MjRaoIIK
# NzCCBP4wggPmoAMCAQICEA1CSuC+Ooj/YEAhzhQA8N0wDQYJKoZIhvcNAQELBQAw
# cjELMAkGA1UEBhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQ
# d3d3LmRpZ2ljZXJ0LmNvbTExMC8GA1UEAxMoRGlnaUNlcnQgU0hBMiBBc3N1cmVk
# IElEIFRpbWVzdGFtcGluZyBDQTAeFw0yMTAxMDEwMDAwMDBaFw0zMTAxMDYwMDAw
# MDBaMEgxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjEgMB4G
# A1UEAxMXRGlnaUNlcnQgVGltZXN0YW1wIDIwMjEwggEiMA0GCSqGSIb3DQEBAQUA
# A4IBDwAwggEKAoIBAQDC5mGEZ8WK9Q0IpEXKY2tR1zoRQr0KdXVNlLQMULUmEP4d
# yG+RawyW5xpcSO9E5b+bYc0VkWJauP9nC5xj/TZqgfop+N0rcIXeAhjzeG28ffnH
# bQk9vmp2h+mKvfiEXR52yeTGdnY6U9HR01o2j8aj4S8bOrdh1nPsTm0zinxdRS1L
# sVDmQTo3VobckyON91Al6GTm3dOPL1e1hyDrDo4s1SPa9E14RuMDgzEpSlwMMYpK
# jIjF9zBa+RSvFV9sQ0kJ/SYjU/aNY+gaq1uxHTDCm2mCtNv8VlS8H6GHq756Wwog
# L0sJyZWnjbL61mOLTqVyHO6fegFz+BnW/g1JhL0BAgMBAAGjggG4MIIBtDAOBgNV
# HQ8BAf8EBAMCB4AwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcD
# CDBBBgNVHSAEOjA4MDYGCWCGSAGG/WwHATApMCcGCCsGAQUFBwIBFhtodHRwOi8v
# d3d3LmRpZ2ljZXJ0LmNvbS9DUFMwHwYDVR0jBBgwFoAU9LbhIB3+Ka7S5GGlsqIl
# ssgXNW4wHQYDVR0OBBYEFDZEho6kurBmvrwoLR1ENt3janq8MHEGA1UdHwRqMGgw
# MqAwoC6GLGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9zaGEyLWFzc3VyZWQtdHMu
# Y3JsMDKgMKAuhixodHRwOi8vY3JsNC5kaWdpY2VydC5jb20vc2hhMi1hc3N1cmVk
# LXRzLmNybDCBhQYIKwYBBQUHAQEEeTB3MCQGCCsGAQUFBzABhhhodHRwOi8vb2Nz
# cC5kaWdpY2VydC5jb20wTwYIKwYBBQUHMAKGQ2h0dHA6Ly9jYWNlcnRzLmRpZ2lj
# ZXJ0LmNvbS9EaWdpQ2VydFNIQTJBc3N1cmVkSURUaW1lc3RhbXBpbmdDQS5jcnQw
# DQYJKoZIhvcNAQELBQADggEBAEgc3LXpmiO85xrnIA6OZ0b9QnJRdAojR6OrktIl
# xHBZvhSg5SeBpU0UFRkHefDRBMOG2Tu9/kQCZk3taaQP9rhwz2Lo9VFKeHk2eie3
# 8+dSn5On7UOee+e03UEiifuHokYDTvz0/rdkd2NfI1Jpg4L6GlPtkMyNoRdzDfTz
# ZTlwS/Oc1np72gy8PTLQG8v1Yfx1CAB2vIEO+MDhXM/EEXLnG2RJ2CKadRVC9S0y
# OIHa9GCiurRS+1zgYSQlT7LfySmoc0NR2r1j1h9bm/cuG08THfdKDXF+l7f0P4Tr
# weOjSaH6zqe/Vs+6WXZhiV9+p7SOZ3j5NpjhyyjaW4emii8wggUxMIIEGaADAgEC
# AhAKoSXW1jIbfkHkBdo2l8IVMA0GCSqGSIb3DQEBCwUAMGUxCzAJBgNVBAYTAlVT
# MRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5j
# b20xJDAiBgNVBAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAeFw0xNjAx
# MDcxMjAwMDBaFw0zMTAxMDcxMjAwMDBaMHIxCzAJBgNVBAYTAlVTMRUwEwYDVQQK
# EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xMTAvBgNV
# BAMTKERpZ2lDZXJ0IFNIQTIgQXNzdXJlZCBJRCBUaW1lc3RhbXBpbmcgQ0EwggEi
# MA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC90DLuS82Pf92puoKZxTlUKFe2
# I0rEDgdFM1EQfdD5fU1ofue2oPSNs4jkl79jIZCYvxO8V9PD4X4I1moUADj3Lh47
# 7sym9jJZ/l9lP+Cb6+NGRwYaVX4LJ37AovWg4N4iPw7/fpX786O6Ij4YrBHk8JkD
# bTuFfAnT7l3ImgtU46gJcWvgzyIQD3XPcXJOCq3fQDpct1HhoXkUxk0kIzBdvOw8
# YGqsLwfM/fDqR9mIUF79Zm5WYScpiYRR5oLnRlD9lCosp+R1PrqYD4R/nzEU1q3V
# 8mTLex4F0IQZchfxFwbvPc3WTe8GQv2iUypPhR3EHTyvz9qsEPXdrKzpVv+TAgMB
# AAGjggHOMIIByjAdBgNVHQ4EFgQU9LbhIB3+Ka7S5GGlsqIlssgXNW4wHwYDVR0j
# BBgwFoAUReuir/SSy4IxLVGLp6chnfNtyA8wEgYDVR0TAQH/BAgwBgEB/wIBADAO
# BgNVHQ8BAf8EBAMCAYYwEwYDVR0lBAwwCgYIKwYBBQUHAwgweQYIKwYBBQUHAQEE
# bTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wQwYIKwYB
# BQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3Vy
# ZWRJRFJvb3RDQS5jcnQwgYEGA1UdHwR6MHgwOqA4oDaGNGh0dHA6Ly9jcmw0LmRp
# Z2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcmwwOqA4oDaGNGh0
# dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5j
# cmwwUAYDVR0gBEkwRzA4BgpghkgBhv1sAAIEMCowKAYIKwYBBQUHAgEWHGh0dHBz
# Oi8vd3d3LmRpZ2ljZXJ0LmNvbS9DUFMwCwYJYIZIAYb9bAcBMA0GCSqGSIb3DQEB
# CwUAA4IBAQBxlRLpUYdWac3v3dp8qmN6s3jPBjdAhO9LhL/KzwMC/cWnww4gQiyv
# d/MrHwwhWiq3BTQdaq6Z+CeiZr8JqmDfdqQ6kw/4stHYfBli6F6CJR7Euhx7LCHi
# 1lssFDVDBGiy23UC4HLHmNY8ZOUfSBAYX4k4YU1iRiSHY4yRUiyvKYnleB/WCxSl
# gNcSR3CzddWThZN+tpJn+1Nhiaj1a5bA9FhpDXzIAbG5KHW3mWOFIoxhynmUfln8
# jA/jb7UBJrZspe6HUSHkWGCbugwtK22ixH67xCUrRwIIfEmuE7bhfEJCKMYYVs9B
# NLZmXbZ0e/VWMyIvIjayS6JKldj1po5SMYICTTCCAkkCAQEwgYYwcjELMAkGA1UE
# BhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2lj
# ZXJ0LmNvbTExMC8GA1UEAxMoRGlnaUNlcnQgU0hBMiBBc3N1cmVkIElEIFRpbWVz
# dGFtcGluZyBDQQIQDUJK4L46iP9gQCHOFADw3TANBglghkgBZQMEAgEFAKCBmDAa
# BgkqhkiG9w0BCQMxDQYLKoZIhvcNAQkQAQQwHAYJKoZIhvcNAQkFMQ8XDTIxMDUw
# NzIxMzcyNFowKwYLKoZIhvcNAQkQAgwxHDAaMBgwFgQU4deCqOGRvu9ryhaRtaq0
# lKYkm/MwLwYJKoZIhvcNAQkEMSIEIC/TGDZKsb7UNg5p9Fu8bXWXcOQsK1WWKhMG
# UuuLLNrmMA0GCSqGSIb3DQEBAQUABIIBAFqZDShQHXzdcmmOYKFSC3Iw1BRa6KEd
# lUtxjdofSCiTx8le2sWo8Nb9uC/UFB+t9TiBsvn+ijpf4anje+taoToHnoVgLeE5
# AD4yaJ14BslHWc08PD2dE0fD5Rajj+2DZUdvDkC2AhC2hWYgNiiaL3p9Qw6AXOnS
# ax1cM3tO29vdKRQAJIsplkoNtPuwhlYBw3UlZEI/5TcAC231nIqzUeFps5u3v1oO
# Pf4tKkm2e6wqtvGxmutyA1wI9EgGHJwgxDkOXufFbnlffAHuv4Phs2EL1kyjSLob
# xvcrCwM2aslxtjrZ4Fq0qUN9ulwduDAXHIDFb94M2ziqCCVTesr+mm4=
# SIG # End signature block