MocUpgradePrecheck.Tests.ps1
|
BeforeAll { Import-Module Pester ipmo .\Moc.psm1 Import-Module $PSScriptRoot\MocUpgradePrecheckTestMocks.psm1 $cluster = "s-cluster" } AfterAll { Remove-Module Moc Remove-Module MocUpgradePrecheckTestMocks } # Important note on mocks # Module name param tells pester to inject the mock into the corresponding # modules scope, so the internal functions of MocNodesHelper will use the mock # instead of the regular version of the function Describe "Verify MocNodesMatchFailoverClusterNodes" { BeforeAll { $activeStatus = @{ "statuses" = @{ "RunningState" = "Active" } } $inactiveStatus = @{ "statuses" = @{ "RunningState" = "Inactive" "Error" = "Precheck Error Message" } } } It "test fails when Get-MocConfig throws with [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-MocConfig { throw "Mock exception" } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 1 $result.Severity | Should -Be "CRITICAL" $result.Status | Should -Be "FAILURE" } It "test fails when Get-MocConfig throws with [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-MocConfig { @{ "cloudLocation" = "MocLocation" } } Mock -ModuleName "Moc" Get-MocNode { throw "Mock exception" } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 1 $result.Severity | Should -Be "CRITICAL" $result.Status | Should -Be "FAILURE" } It "MocNodes test succeeds with [SUCCESS][INFORMATIONAL]" { Mock -ModuleName "Moc" Get-MocNode { @( @{ "name" = "ABC-D120301" "properties" = $activeStatus }, @{ "name" = "abc-d120302" "properties" = $activeStatus } ) } -ParameterFilter {$Location -ne $null} # MOC node can be exact case match, or full lowercase of the failover cluster name Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "ABC-D120301" "State" = "Up" }, @{ "Name" = "Abc-D120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "cloudLocation" = "MocLocation" } } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 5 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "ABC-D120301" $result[2].Severity | Should -Be "INFORMATIONAL" $result[2].Status | Should -Be "SUCCESS" $result[2].TargetResourceName | Should -Be "Abc-D120302" $result[3].Severity | Should -Be "INFORMATIONAL" $result[3].Status | Should -Be "SUCCESS" $result[3].TargetResourceName | Should -Be "ABC-D120301" $result[4].Severity | Should -Be "INFORMATIONAL" $result[4].Status | Should -Be "SUCCESS" $result[4].TargetResourceName | Should -Be "abc-d120302" } It "MocNodes test succeeds fails when node state is inactive [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-MocNode { # Since there is a single element, add the comma to make sure it gets # returned as an array type, not a single object. Same applies to below ,@( @{ "name" = "ABC-D120301" "properties" = $inactiveStatus } ) } -ParameterFilter {$Location -ne $null} Mock -ModuleName "Moc" Get-ClusterNode { ,@( @{ "Name" = "ABC-D120301" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "cloudLocation" = "MocLocation" } } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 3 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "ABC-D120301" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "FAILURE" $result[2].TargetResourceName | Should -Be "ABC-D120301" $result[2].Description -match "unexpected running state.*Precheck Error Message" | Should -BeTrue } It "MocNodes test fails due to differences in casing or name with [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-MocNode { @( @{ "name" = "ABC-D120301" "properties" = $activeStatus }, @{ "name" = "ABC-D120302" "properties" = $activeStatus } ) } -ParameterFilter {$Location -ne $null} # The MOC name needs to be either fully lowercase of the Failover cluster name # or an exact match, anything else should fail Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "Abc-d120301" "State" = "Up" }, @{ "Name" = "EFG-D120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "cloudLocation" = "MocLocation" } } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 5 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "CRITICAL" $result[1].Status | Should -Be "FAILURE" $result[1].TargetResourceName | Should -Be "abc-d120301" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "FAILURE" $result[2].TargetResourceName | Should -Be "EFG-D120302" $result[3].Severity | Should -Be "INFORMATIONAL" $result[3].Status | Should -Be "SUCCESS" $result[3].TargetResourceName | Should -Be "Abc-d120301" $result[4].Severity | Should -Be "INFORMATIONAL" $result[4].Status | Should -Be "SUCCESS" $result[4].TargetResourceName | Should -Be "ABC-D120302" } It "MocNodes test fails due to additional MOC nodes [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-MocNode { @( @{ "name" = "ABC-D120301" "properties" = $activeStatus }, @{ "name" = "ABC-D120302" "properties" = $activeStatus }, @{ "name" = "ABC-D120303" "properties" = $activeStatus } ) } -ParameterFilter {$Location -ne $null} Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "ABC-D120301" "State" = "Up" }, @{ "Name" = "ABC-D120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "cloudLocation" = "MocLocation" } } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 6 $result[0].Severity | Should -Be "CRITICAL" $result[0].Status | Should -Be "FAILURE" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "ABC-D120301" $result[2].Severity | Should -Be "INFORMATIONAL" $result[2].Status | Should -Be "SUCCESS" $result[2].TargetResourceName | Should -Be "ABC-D120302" $result[3].Severity | Should -Be "INFORMATIONAL" $result[3].Status | Should -Be "SUCCESS" $result[3].TargetResourceName | Should -Be "ABC-D120301" $result[4].Severity | Should -Be "INFORMATIONAL" $result[4].Status | Should -Be "SUCCESS" $result[4].TargetResourceName | Should -Be "ABC-D120302" $result[5].Severity | Should -Be "INFORMATIONAL" $result[5].Status | Should -Be "SUCCESS" $result[5].TargetResourceName | Should -Be "ABC-D120303" } It "MocNodes test fails due to additional cluster nodes and case change [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-MocNode { @( @{ "name" = "ABC-D120301" "properties" = $activeStatus }, @{ "name" = "ABC-D120302" "properties" = $activeStatus } ) } -ParameterFilter {$Location -ne $null} Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "ABC-D120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" }, @{ "Name" = "abc-d120303" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "cloudLocation" = "MocLocation" } } $result = Test-MocNodesMatchFailoverClusterNodes -Cluster $cluster $result.count | Should -Be 6 $result[0].Severity | Should -Be "CRITICAL" $result[0].Status | Should -Be "FAILURE" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "ABC-D120301" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "FAILURE" $result[2].TargetResourceName | Should -Be "abc-d120302" $result[3].Severity | Should -Be "CRITICAL" $result[3].Status | Should -Be "FAILURE" $result[3].TargetResourceName | Should -Be "abc-d120303" $result[4].Severity | Should -Be "INFORMATIONAL" $result[4].Status | Should -Be "SUCCESS" $result[4].TargetResourceName | Should -Be "ABC-D120301" $result[5].Severity | Should -Be "INFORMATIONAL" $result[5].Status | Should -Be "SUCCESS" $result[5].TargetResourceName | Should -Be "ABC-D120302" } } Describe "Verify Test-HCIClusterAndNodeRegistration" { It "Succeeds when cluster is connected and registered, and all nodes are active [SUCCESS][INFORMATIONAL]" { Mock -ModuleName "Moc" Get-AzureStackHCI { @{ "RegistrationStatus" = "Registered" "ConnectionStatus" = "Connected" } } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-AzureStackHCISubscriptionStatus { @( @{ "SubscriptionName" = "Azure Stack HCI" "Status" = "Active" } ) } # this is to ensure the Test-HCIClusterAndNodeRegistration is called in completely irrespecive of the test machine osbuild Mock -ModuleName "Moc" Get-CimInstance { @{ "BuildNumber" = 26000 } } $result = Test-HCIClusterAndNodeRegistration -Cluster $cluster $result.count | Should -Be 3 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "abc-d120301" $result[2].Severity | Should -Be "INFORMATIONAL" $result[2].Status | Should -Be "SUCCESS" $result[2].TargetResourceName | Should -Be "abc-d120302" } It "Succeeds when cluster is connected and registered, and all nodes are active with Preview Subscription Name [SUCCESS][INFORMATIONAL]" { Mock -ModuleName "Moc" Get-AzureStackHCI { @{ "RegistrationStatus" = "Registered" "ConnectionStatus" = "Connected" } } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-AzureStackHCISubscriptionStatus { @( @{ "SubscriptionName" = "Azure Stack HCI Preview" "Status" = "Active" } ) } # this is to ensure the Test-HCIClusterAndNodeRegistration is called in completely irrespecive of the test machine osbuild Mock -ModuleName "Moc" Get-CimInstance { @{ "BuildNumber" = 26000 } } $result = Test-HCIClusterAndNodeRegistration -Cluster $cluster $result.count | Should -Be 3 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "abc-d120301" $result[2].Severity | Should -Be "INFORMATIONAL" $result[2].Status | Should -Be "SUCCESS" $result[2].TargetResourceName | Should -Be "abc-d120302" } It "Fails when cluster is connected and registered, but nodes do not have valid subscription name [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-AzureStackHCI { @{ "RegistrationStatus" = "Registered" "ConnectionStatus" = "Connected" } } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-AzureStackHCISubscriptionStatus { @( @{ "SubscriptionName" = "Azure Stack Invalid" "Status" = "Active" } ) } # this is to ensure the Test-HCIClusterAndNodeRegistration is called in completely irrespecive of the test machine osbuild Mock -ModuleName "Moc" Get-CimInstance { @{ "BuildNumber" = 26000 } } $result = Test-HCIClusterAndNodeRegistration -Cluster $cluster $result.count | Should -Be 3 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "s-cluster" $result[1].Severity | Should -Be "CRITICAL" $result[1].Status | Should -Be "FAILURE" $result[1].TargetResourceName | Should -Be "abc-d120301" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "FAILURE" $result[2].TargetResourceName | Should -Be "abc-d120302" } It "Failed when cluster is not registered, and all nodes are inactive [FAILURE][CRITICAL]" { Mock -ModuleName "Moc" Get-AzureStackHCI { @{ "RegistrationStatus" = "Unregistered" "ConnectionStatus" = "Disconnected" } } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-AzureStackHCISubscriptionStatus { @( @{ "SubscriptionName" = "Azure Stack HCI" "Status" = "Inactive" } ) } # this is to ensure the Test-HCIClusterAndNodeRegistration is called in completely irrespecive of the test machine osbuild Mock -ModuleName "Moc" Get-CimInstance { @{ "BuildNumber" = 26000 } } $result = Test-HCIClusterAndNodeRegistration -Cluster $cluster $result.count | Should -Be 3 $result[0].Severity | Should -Be "CRITICAL" $result[0].Status | Should -Be "FAILURE" $result[0].TargetResourceName | Should -Be "s-cluster" $result[0].Description -match "Registration Status" | Should -BeTrue $result[1].Severity | Should -Be "CRITICAL" $result[1].Status | Should -Be "FAILURE" $result[1].TargetResourceName | Should -Be "abc-d120301" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "FAILURE" $result[2].TargetResourceName | Should -Be "abc-d120302" } } Describe "Verify Test-MocNodeConnectivity" { It "Should fail when a network connection between cloudagent and nodeagent is unhealthy" { Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "mockName" } } Mock -ModuleName "Moc" Get-FailoverCluster { @{ "domain" = "contoso.com" } } Mock -ModuleName "Moc" Get-FailoverClusterGroup { @{ "ownernode" = @{ "name" = "abc-d120302" } } } Mock -ModuleName "Moc" Invoke-Command { @{ "TcpTestSucceeded" = $false } } Mock -ModuleName "Moc" Get-CloudFqdn { "abc-d120302.contoso.com" } $result = Test-MocNodeConnectivity -Cluster $cluster $result.count | Should -Be 4 $result[0].Severity | Should -Be "CRITICAL" $result[0].Status | Should -Be "FAILURE" $result[0].TargetResourceName | Should -Be "abc-d120301" $result[0].Description | Should -Be "TCP test from CloudAgent Node abc-d120302 to nodeagent abc-d120301.contoso.com failed" $result[1].Severity | Should -Be "CRITICAL" $result[1].Status | Should -Be "FAILURE" $result[1].TargetResourceName | Should -Be "abc-d120302" $result[1].Description | Should -Be "TCP test from CloudAgent Node abc-d120302 to nodeagent abc-d120302.contoso.com failed" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "FAILURE" $result[2].TargetResourceName | Should -Be "abc-d120301" $result[2].Description | Should -Be "TCP test from cluster node abc-d120301 to CloudAgent fqdn abc-d120302.contoso.com failed" $result[3].Severity | Should -Be "CRITICAL" $result[3].Status | Should -Be "FAILURE" $result[3].TargetResourceName | Should -Be "abc-d120302" $result[3].Description | Should -Be "TCP test from cluster node abc-d120302 to CloudAgent fqdn abc-d120302.contoso.com failed" } It "Should succeed when all agents are connected" { Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "mockName" } } Mock -ModuleName "Moc" Get-FailoverCluster { @{ "domain" = "contoso.com" } } Mock -ModuleName "Moc" Get-FailoverClusterGroup { @{ "ownernode" = @{ "name" = "abc-d120302" } } } Mock -ModuleName "Moc" Invoke-Command { @{ "TcpTestSucceeded" = $true } } Mock -ModuleName "Moc" Get-CloudFqdn { "abc-d120302.contoso.com" } $result = Test-MocNodeConnectivity -Cluster $cluster $result.count | Should -Be 4 $result[0].Severity | Should -Be "INFORMATIONAL" $result[0].Status | Should -Be "SUCCESS" $result[0].TargetResourceName | Should -Be "abc-d120301" $result[0].Description | Should -Be "TCP test from CloudAgent Node abc-d120302 to nodeagent abc-d120301.contoso.com succeeded" $result[1].Severity | Should -Be "INFORMATIONAL" $result[1].Status | Should -Be "SUCCESS" $result[1].TargetResourceName | Should -Be "abc-d120302" $result[1].Description | Should -Be "TCP test from CloudAgent Node abc-d120302 to nodeagent abc-d120302.contoso.com succeeded" $result[2].Severity | Should -Be "INFORMATIONAL" $result[2].Status | Should -Be "SUCCESS" $result[2].TargetResourceName | Should -Be "abc-d120301" $result[2].Description | Should -Be "TCP test from cluster node abc-d120301 to CloudAgent fqdn abc-d120302.contoso.com succeeded" $result[3].Severity | Should -Be "INFORMATIONAL" $result[3].Status | Should -Be "SUCCESS" $result[3].TargetResourceName | Should -Be "abc-d120302" $result[3].Description | Should -Be "TCP test from cluster node abc-d120302 to CloudAgent fqdn abc-d120302.contoso.com succeeded" } It "Should fail when Invoke-Command throws an exception" { Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "mockName" } } Mock -ModuleName "Moc" Get-FailoverCluster { @{ "domain" = "contoso.com" } } Mock -ModuleName "Moc" Get-FailoverClusterGroup { @{ "ownernode" = @{ "name" = "abc-d120302" } } } Mock -ModuleName "Moc" Invoke-Command { throw "Mock exception" } Mock -ModuleName "Moc" Get-CloudFqdn { "abc-d120302.contoso.com" } $result = Test-MocNodeConnectivity -Cluster $cluster $result.count | Should -Be 4 $result[0].Severity | Should -Be "CRITICAL" $result[0].Status | Should -Be "ERROR" $result[0].TargetResourceName | Should -Be "abc-d120301" $result[0].Description | Should -Be "Unable to check CloudAgent Node to Nodeagent connection due to exception: Mock exception" $result[1].Severity | Should -Be "CRITICAL" $result[1].Status | Should -Be "ERROR" $result[1].TargetResourceName | Should -Be "abc-d120302" $result[1].Description | Should -Be "Unable to check CloudAgent Node to Nodeagent connection due to exception: Mock exception" $result[2].Severity | Should -Be "CRITICAL" $result[2].Status | Should -Be "ERROR" $result[2].TargetResourceName | Should -Be "abc-d120301" $result[2].Description | Should -Be "Unable to check Nodeagent Node to Cloudagent connection due to exception: Mock exception" $result[3].Severity | Should -Be "CRITICAL" $result[3].Status | Should -Be "ERROR" $result[3].TargetResourceName | Should -Be "abc-d120302" $result[3].Description | Should -Be "Unable to check Nodeagent Node to Cloudagent connection due to exception: Mock exception" } } Describe "Verify Test-MOCUpdateReadinessInternal" { It "Should return null if not a failover cluster" { Mock -ModuleName "Moc" Get-FailoverCluster { $null } $result = Test-MOCUpdateReadinessInternal $result | Should -BeNullOrEmpty } It "Should skip HCI test in case of non-HCI setup" { Mock -ModuleName "Moc" Get-FailoverCluster { $null } # return something not matching the HCI SKU (406) Mock -ModuleName "Moc" Get-CimInstance { @{ "OperatingSystemSKU" = 8 } } Mock -ModuleName "Moc" Test-MocNodesMatchFailoverClusterNodes {} Mock -ModuleName "Moc" Test-ProxyEnvironmentVariables {} Mock -ModuleName "Moc" Test-MocNodeConnectivity {} Mock -ModuleName "Moc" Test-HCIClusterAndNodeRegistration {} $result = Test-MOCUpdateReadinessInternal -Cluster $cluster Should -Invoke Test-HCIClusterAndNodeRegistration -ModuleName "Moc" -Exactly 0 Should -Invoke Test-ProxyEnvironmentVariables -ModuleName "Moc" -Exactly 1 # If we fix and add back Test-EventLogServiceOnNodes, we should either mock it # or increment the count here, since it calls Get-FailoverCluster Should -Invoke Get-FailoverCluster -ModuleName "Moc" -Exactly 0 } } Describe "Verify Confirm-MocAgentReadiness" { It "Should return null when there is no failover cluster" { Mock -ModuleName "Moc" Get-FailoverCluster { $null } Mock -ModuleName "Moc" Get-ClusterGroup {} $result = Confirm-MocAgentReadiness $result | Should -BeNullOrEmpty # ensure we never get to the point of calling Get-ClusterGroup Should -Invoke Get-ClusterGroup -ModuleName "Moc" -Exactly 0 } It "Should succeed when all agents are already online" { # just return anything non-null Mock -ModuleName "Moc" Get-FailoverCluster { @{ "Name" = $cluster } } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "test" } } Mock -ModuleName "Moc" Get-ClusterGroup { @{ "State" = "Online" } } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Invoke-Command { $false } -ParameterFilter { $ComputerName -eq "abc-d120301"} Mock -ModuleName "Moc" Invoke-Command { $false } -ParameterFilter { $ComputerName -eq "abc-d120302"} Mock -ModuleName "Moc" Start-ClusterGroup {} Mock -ModuleName "Moc" Wait-ForActiveNodes {} Confirm-MocAgentReadiness # make sure this is never called, indicating we took no action Should -Invoke Start-ClusterGroup -ModuleName "Moc" -Exactly 0 Should -Invoke Wait-ForActiveNodes -ModuleName "Moc" -Exactly 0 } It "Should exit early but not throw when cloudagent does not become ready" { # just return anything non-null Mock -ModuleName "Moc" Get-FailoverCluster { @{ "Name" = $cluster } } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "test" } } Mock -ModuleName "Moc" Get-ClusterGroup { @{ "State" = "Offline" } } Mock -ModuleName "Moc" Get-ClusterNode {} Mock -ModuleName "Moc" Start-ClusterGroup {} Mock -ModuleName "Moc" Wait-ForCloudAgentEndpoint { throw "Test exception" } { Confirm-MocAgentReadiness } | Should -Not -Throw Should -Invoke Start-ClusterGroup -ModuleName "Moc" -Exactly 1 # make sure this is never called, indicating we exited prior to this Should -Invoke Get-ClusterNode -ModuleName "Moc" -Exactly 0 } It "Should wait for cloudagent and nodeagent if they are offline" { # just return anything non-null Mock -ModuleName "Moc" Get-FailoverCluster { @{ "Name" = $cluster } } Mock -ModuleName "Moc" Get-ClusterGroup { @{ "State" = "Offline" } } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "test" } } Mock -ModuleName "Moc" Start-ClusterGroup {} Mock -ModuleName "Moc" Wait-ForCloudAgentEndpoint { $null } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Invoke-Command { $false } -ParameterFilter { $ComputerName -eq "abc-d120301"} Mock -ModuleName "Moc" Invoke-Command { $true } -ParameterFilter { $ComputerName -eq "abc-d120302"} Mock -ModuleName "Moc" Wait-ForActiveNodes { $true } { Confirm-MocAgentReadiness } | Should -Not -Throw Should -Invoke Start-ClusterGroup -ModuleName "Moc" -Exactly 1 Should -Invoke Wait-ForCloudAgentEndpoint -ModuleName "Moc" -Exactly 1 Should -Invoke Wait-ForActiveNodes -ModuleName "Moc" -Exactly 1 } It "Should not throw even if we have an unexpected exception" { # just return anything non-null Mock -ModuleName "Moc" Get-FailoverCluster { @{ "Name" = $cluster } } Mock -ModuleName "Moc" Get-MocConfig { @{ "clusterRoleName" = "test" } } Mock -ModuleName "Moc" Get-ClusterGroup { @{ "State" = "Offline" } } Mock -ModuleName "Moc" Start-ClusterGroup {} Mock -ModuleName "Moc" Wait-ForCloudAgentEndpoint { $null } Mock -ModuleName "Moc" Get-ClusterNode { @( @{ "Name" = "abc-d120301" "State" = "Up" }, @{ "Name" = "abc-d120302" "State" = "Up" } ) } Mock -ModuleName "Moc" Invoke-Command { $false } -ParameterFilter { $ComputerName -eq "abc-d120301"} Mock -ModuleName "Moc" Invoke-Command { $true } -ParameterFilter { $ComputerName -eq "abc-d120302"} Mock -ModuleName "Moc" Wait-ForActiveNodes { throw "unexpected failure" } { Confirm-MocAgentReadiness } | Should -Not -Throw Should -Invoke Start-ClusterGroup -ModuleName "Moc" -Exactly 1 Should -Invoke Wait-ForCloudAgentEndpoint -ModuleName "Moc" -Exactly 1 Should -Invoke Wait-ForActiveNodes -ModuleName "Moc" -Exactly 1 } } # SIG # Begin signature block # MIInRAYJKoZIhvcNAQcCoIInNTCCJzECAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCDlnO6PHWNmcw/Y # O9ysN9XMYZf+HSn5mgbDgmF7M2FXwaCCDLowggX1MIID3aADAgECAhMzAAACHU0Z # yE7XD1dIAAAAAAIdMA0GCSqGSIb3DQEBCwUAMFcxCzAJBgNVBAYTAlVTMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBD # b2RlIFNpZ25pbmcgUENBIDIwMjQwHhcNMjYwNDE2MTg1OTQzWhcNMjcwNDE1MTg1 # OTQzWjB0MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYD # VQQDExVNaWNyb3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IB # DwAwggEKAoIBAQDQvewXxx9gZZFC6Ys1WBay8BJ8kGA4JQnH5CMafqOASlTpK9H8 # o5ZXTXt0caVQTNMUPt445wXYD+dFtaKWTwDn1I52oUSrC9vJin1Gsqt+zyKJL5Dg # 3eQXbQNR61DmMy20GLTIO3SFed9Rfi/ophgCLGFLDR3r0KvHjwMb/jYWS0celV/4 # Lz27LfAekm8v9E5IXaeiXbAUYZKK090n4CVl3JBtbN+9DtI9SNu/yjvozW52/u7R # X/Ttpa/KDlpuokZ+Zcbvmtd9ur9gFLvZzh41o9MsE/clQtdaFWGvuo6Jua/ntpgk # ey3E5/vBFe+MJPG6phdnuo6r57ZudCudiI1bAgMBAAGjggGbMIIBlzAOBgNVHQ8B # Af8EBAMCB4AwHwYDVR0lBBgwFgYKKwYBBAGCN0wIAQYIKwYBBQUHAwMwHQYDVR0O # BBYEFH6QuMwqcPG0hQlQ6c5jCtTTLrVeMEUGA1UdEQQ+MDykOjA4MR4wHAYDVQQL # ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xFjAUBgNVBAUTDTIzMDAxMis1MDc1NTkw # HwYDVR0jBBgwFoAUf1k/VCHarU/vBeXmo9ctBpQSCDEwYAYDVR0fBFkwVzBVoFOg # UYZPaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9jcmwvTWljcm9zb2Z0 # JTIwQ29kZSUyMFNpZ25pbmclMjBQQ0ElMjAyMDI0LmNybDBtBggrBgEFBQcBAQRh # MF8wXQYIKwYBBQUHMAKGUWh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMv # Y2VydHMvTWljcm9zb2Z0JTIwQ29kZSUyMFNpZ25pbmclMjBQQ0ElMjAyMDI0LmNy # dDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4ICAQBKTbYOjzwTG/DXGaz9 # s6+fQeaTtDcFmMY+5UyVFCyj7Pv+5i37qfX8lSL/tBIfYQfWsMuBQlfZurJD6r4H # VJ2CeH+1fgiq8dcHdVKoZ3Sa2qXoX3cq9iS8cVb06B7+5/XJ7I0OxHH9fDsvJ3T3 # w5V/ZtAIFmLrl+P0CtG+92uzRsn0nTbdFjOkLMLWPLAU3THohKRlSEMgFJpPkm5n # 5UAZ35xX6FWCrDLsSKb555bTifwa8mJBwdlof0bmfYidH+dxZ1FdDxvLnNl9zeKs # A4kejaaIqqIPguhwAti5Ql7BlTNoJNwxCvBmqW2MQLnCkYN/VVUsR3V2x/rcTNzo # Bf/Z/SpROvdaA2ZOOd1uioXJt3tdLQ7vHpqpib0KfWr/FWXW10q38VxfCnRQBqzb # SuztR7nEMuzX7Ck+B/XaPDXd1qh72+QYyB0Z2VzWmO9zsnb9Uq/dwu8LGeQqnyu6 # 7SDGACvnXii2fb9+US492VTnXSnFKyqwgzUyFMtZK1/sHYTv6bG4TtQUygQxTN+Z # V+aJIlKO2MqZ7bKrAnOzS9m6NgoTdWOq11bTOZwKlIEV/EhV9SWkDmdpR/hPPT2v # 6TEj4F8PT/zHjRezIU5c/DGlt/VhY/pK0XkJtEyMmmS1BMtjU/rqBZVMIm3dnxQs # /TBByr+Cf8Z1r7aifQVQ+WSqzjCCBr0wggSloAMCAQICEzMAAAA5O7Y3Gb8GHWcA # AAAAADkwDQYJKoZIhvcNAQEMBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpX # YXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQg # Q29ycG9yYXRpb24xMjAwBgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmljYXRl # IEF1dGhvcml0eSAyMDExMB4XDTI0MDgwODIwNTQxOFoXDTM2MDMyMjIyMTMwNFow # VzELMAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEo # MCYGA1UEAxMfTWljcm9zb2Z0IENvZGUgU2lnbmluZyBQQ0EgMjAyNDCCAiIwDQYJ # KoZIhvcNAQEBBQADggIPADCCAgoCggIBANgBnB7jOMeqlRYHNa265v4IY9fH8TKh # emHfPINe1gpLaV3dhg324WwH06LcHbpnsBukCDNitryo0dtS/EW6I/yEL/bLSY8h # KpbfQuWusBPr9qazYcDxCW/qnjb5JsI1s8bNOg3bVATvQVL4tcf03aTycsz8QeCd # M0l/yHRObJ9QqazM1r6VPEOJ7LL+uEEb73w6QCuhs89a1uv1zerOYMnsneRRwCbp # yW11IcggU0cRKDDq1pjVJzIbIF6+oiXXbReOsgeI8zu1FyQfK0fVkaya8SmVHQ/t # Of23mZ4W9k0Ri22QW9p3UgSC5OUDktKxxcCmGL6tXLfOGSWHIIV4YrTJTT6PNty5 # REojHJuZHArkF9VnHTERWoTjAzfI3kP+5b4alUdhgAZ7ttOu1bVnXfHaqPYl2rPs # 20ji03LOVWsh/radgE17es5hL+t6lV0eVHrVhsssROWJuz2MXMCt7iw7lFPG9LXK # Gjsmonn2gotGdHIuEg5JnJMJVmixd5LRlkmgYRZKzhxSCwyoGIq0PhaA7Y+VPct5 # pCHkijcIIDm0nlkK+0KyepolcqGm0T/GYQRMhHJlGOOmVQop36wUVUYklUy++vDW # eEgEo4s7hxN6mIbf2MSIQ/iIfMZgJxC69oukMUXCrOC3SkE/xIkgpfl22MM1itkZ # 35nNXkMolU1lAgMBAAGjggFOMIIBSjAOBgNVHQ8BAf8EBAMCAYYwEAYJKwYBBAGC # NxUBBAMCAQAwHQYDVR0OBBYEFH9ZP1Qh2q1P7wXl5qPXLQaUEggxMBkGCSsGAQQB # gjcUAgQMHgoAUwB1AGIAQwBBMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU # ci06AjGQQ7kUBU7h6qfHMdEjiTQwWgYDVR0fBFMwUTBPoE2gS4ZJaHR0cDovL2Ny # bC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0MjAx # MV8yMDExXzAzXzIyLmNybDBeBggrBgEFBQcBAQRSMFAwTgYIKwYBBQUHMAKGQmh0 # dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2VydHMvTWljUm9vQ2VyQXV0MjAx # MV8yMDExXzAzXzIyLmNydDANBgkqhkiG9w0BAQwFAAOCAgEAFJQfOChP7onn6fLI # MKrSlN1WYKwDFgAddymOUO3FrM8d7B/W/iQ6DxXsDn7D5W4wMwYeLystcEqfkjz4 # NURRgazyMu5yRzQh4LqjA4tStTcJh1opExo7nn5PuPBYnbu0+THSuVHTe0VTTPVh # ily/piFrDo3axQ9P4C+Ol5yet+2gTfekICS5xS+cYfSIvgn0JksVBVMYVI5QFu/q # hnLhsEFEUzG8fvv0hjgkO+lkpV9ty6GkN4vdnd7ya6Q6aR9y34aiM1qmxaxBi6OU # nyNl6fkuun/diTFnYDLTppOkr/mg5WSfCiDVMNCxtj4wPKC5OmHm1DQIt/MNokbb # H3UGsFP1QbzsLocuSqLCvH09Io3fDPTmscR9Y75G4qX7RTX8AdBPo0I6OEojf39z # uFZt0qOHm65YWQE69cZM2ueE1MB05dNNgHK9gTE7zKvK/fg8B2qjW88MT/WF5V5u # vZGtqa9FSL2RazArA+rDPuf6JGYz4HpgMZHB4S6szWSKYBv0VisCzfxgeU+dquXW # 9bd0auYlOB58DPcOYKdc3Se94g+xL4pcEhbB54JOgAkwYTu/9dLeH2pDqeJZAABV # DWRQCaXfO5LgyKwKCLYXpigrZYCjUSBcr+Ve8PFWMhVTQl0v4q8J/AUmQN5W4n10 # 1cY2L4A7GTQG1h32HHAvfQESWP0xghngMIIZ3AIBATBuMFcxCzAJBgNVBAYTAlVT # MR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jv # c29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMjQCEzMAAAIdTRnITtcPV0gAAAAAAh0w # DQYJYIZIAWUDBAIBBQCggZAwGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwLwYJ # KoZIhvcNAQkEMSIEILPGgMaaFBtNmwAskUPgYPyZNG4nRrGulRASht8z2hfTMEIG # CisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8AcwBvAGYAdKEagBhodHRwOi8v # d3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEBBQAEggEAhv6pCY/bBPmo5JB1 # Tl9lvX6qL/yxrMq85i0OJQ2w7dSwzAEHqh51Omdvp88yA+wvCtcZCcPFPNX7KKHN # geBPcDjRjpQstyV9PerTuapsYkwrtQN7zqYP/viCG5bNfFGyykTGkeOsItymMZow # tCSmQne5sXuqeW0LB/3jcOAr7z4pOoQ9AgWOUYQSHxK5ksuAUhC2Keo1/24atCfX # h8sQHbMHI1MGNSKWxR+c3NA4kGK2dFGnvX0VMTSWWfuqtvNk2BVb7fDrDu8+wZu7 # a56hjP81I9vBWdGwOhu/ZAGC2hOHkAWaEpr1Yg8IoNAsw17nK7wiQsJAHYXggyfb # z1kxlKGCF7AwghesBgorBgEEAYI3AwMBMYIXnDCCF5gGCSqGSIb3DQEHAqCCF4kw # gheFAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFaBgsqhkiG9w0BCRABBKCCAUkEggFF # MIIBQQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFlAwQCAQUABCBP7YcCOETl8mUI # vDCCv1jTUaLhbgcDqjQxxJu0+XiPPAIGahGS4/vsGBMyMDI2MDUyODAwNDE1NS43 # MDFaMASAAgH0oIHZpIHWMIHTMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu # Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExp # bWl0ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjo2NTFBLTA1RTAtRDk0NzEl # MCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaCCEf4wggcoMIIF # EKADAgECAhMzAAACFRgD04EHJnxTAAEAAAIVMA0GCSqGSIb3DQEBCwUAMHwxCzAJ # BgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25k # MR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jv # c29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMB4XDTI1MDgxNDE4NDgyMFoXDTI2MTEx # MzE4NDgyMFowgdMxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAw # DgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x # LTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJhdGlvbnMgTGltaXRlZDEn # MCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjY1MUEtMDVFMC1EOTQ3MSUwIwYDVQQD # ExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIICIjANBgkqhkiG9w0BAQEF # AAOCAg8AMIICCgKCAgEAw3HV3hVxL0lEYPV03XeNKZ517VIbgexhlDPdpXwDS0BY # txPwi4XYpZR1ld0u6cr2Xjuugdg50DUx5WHL0QhY2d9vkJSk02rE/75hcKt91m2I # h287QRxRMmFu3BF6466k8qp5uXtfe6uciq49YaS8p+dzv3uTarD4hQ8UT7La95pO # JiRqxxd0qOGLECvHLEXPXioNSx9pyhzhm6lt7ezLxJeFVYtxShkavPoZN0dOCiYe # h4KgoKoyagzMuSiLCiMUW4Ue4Qsm658FJNGTNh7V5qXYVA6k5xjw5WeWdKOz0i9A # 5jBcbY9fVOo/cA8i1bytzcDTxb3nctcly8/OYeNstkab/Isq3Cxe1vq96fIHE1+Z # GmJjka1sodwqPycVp/2tb+BjulPL5D6rgUXTPF84U82RLKHV57bB8fHRpgnjcWBQ # uXPgVeSXpERWimt0NF2lCOLzqgrvS/vYqde5Ln9YlKKhAZ/xDE0TLIIr6+I/2JTt # XP34nfjTENVqMBISWcakIxAwGb3RB5yHCxynIFNVLcfKAsEdC5U2em0fAvmVv0so # nqnv17cuaYi2eCLWhoK1Ic85Dw7s/lhcXrBpY4n/Rl5l3wHzs4vOIhu87DIy5QUa # EupEsyY0NWqgI4BWl6v1wgse+l8DWFeUXofhUuCgVTuTHN3K8idoMbn8Q3edUIEC # AwEAAaOCAUkwggFFMB0GA1UdDgQWBBSJIXfxcqAwFqGj9jdwQtdSqadj1zAfBgNV # HSMEGDAWgBSfpxVdAF5iXYP05dJlpxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQhk5o # dHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNyb3NvZnQlMjBU # aW1lLVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwG # CCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRz # L01pY3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNV # HRMBAf8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMA4GA1UdDwEB/wQEAwIH # gDANBgkqhkiG9w0BAQsFAAOCAgEAd42HtV+kGbvxzLBTC5O7vkCIBPy/BwpjCzeL # 53hAiEOebp+VdNnwm9GVCfYq3KMfrj4UvKQTUAaS5Zkwe1gvZ3ljSSnCOyS5OwNu # 9dpg3ww+QW2eOcSLkyVAWFrLn6Iig3TC/zWMvVhqXtdFhG2KJ1lSbN222csY3E3/ # BrGluAlvET9gmxVyyxNy59/7JF5zIGcJibydxs94JL1BtPgXJOfZzQ+/3iTc6eDt # maWT6DKdnJocp8wkXKWPIsBEfkD6k1Qitwvt0mHrORah75SjecOKt4oWayVLkPTh # o12e0ongEg1cje5fxSZGthrMrWKvI4R7HEC7k8maH9ePA3ViH0CVSSOefaPTGMzI # hHCo5p3jG5SMcyO3eA9uEaYQJITJlLG3BwwGmypY7C/8/nj1SOhgx1HgJ0ywOJL9 # xfP4AOcWmCfbsqgGbCaC7WH5sINdzfMar8V7YNFqkbCGUKhc8GpIyE+MKnyVn33j # suaGAlNRg7dVRUSoYLJxvUsw9GOwyBpBwbE9sqOLm+HsO00oF23PMio7WFXcFTZA # jp3ujihBAfLrXICgGOHPdkZ042u1LZqOcnlr3XzvgMe+mPPyasW8f0rtzJj3V5E/ # EKiyQlPxj9Mfq2x9himnlXWGZCVPeEBROrNbDYBfazTyLNCOTsRtksOSV3FBtPnp # QtLN754wggdxMIIFWaADAgECAhMzAAAAFcXna54Cm0mZAAAAAAAVMA0GCSqGSIb3 # DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4G # A1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTIw # MAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkgMjAx # MDAeFw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMyMjVaMHwxCzAJBgNVBAYTAlVT # MRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQK # ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1l # LVN0YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA # 5OGmTOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51yMo1V/YBf2xK4OK9uT4XYDP/ # XE/HZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY6GB9alKDRLemjkZrBxTzxXb1 # hlDcwUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9cmmvHaus9ja+NSZk2pg7uhp7 # M62AW36MEBydUv626GIl3GoPz130/o5Tz9bshVZN7928jaTjkY+yOSxRnOlwaQ3K # Ni1wjjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDuaRr3tpK56KTesy+uDRedGbsoy # 1cCGMFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74kpEeHT39IM9zfUGaRnXNxF80 # 3RKJ1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2K26oElHovwUDo9Fzpk03dJQc # NIIP8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5TI4CvEJoLhDqhFFG4tG9ahha # YQFzymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZki1ugpoMhXV8wdJGUlNi5UPkL # iWHzNgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9QBXpsxREdcu+N+VLEhReTwDwV # 2xo3xwgVGD94q0W29R6HXtqPnhZyacaue7e3PmriLq0CAwEAAaOCAd0wggHZMBIG # CSsGAQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUCBBYEFCqnUv5kxJq+gpE8RjUp # zxD/LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJlpxtTNRnpcjBcBgNVHSAEVTBT # MFEGDCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIBFjNodHRwOi8vd3d3Lm1pY3Jv # c29mdC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9yeS5odG0wEwYDVR0lBAwwCgYI # KwYBBQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwCwYDVR0PBAQDAgGG # MA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU1fZWy4/oolxiaNE9lJBb186a # GMQwVgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2NybC5taWNyb3NvZnQuY29tL3Br # aS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIwMTAtMDYtMjMuY3JsMFoGCCsG # AQUFBwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDovL3d3dy5taWNyb3NvZnQuY29t # L3BraS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcnQwDQYJKoZIhvcN # AQELBQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/ypb+pcFLY+TkdkeLEGk5c9MTO1 # OdfCcTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulmZzpTTd2YurYeeNg2LpypglYA # A7AFvonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM9W0jVOR4U3UkV7ndn/OOPcbz # aN9l9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECWOKz3+SmJw7wXsFSFQrP8DJ6L # GYnn8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4FOmRsqlb30mjdAy87JGA0j3m # Sj5mO0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3UwxTSwethQ/gpY3UA8x1RtnWN0 # SCyxTkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPXfx5bRAGOWhmRaw2fpCjcZxko # JLo4S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVXVAmxaQFEfnyhYWxz/gq77EFm # PWn9y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGConsXHRWJjXD+57XQKBqJC482 # 2rpM+Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU5nR0W2rRnj7tfqAxM328y+l7 # vzhwRNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEGahC0HVUzWLOhcGbyoYIDWTCC # AkECAQEwggEBoYHZpIHWMIHTMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu # Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExp # bWl0ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjo2NTFBLTA1RTAtRDk0NzEl # MCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsO # AwIaAxUAj6eTejbuYE1Ifjbfrt6tXevCUSCggYMwgYCkfjB8MQswCQYDVQQGEwJV # UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE # ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGlt # ZS1TdGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAO3B/60wIhgPMjAyNjA1 # MjcyMzQxMzNaGA8yMDI2MDUyODIzNDEzM1owdzA9BgorBgEEAYRZCgQBMS8wLTAK # AgUA7cH/rQIBADAKAgEAAgIHRgIB/zAHAgEAAgIS7jAKAgUA7cNRLQIBADA2Bgor # BgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAID # AYagMA0GCSqGSIb3DQEBCwUAA4IBAQC0/rU4R9nuI7TZt7k+F3misvKw3tucw9Rz # SbKiwoCQePkR9i2/Nm8/InMFc6jeU/Zb5H1lq9GR8U7wtdHbhxa8inzX5xvVj8QJ # hOJ38DKLn4Si5R/F4kgssV2+jhtafZKt3Gx5fp9nDAGy33ENiMfXYQekEdrQIcvl # Imed0bqptwBJIbDnukHd09W76tQUut5U6Y7NyZrY6ZgnljzLBBz+1vl0d19sAWqU # puN7syjEMURKnU+neNmL3iMoGQCCUUltZFhcJLOZdd8JJY8h6NiHlbd5DVZUkRu/ # iAG3o6rDbb6y2xbmKJvlcZLGKSbR1AtGFodBzsXOAha7o3gMBheOMYIEDTCCBAkC # AQEwgZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNV # BAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQG # A1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAIVGAPTgQcm # fFMAAQAAAhUwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG # 9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQg9O1OZxSUv2aJw1pKHzOrUDakmDDH7bnr # CcFdY2fFva4wgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCBwEPR2PDrTFLcr # tQsKrUi7oz5JNRCF/KRHMihSNe7sijCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMw # EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN # aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0 # YW1wIFBDQSAyMDEwAhMzAAACFRgD04EHJnxTAAEAAAIVMCIEIHidcqWwrlx/BG9R # sEBcGvcQ2lsq6tH3oMWtmtp3NlDwMA0GCSqGSIb3DQEBCwUABIICAI2u50CsWzWa # cHOpcK5WcUo4YGyc1P23WG/DvwTIVSbQu5G2OcD8WnicJR6FjeD6EZ3ShMw94PAf # pFFfZyaGk3Uj7+ix1XHjqaRxQgEfJfwncnb/eI3xBhLT4pZg1IYqoiJuwcF1aSwc # VcjsChYbtg3fpzmUWUu3KfwTSLc25jvZMbZ0J1ZwJoVPj+Yok4KV6lwUrWtIZ/kG # FtbwsiHVYy64V9nx6KKLtzZqtK5Pwef6EcUgWFFfwLRABvLmx+IPQv5H2IUQOn76 # gBMNPcO31lg6OJVdG3JxnHfgkJElqc/Efs4QFRi97VEcP4Pevo594M3Gv6RGNrvw # uGK58paH+FD2V/lDYEMcZjsm55zbIJxGt1SNg9hC4sUZpXclBeKH6vj5pI1z9e2N # anMxTb8no05EuWU/SnE/buNx2ocOHJrAFORcyxoz0JV9QFkYrLcg2Y6ey7sOYplR # FPPFszzQ3tcZVgvdbEMP+p7AGRvXld6VY7o0hNeSsIInSXlgNGe8PGKIJCxmqNKZ # DJeUHKV3utE+k0pded7rAvW2qvPNjZyL/S1hXVMVsimxokKxqiJh/NiTaJ0DJgc1 # zqWGaDj5NujRtQkyk/B2pfX5jdSA9d4BO7zTicox5616jy6mk7fQTJ8t4d85nzEH # DfD7WoVutdYKQQicw7ftf0eetD3oOQOB # SIG # End signature block |