Lib/DscResource.ps1
function ImportDscResource { <# .SYNOPSIS Imports a DSC module resource. .DESCRIPTION Imports a DSC resource as Test-<Prefix>TargetResource and Set-<Prefix>TargetResource etc. #> [CmdletBinding()] param ( ## DSC resource's module name containing the resource [Parameter(Mandatory, ValueFromPipeline)] [System.String] $ModuleName, ## DSC resource's name to import [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String] $ResourceName, ## Local prefix, defaults to the resource name [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Prefix = $ResourceName, ## Use the built-in/default DSC resource [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $UseDefault ) process { ## Check whether the resource is already imported/registered Write-Debug ($localized.CheckingDscResource -f $ModuleName, $ResourceName); $testCommandName = 'Test-{0}TargetResource' -f $Prefix; if (-not (Get-Command -Name $testCommandName -ErrorAction SilentlyContinue)) { if ($UseDefault) { WriteVerbose ($localized.ImportingDscResource -f $ModuleName, $ResourceName); $resourcePath = GetDscModule -ModuleName $ModuleName -ResourceName $ResourceName -ErrorAction Stop; } else { WriteVerbose ($localized.ImportingBundledDscResource -f $ModuleName, $ResourceName); $dscModuleRootPath = '{0}\{1}\{2}\DSCResources' -f $labDefaults.ModuleRoot, $labDefaults.DscResourceDirectory, $ModuleName; $dscResourcePath = '{0}\{0}.psm1' -f $ResourceName; $resourcePath = Join-Path -Path $dscModuleRootPath -ChildPath $dscResourcePath; } if ($resourcePath) { ## Import the DSC module into the module's global scope to improve performance Import-Module -Name $resourcePath -Prefix $Prefix -Force -Verbose:$false -Scope Global; } } else { Write-Debug -Message ($localized.DscResourceAlreadyImported -f $ModuleName, $ResourceName); } } #end process } #end function ImportDscResource function GetDscResource { <# .SYNOPSIS Gets the ResourceName DSC resource configuration. .DESCRIPTION The GetDscResource cmdlet invokes the target $ResourceName\Get-TargetResource function using the supplied $Parameters hashtable. #> [CmdletBinding()] [OutputType([System.Collections.Hashtable])] param ( ## Name of the DSC resource to get [Parameter(Mandatory, ValueFromPipeline)] [System.String] $ResourceName, ## The DSC resource's Get-TargetResource parameter hashtable [Parameter(Mandatory)] [System.Collections.Hashtable] $Parameters ) process { $getTargetResourceCommand = 'Get-{0}TargetResource' -f $ResourceName; WriteVerbose ($localized.InvokingCommand -f $getTargetResourceCommand); # Code to factor in the parameters which can be passed to the Get-<Prefix>TargetResource function. $CommandInfo = Get-Command -Name $getTargetResourceCommand; $RemoveParameters = $Parameters.Keys | where -filter {$($CommandInfo.Parameters.Keys) -notcontains $PSItem}; $RemoveParameters | ForEach-Object -Process { [ref] $null = $Parameters.Remove($PSItem) }; return (& $getTargetResourceCommand @Parameters); } #end process } #end function GetDscResource function TestDscResource { <# .SYNOPSIS Tests the ResourceName DSC resource to determine if it's in the desired state. .DESCRIPTION The TestDscResource cmdlet invokes the target $ResourceName\Test-TargetResource function using the supplied $Parameters hastable. #> [CmdletBinding()] [OutputType([System.Boolean])] param ( ## Name of the DSC resource to test [Parameter(Mandatory, ValueFromPipeline)] [System.String] $ResourceName, ## The DSC resource's Test-TargetResource parameter hashtable [Parameter(Mandatory)] [System.Collections.Hashtable] $Parameters ) process { $testTargetResourceCommand = 'Test-{0}TargetResource' -f $ResourceName; WriteVerbose ($localized.InvokingCommand -f $testTargetResourceCommand); $Parameters.Keys | ForEach-Object { Write-Debug -Message ($localized.CommandParameter -f $_, $Parameters.$_); } $testDscResourceResult = & $testTargetResourceCommand @Parameters; if (-not $testDscResourceResult) { WriteVerbose ($localized.TestFailed -f $testTargetResourceCommand); } return $testDscResourceResult; } #end process } #end function TestDscResource function SetDscResource { <# .SYNOPSIS Runs the ResourceName DSC resource ensuring it's in the desired state. .DESCRIPTION The SetDscResource cmdlet invokes the target $ResourceName\Set-TargetResource function using the supplied $Parameters hastable. #> [CmdletBinding()] param ( ## Name of the DSC resource to invoke [Parameter(Mandatory, ValueFromPipeline)] [System.String] $ResourceName, ## The DSC resource's Set-TargetResource parameter hashtable [Parameter(Mandatory)] [System.Collections.Hashtable] $Parameters ) process { $setTargetResourceCommand = 'Set-{0}TargetResource' -f $ResourceName; WriteVerbose ($localized.InvokingCommand -f $setTargetResourceCommand); $Parameters.Keys | ForEach-Object { Write-Debug -Message ($localized.CommandParameter -f $_, $Parameters.$_); } return (& $setTargetResourceCommand @Parameters); } #end process } #end function SetDscResource function InvokeDscResource { <# .SYNOPSIS Runs the ResourceName DSC resource ensuring it's in the desired state. .DESCRIPTION The InvokeDscResource cmdlet invokes the target $ResourceName\Test-TargetResource function using the supplied $Parameters hastable. If the resource is not in the desired state, the $ResourceName\Set-TargetResource function is called with the $Parameters hashtable to attempt to correct the resource. #> [CmdletBinding()] param ( [Parameter(Mandatory)] [System.String] $ResourceName, [Parameter(Mandatory)] [System.Collections.Hashtable] $Parameters ) process { if (-not (TestDscResource @PSBoundParameters)) { if ($ResourceName -match 'PendingReboot') { throw $localized.PendingRebootWarning; } return (SetDscResource @PSBoundParameters); } else { $setTargetResourceCommand = 'Set-{0}TargetResource' -f $ResourceName; WriteVerbose ($localized.SkippingCommand -f $setTargetResourceCommand); } } #end process } #end function InvokeDscResource function GetDscResourcePSGalleryUri { <# .SYNOPSIS Returns the DSC resource direct download Uri #> [CmdletBinding()] [OutputType([System.String])] param ( ## PowerShell DSC resource module name [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Name, ## The minimum version of the DSC module required [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.Version] $MinimumVersion, ## The exact version of the DSC module required [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.Version] $RequiredVersion, ## Direct download Uri [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Uri, ## Catch all, for splatting parameters [Parameter(ValueFromRemainingArguments)] $RemainingArguments ) process { if ($PSBoundParameters.ContainsKey('Uri')) { return $Uri; } elseif ($PSBoundParameters.ContainsKey('RequiredVersion')) { ## Download the specific version return ('http://www.powershellgallery.com/api/v2/package/{0}/{1}' -f $Name, $RequiredVersion); } else { ## Download the latest version return ('http://www.powershellgallery.com/api/v2/package/{0}' -f $Name); } } #end process } #end function GetDscResourcePSGalleryUri function InvokeDscResourceDownload { <# .SYNOPSIS Downloads a DSC resource if it has not already been downloaded or the checksum is incorrect. #> [CmdletBinding()] [OutputType([System.String])] param ( ## PowerShell DSC resource modules [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName)] [System.Collections.Hashtable[]] $DSCResource, ## Force a download, overwriting any existing resources [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force, ## Catch all, for splatting parameters [Parameter(ValueFromRemainingArguments)] $RemainingArguments ) process { foreach ($resourceDefinition in $DSCResource) { ## Ensure we at least have a -MinimumVersion key if ((-not $resourceDefinition.ContainsKey('MinimumVersion')) -and (-not $resourceDefinition.ContainsKey('RequiredVersion'))) { $resourceDefinition['MinimumVersion'] = '0.0'; } if ((-not (TestModule @resourceDefinition) -or $Force)) { switch ($resourceDefinition.Provider) { 'GitHub' { [ref] $null = InvokeDscResourceDownloadFromGitHub @resourceDefinition -Force:$Force; } default { ## Use the PSGallery provider by default. [ref] $null = InvokeDscResourceDownloadFromPSGallery @resourceDefinition -Force:$Force; } } } #end if module not present $module = GetModule -Name $resourceDefinition.Name; Write-Output (Get-Item -Path $module.Path).Directory; } #end foreach DSC resource } #end process } #end function InvokeDscResourceDownload function InvokeDscResourceDownloadFromPSGallery { <# .SYNOPSIS Downloads a DSC resource if it has not already been downloaded from the Powershell Gallery. #> [CmdletBinding()] [OutputType([System.IO.DirectoryInfo])] param ( ## PowerShell DSC resource module name [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Name, ## The minimum version of the DSC module required [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.Version] $MinimumVersion, ## The exact version of the DSC module required [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.Version] $RequiredVersion, ## Force a download, overwriting any existing resources [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force, ## Catch all, for splatting parameters [Parameter(ValueFromRemainingArguments)] $RemainingArguments ) begin { ## Cannot pass -Force to GetDscResourcePSGalleryUri or SetResourceDownload [ref] $null = $PSBoundParameters.Remove('Force'); } process { $windowsPowerShellModules = Join-Path -Path $env:ProgramFiles -ChildPath '\WindowsPowerShell\Modules'; $tempModuleFilename = '{0}.zip' -f $Name; $tempDestinationPath = Join-Path -Path $env:Temp -ChildPath $tempModuleFilename; $psGalleryUri = GetDscResourcePSGalleryUri @PSBoundParameters; $tempFileInfo = SetResourceDownload -DestinationPath $tempDestinationPath -Uri $psGalleryUri; ## Extract .Zip to PSModulePath $modulePath = Join-Path $windowsPowerShellModules -ChildPath $Name; [ref] $null = ExpandZipArchive -Path $tempFileInfo -DestinationPath $modulePath -ExcludeNuSpecFiles -Force:$Force; return (Get-Item -Path $modulePath); } #end process } #end function function InvokeDscResourceDownloadFromGitHub { <# .SYNOPSIS Downloads a DSC resource if it has not already been downloaded from Github. .NOTES Uses the GitHubRepository module! #> [CmdletBinding()] [OutputType([System.IO.DirectoryInfo])] param ( ## PowerShell DSC resource module name [Parameter(Mandatory, ValueFromPipeline, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Name, ## The GitHub repository owner, typically 'PowerShell' [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Owner, ## The GitHub repository name, normally the DSC module's name [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Repository = $Name, ## The GitHub branch to download, defaults to the 'master' branch [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Branch = 'master', ## Override the local directory name. Only used if the repository name does not ## match the DSC module name [Parameter(ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $OverrideRepositoryName = $Name, ## Force a download, overwriting any existing resources [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force, ## Catch all, for splatting parameters [Parameter(ValueFromRemainingArguments)] $RemainingArguments ) begin { ## Bootstrap the GithubRepository module if (-not (TestModule -Name GitHubRepository -MinimumVersion '0.9.3')) { [ref] $null = InvokeDscResourceDownloadFromPSGallery -Name 'GitHubRepository'; } } process { Import-Module -Name 'GitHubRepository' -Verbose:$false; $installGitHubRepositoryParams = @{ Owner = $Owner; Repository = $Repository; Branch = $Branch; OverrideRepository = $OverrideRepositoryName; } return (Install-GitHubRepository @installGitHubRepositoryParams -Verbose:$false -Force:$Force); } #end process } #end function Invoke-DscResourceDownloadFromGitHub # SIG # Begin signature block # MIIXtwYJKoZIhvcNAQcCoIIXqDCCF6QCAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB # gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR # AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQU2a91BjsG6bg4OF7BjslFZiKC # NESgghLqMIID7jCCA1egAwIBAgIQfpPr+3zGTlnqS5p31Ab8OzANBgkqhkiG9w0B # AQUFADCBizELMAkGA1UEBhMCWkExFTATBgNVBAgTDFdlc3Rlcm4gQ2FwZTEUMBIG # A1UEBxMLRHVyYmFudmlsbGUxDzANBgNVBAoTBlRoYXd0ZTEdMBsGA1UECxMUVGhh # d3RlIENlcnRpZmljYXRpb24xHzAdBgNVBAMTFlRoYXd0ZSBUaW1lc3RhbXBpbmcg # Q0EwHhcNMTIxMjIxMDAwMDAwWhcNMjAxMjMwMjM1OTU5WjBeMQswCQYDVQQGEwJV # UzEdMBsGA1UEChMUU3ltYW50ZWMgQ29ycG9yYXRpb24xMDAuBgNVBAMTJ1N5bWFu # dGVjIFRpbWUgU3RhbXBpbmcgU2VydmljZXMgQ0EgLSBHMjCCASIwDQYJKoZIhvcN # AQEBBQADggEPADCCAQoCggEBALGss0lUS5ccEgrYJXmRIlcqb9y4JsRDc2vCvy5Q # WvsUwnaOQwElQ7Sh4kX06Ld7w3TMIte0lAAC903tv7S3RCRrzV9FO9FEzkMScxeC # i2m0K8uZHqxyGyZNcR+xMd37UWECU6aq9UksBXhFpS+JzueZ5/6M4lc/PcaS3Er4 # ezPkeQr78HWIQZz/xQNRmarXbJ+TaYdlKYOFwmAUxMjJOxTawIHwHw103pIiq8r3 # +3R8J+b3Sht/p8OeLa6K6qbmqicWfWH3mHERvOJQoUvlXfrlDqcsn6plINPYlujI # fKVOSET/GeJEB5IL12iEgF1qeGRFzWBGflTBE3zFefHJwXECAwEAAaOB+jCB9zAd # BgNVHQ4EFgQUX5r1blzMzHSa1N197z/b7EyALt0wMgYIKwYBBQUHAQEEJjAkMCIG # CCsGAQUFBzABhhZodHRwOi8vb2NzcC50aGF3dGUuY29tMBIGA1UdEwEB/wQIMAYB # Af8CAQAwPwYDVR0fBDgwNjA0oDKgMIYuaHR0cDovL2NybC50aGF3dGUuY29tL1Ro # YXd0ZVRpbWVzdGFtcGluZ0NBLmNybDATBgNVHSUEDDAKBggrBgEFBQcDCDAOBgNV # HQ8BAf8EBAMCAQYwKAYDVR0RBCEwH6QdMBsxGTAXBgNVBAMTEFRpbWVTdGFtcC0y # MDQ4LTEwDQYJKoZIhvcNAQEFBQADgYEAAwmbj3nvf1kwqu9otfrjCR27T4IGXTdf # plKfFo3qHJIJRG71betYfDDo+WmNI3MLEm9Hqa45EfgqsZuwGsOO61mWAK3ODE2y # 0DGmCFwqevzieh1XTKhlGOl5QGIllm7HxzdqgyEIjkHq3dlXPx13SYcqFgZepjhq # IhKjURmDfrYwggSjMIIDi6ADAgECAhAOz/Q4yP6/NW4E2GqYGxpQMA0GCSqGSIb3 # DQEBBQUAMF4xCzAJBgNVBAYTAlVTMR0wGwYDVQQKExRTeW1hbnRlYyBDb3Jwb3Jh # dGlvbjEwMC4GA1UEAxMnU3ltYW50ZWMgVGltZSBTdGFtcGluZyBTZXJ2aWNlcyBD # QSAtIEcyMB4XDTEyMTAxODAwMDAwMFoXDTIwMTIyOTIzNTk1OVowYjELMAkGA1UE # BhMCVVMxHTAbBgNVBAoTFFN5bWFudGVjIENvcnBvcmF0aW9uMTQwMgYDVQQDEytT # eW1hbnRlYyBUaW1lIFN0YW1waW5nIFNlcnZpY2VzIFNpZ25lciAtIEc0MIIBIjAN # BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAomMLOUS4uyOnREm7Dv+h8GEKU5Ow # mNutLA9KxW7/hjxTVQ8VzgQ/K/2plpbZvmF5C1vJTIZ25eBDSyKV7sIrQ8Gf2Gi0 # jkBP7oU4uRHFI/JkWPAVMm9OV6GuiKQC1yoezUvh3WPVF4kyW7BemVqonShQDhfu # ltthO0VRHc8SVguSR/yrrvZmPUescHLnkudfzRC5xINklBm9JYDh6NIipdC6Anqh # d5NbZcPuF3S8QYYq3AhMjJKMkS2ed0QfaNaodHfbDlsyi1aLM73ZY8hJnTrFxeoz # C9Lxoxv0i77Zs1eLO94Ep3oisiSuLsdwxb5OgyYI+wu9qU+ZCOEQKHKqzQIDAQAB # o4IBVzCCAVMwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcDCDAO # BgNVHQ8BAf8EBAMCB4AwcwYIKwYBBQUHAQEEZzBlMCoGCCsGAQUFBzABhh5odHRw # Oi8vdHMtb2NzcC53cy5zeW1hbnRlYy5jb20wNwYIKwYBBQUHMAKGK2h0dHA6Ly90 # cy1haWEud3Muc3ltYW50ZWMuY29tL3Rzcy1jYS1nMi5jZXIwPAYDVR0fBDUwMzAx # oC+gLYYraHR0cDovL3RzLWNybC53cy5zeW1hbnRlYy5jb20vdHNzLWNhLWcyLmNy # bDAoBgNVHREEITAfpB0wGzEZMBcGA1UEAxMQVGltZVN0YW1wLTIwNDgtMjAdBgNV # HQ4EFgQURsZpow5KFB7VTNpSYxc/Xja8DeYwHwYDVR0jBBgwFoAUX5r1blzMzHSa # 1N197z/b7EyALt0wDQYJKoZIhvcNAQEFBQADggEBAHg7tJEqAEzwj2IwN3ijhCcH # bxiy3iXcoNSUA6qGTiWfmkADHN3O43nLIWgG2rYytG2/9CwmYzPkSWRtDebDZw73 # BaQ1bHyJFsbpst+y6d0gxnEPzZV03LZc3r03H0N45ni1zSgEIKOq8UvEiCmRDoDR # EfzdXHZuT14ORUZBbg2w6jiasTraCXEQ/Bx5tIB7rGn0/Zy2DBYr8X9bCT2bW+IW # yhOBbQAuOA2oKY8s4bL0WqkBrxWcLC9JG9siu8P+eJRRw4axgohd8D20UaF5Mysu # e7ncIAkTcetqGVvP6KUwVyyJST+5z3/Jvz4iaGNTmr1pdKzFHTx/kuDDvBzYBHUw # ggUZMIIEAaADAgECAhADViTO4HBjoJNSwH9//cwJMA0GCSqGSIb3DQEBCwUAMHIx # CzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3 # dy5kaWdpY2VydC5jb20xMTAvBgNVBAMTKERpZ2lDZXJ0IFNIQTIgQXNzdXJlZCBJ # RCBDb2RlIFNpZ25pbmcgQ0EwHhcNMTUwNTE5MDAwMDAwWhcNMTcwODIzMTIwMDAw # WjBgMQswCQYDVQQGEwJHQjEPMA0GA1UEBxMGT3hmb3JkMR8wHQYDVQQKExZWaXJ0 # dWFsIEVuZ2luZSBMaW1pdGVkMR8wHQYDVQQDExZWaXJ0dWFsIEVuZ2luZSBMaW1p # dGVkMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAqLQmabdimcQtYPTQ # 9RSjv3ThEmFTRJt/MzseYYtZpBTcR6BnSfj8RfkC4aGZvspFgH0cGP/SNJh1w67b # iX9oT5NFL9sUJHUsVdyPBA1LhpWcF09PP28mGGKO3oQHI4hTLD8etiIlF9qFantd # 1Pmo0jdqT4uErSmx0m4kYGUUTa5ZPAK0UZSuAiNX6iNIL+rj/BPbI3nuPJzzx438 # oHYkZGRtsx11+pLA6hIKyUzRuIDoI7JQ0nZ0MkCziVyc6xGfS54JVLaVCEteTKPz # Gc4yyvCqp6Tfe9gs8UuxJiEMdH5fvllTU4aoXbm+W8tonkE7i/19rv8S1A2VPiVV # xNLbpwIDAQABo4IBuzCCAbcwHwYDVR0jBBgwFoAUWsS5eyoKo6XqcQPAYPkt9mV1 # DlgwHQYDVR0OBBYEFP2RNOWYipdNCSRVb5jIcyRp9tUDMA4GA1UdDwEB/wQEAwIH # gDATBgNVHSUEDDAKBggrBgEFBQcDAzB3BgNVHR8EcDBuMDWgM6Axhi9odHRwOi8v # Y3JsMy5kaWdpY2VydC5jb20vc2hhMi1hc3N1cmVkLWNzLWcxLmNybDA1oDOgMYYv # aHR0cDovL2NybDQuZGlnaWNlcnQuY29tL3NoYTItYXNzdXJlZC1jcy1nMS5jcmww # QgYDVR0gBDswOTA3BglghkgBhv1sAwEwKjAoBggrBgEFBQcCARYcaHR0cHM6Ly93 # d3cuZGlnaWNlcnQuY29tL0NQUzCBhAYIKwYBBQUHAQEEeDB2MCQGCCsGAQUFBzAB # hhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wTgYIKwYBBQUHMAKGQmh0dHA6Ly9j # YWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFNIQTJBc3N1cmVkSURDb2RlU2ln # bmluZ0NBLmNydDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4IBAQCclXHR # DhDyJr81eiD0x+AL04ryDwdKT+PooKYgOxc7EhRn59ogxNO7jApQPSVo0I11Zfm6 # zQ6K6RPWhxDenflf2vMx7a0tIZlpHhq2F8praAMykK7THA9F3AUxIb/lWHGZCock # yD/GQvJek3LSC5NjkwQbnubWYF/XZTDzX/mJGU2DcG1OGameffR1V3xODHcUE/K3 # PWy1bzixwbQCQA96GKNCWow4/mEW31cupHHSo+XVxmjTAoC93yllE9f4Kdv6F29H # bRk0Go8Yn8WjWeLE/htxW/8ruIj0KnWkG+YwmZD+nTegYU6RvAV9HbJJYUEIfhVy # 3DeK5OlY9ima2sdtMIIFMDCCBBigAwIBAgIQBAkYG1/Vu2Z1U0O1b5VQCDANBgkq # hkiG9w0BAQsFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5j # MRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBB # c3N1cmVkIElEIFJvb3QgQ0EwHhcNMTMxMDIyMTIwMDAwWhcNMjgxMDIyMTIwMDAw # WjByMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQL # ExB3d3cuZGlnaWNlcnQuY29tMTEwLwYDVQQDEyhEaWdpQ2VydCBTSEEyIEFzc3Vy # ZWQgSUQgQ29kZSBTaWduaW5nIENBMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIB # CgKCAQEA+NOzHH8OEa9ndwfTCzFJGc/Q+0WZsTrbRPV/5aid2zLXcep2nQUut4/6 # kkPApfmJ1DcZ17aq8JyGpdglrA55KDp+6dFn08b7KSfH03sjlOSRI5aQd4L5oYQj # ZhJUM1B0sSgmuyRpwsJS8hRniolF1C2ho+mILCCVrhxKhwjfDPXiTWAYvqrEsq5w # MWYzcT6scKKrzn/pfMuSoeU7MRzP6vIK5Fe7SrXpdOYr/mzLfnQ5Ng2Q7+S1TqSp # 6moKq4TzrGdOtcT3jNEgJSPrCGQ+UpbB8g8S9MWOD8Gi6CxR93O8vYWxYoNzQYIH # 5DiLanMg0A9kczyen6Yzqf0Z3yWT0QIDAQABo4IBzTCCAckwEgYDVR0TAQH/BAgw # BgEB/wIBADAOBgNVHQ8BAf8EBAMCAYYwEwYDVR0lBAwwCgYIKwYBBQUHAwMweQYI # KwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5j # b20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdp # Q2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwgYEGA1UdHwR6MHgwOqA4oDaGNGh0dHA6 # Ly9jcmw0LmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcmww # OqA4oDaGNGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJ # RFJvb3RDQS5jcmwwTwYDVR0gBEgwRjA4BgpghkgBhv1sAAIEMCowKAYIKwYBBQUH # AgEWHGh0dHBzOi8vd3d3LmRpZ2ljZXJ0LmNvbS9DUFMwCgYIYIZIAYb9bAMwHQYD # VR0OBBYEFFrEuXsqCqOl6nEDwGD5LfZldQ5YMB8GA1UdIwQYMBaAFEXroq/0ksuC # MS1Ri6enIZ3zbcgPMA0GCSqGSIb3DQEBCwUAA4IBAQA+7A1aJLPzItEVyCx8JSl2 # qB1dHC06GsTvMGHXfgtg/cM9D8Svi/3vKt8gVTew4fbRknUPUbRupY5a4l4kgU4Q # pO4/cY5jDhNLrddfRHnzNhQGivecRk5c/5CxGwcOkRX7uq+1UcKNJK4kxscnKqEp # KBo6cSgCPC6Ro8AlEeKcFEehemhor5unXCBc2XGxDI+7qPjFEmifz0DLQESlE/Dm # ZAwlCEIysjaKJAL+L3J+HNdJRZboWR3p+nRka7LrZkPas7CM1ekN3fYBIM6ZMWM9 # CBoYs4GbT8aTEAb8B4H6i9r5gkn3Ym6hU/oSlBiFLpKR6mhsRDKyZqHnGKSaZFHv # MYIENzCCBDMCAQEwgYYwcjELMAkGA1UEBhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0 # IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTExMC8GA1UEAxMoRGlnaUNl # cnQgU0hBMiBBc3N1cmVkIElEIENvZGUgU2lnbmluZyBDQQIQA1YkzuBwY6CTUsB/ # f/3MCTAJBgUrDgMCGgUAoHgwGAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZBgkq # hkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGC # NwIBFTAjBgkqhkiG9w0BCQQxFgQU09X4HpUazJfIyAQkBAP+a864CxcwDQYJKoZI # hvcNAQEBBQAEggEATKOzrN4bGhcls7kGRyl6ZZ9Ofzz7wIwCJ+2DTTnnstcSjuhM # LCcy4JLiPWD97MlPFmRUKBBdD5Z4j66Z6/EL4EhqAA8cMmvXoCMNxkFqvDat3Vwj # OIO0tdqOu232ilU1KpBHb+7cpdIo9VnvH8VZFJtfh7PWxnYaHUDRo4oMv7tG+grs # UZiOZE3Hym22/opgZ4JTxgfnFMVb8Uv5+XmsBOF1AH2RRS7QpzHQ3+2iCxA3QTFZ # 2UesY3jBbMuSEHsxM4mPItYt5qGtSaeCPpMwOTFBtbtF9zZpuZbuRBk7EIKHa24L # O26R2Fr3r5/gtu4P34z33q/u257X1FOxK+TySqGCAgswggIHBgkqhkiG9w0BCQYx # ggH4MIIB9AIBATByMF4xCzAJBgNVBAYTAlVTMR0wGwYDVQQKExRTeW1hbnRlYyBD # b3Jwb3JhdGlvbjEwMC4GA1UEAxMnU3ltYW50ZWMgVGltZSBTdGFtcGluZyBTZXJ2 # aWNlcyBDQSAtIEcyAhAOz/Q4yP6/NW4E2GqYGxpQMAkGBSsOAwIaBQCgXTAYBgkq # hkiG9w0BCQMxCwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJBTEPFw0xNjA0MTMxMzMz # MThaMCMGCSqGSIb3DQEJBDEWBBQzPsDVGioPmWSaj/4ev1iMKXD3UzANBgkqhkiG # 9w0BAQEFAASCAQBuJCV4v+g1Dyrko1yr7x7z6jdMK9LyrNwK67FIOHKX23rd/4J6 # AaWgLYW/zoScdW9dhp76SgMUfu3GYX9Egb+hKuEmKVdWhzcsyW88YisDvZqGCFxZ # Ebq3s9PLy99kEjSTvFlJQL6s3QIa3PoiDp0tfWBqTsnNHQup0woprJSNJxO/qE9d # lwmWtxpJGyCaL85ZZjmUEMhg9q7AjmR2EMkPHQyOSnkhSCu/VGd/OpPqsn4noRMc # 5DrAgtEG+OQSWPPO74o2kbHmyQ3H+OE3hXkiUOqPusBRFEuYRgrkJKYVBGy6uX2h # PT5aHij47mSnGJ+iQAMAmoU5JWMUVLwpUOPk # SIG # End signature block |