From 271e799e2be864d803f8e0e6cd50908f1a4b823a Mon Sep 17 00:00:00 2001 From: ejleroy Date: Fri, 22 Feb 2019 08:58:13 -0500 Subject: [PATCH 01/10] SqlServerMemory: Fix so auto memory on Azure VMs is reported correctly (#1285) - Changes to SqlServerMemory - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) because the correct memory size was not being detected correctly on Azure VMs (issue #914). --- CHANGELOG.md | 4 ++ .../MSFT_SqlServerMemory.psm1 | 2 +- Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 | 56 ++++++------------- 3 files changed, 23 insertions(+), 39 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2669d5700..10bd510e2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,10 @@ ## 12.3.0.0 - Changes to SqlServerDsc + - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) + because the correct memory size was not being detected correctly on Azure VMs + [issue #914](https://github.com/PowerShell/SqlServerDsc/issues/914) + [issue #1154](https://github.com/PowerShell/SqlServerDsc/issues/1154) - Reverting the change that was made as part of the [issue #1260](https://github.com/PowerShell/SqlServerDsc/issues/1260) in the previous release, as it only mitigated the issue, it did not diff --git a/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 b/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 index 1efba49ec..6b15959e6 100644 --- a/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 +++ b/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 @@ -340,7 +340,7 @@ function Get-SqlDscDynamicMaxMemory { try { - $physicalMemory = ((Get-CimInstance -ClassName Win32_PhysicalMemory).Capacity | Measure-Object -Sum).Sum + $physicalMemory = (Get-CimInstance -ClassName Win32_ComputerSystem).TotalPhysicalMemory $physicalMemoryInMegaBytes = [Math]::Round($physicalMemory / 1MB) # Find how much to save for OS: 20% of total ram for under 15GB / 12.5% for over 20GB diff --git a/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 b/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 index 12d04d32f..2b46e80c1 100644 --- a/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 @@ -157,22 +157,12 @@ try } Mock -CommandName Get-CimInstance -MockWith { - $mockGetCimInstanceMem = @() - $mockGetCimInstanceMem += New-Object -TypeName PSObject -Property @{ - Name = 'Physical Memory' - Tag = 'Physical Memory 0' - Capacity = 8589934592 + New-Object -TypeName PSObject -Property @{ + TotalPhysicalMemory = 17179869184 } - $mockGetCimInstanceMem += New-Object -TypeName PSObject -Property @{ - Name = 'Physical Memory' - Tag = 'Physical Memory 1' - Capacity = 8589934592 - } - - $mockGetCimInstanceMem - } -ParameterFilter { $ClassName -eq 'Win32_PhysicalMemory' } -Verifiable + } -ParameterFilter { $ClassName -eq 'Win32_ComputerSystem' } -Verifiable Mock -CommandName Get-CimInstance -MockWith { $mockGetCimInstanceProc = [PSCustomObject]@{ @@ -317,9 +307,9 @@ try Assert-MockCalled Connect-SQL -Exactly -Times 1 -Scope Context } - It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_PhysicalMemory' { + It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_ComputerSystem' { Assert-MockCalled Get-CimInstance -Exactly -Times 1 -ParameterFilter { - $ClassName -eq 'Win32_PhysicalMemory' + $ClassName -eq 'Win32_ComputerSystem' } -Scope Context } @@ -361,9 +351,9 @@ try Assert-MockCalled Connect-SQL -Exactly -Times 1 -Scope Context } - It 'Should not call the mock function Get-CimInstance with ClassName equal to Win32_PhysicalMemory' { + It 'Should not call the mock function Get-CimInstance with ClassName equal to Win32_ComputerSystem' { Assert-MockCalled Get-CimInstance -Exactly -Times 0 -ParameterFilter { - $ClassName -eq 'Win32_PhysicalMemory' + $ClassName -eq 'Win32_ComputerSystem' } -Scope Context } @@ -401,9 +391,9 @@ try Assert-MockCalled Connect-SQL -Exactly -Times 1 -Scope Context } - It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_PhysicalMemory' { + It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_ComputerSystem' { Assert-MockCalled Get-CimInstance -Exactly -Times 1 -ParameterFilter { - $ClassName -eq 'Win32_PhysicalMemory' + $ClassName -eq 'Win32_ComputerSystem' } -Scope Context } @@ -497,22 +487,12 @@ try } Mock -CommandName Get-CimInstance -MockWith { - $mockGetCimInstanceMem = @() - - $mockGetCimInstanceMem += New-Object -TypeName PSObject -Property @{ - Name = 'Physical Memory' - Tag = 'Physical Memory 0' - Capacity = 17179869184 - } - $mockGetCimInstanceMem += New-Object -TypeName PSObject -Property @{ - Name = 'Physical Memory' - Tag = 'Physical Memory 1' - Capacity = 17179869184 + New-Object -TypeName PSObject -Property @{ + TotalPhysicalMemory = 17179869184 } - $mockGetCimInstanceMem - } -ParameterFilter { $ClassName -eq 'Win32_PhysicalMemory' } -Verifiable + } -ParameterFilter { $ClassName -eq 'Win32_ComputerSystem' } -Verifiable Mock -CommandName Get-CimInstance -MockWith { $mockGetCimInstanceProc = [PSCustomObject]@{ @@ -636,9 +616,9 @@ try Assert-MockCalled Connect-SQL -Exactly -Times 1 -Scope Context } - It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_PhysicalMemory' { + It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_ComputerSystem' { Assert-MockCalled Get-CimInstance -Exactly -Times 1 -ParameterFilter { - $ClassName -eq 'Win32_PhysicalMemory' + $ClassName -eq 'Win32_ComputerSystem' } -Scope Context } @@ -678,9 +658,9 @@ try Assert-MockCalled Connect-SQL -Exactly -Times 1 -Scope Context } - It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_PhysicalMemory' { + It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_ComputerSystem' { Assert-MockCalled Get-CimInstance -Exactly -Times 1 -ParameterFilter { - $ClassName -eq 'Win32_PhysicalMemory' + $ClassName -eq 'Win32_ComputerSystem' } -Scope Context } @@ -768,9 +748,9 @@ try Assert-MockCalled Connect-SQL -Exactly -Times 1 -Scope Context } - It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_PhysicalMemory' { + It 'Should call the mock function Get-CimInstance with ClassName equal to Win32_ComputerSystem' { Assert-MockCalled Get-CimInstance -Exactly -Times 1 -ParameterFilter { - $ClassName -eq 'Win32_PhysicalMemory' + $ClassName -eq 'Win32_ComputerSystem' } -Scope Context } From d8cefea390f22b97ded218aac0c49f3f6fa297c5 Mon Sep 17 00:00:00 2001 From: Johan Ljunggren Date: Fri, 22 Feb 2019 16:41:18 +0100 Subject: [PATCH 02/10] Fix CHANGELOG.md after recent merge (#1288) --- CHANGELOG.md | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 10bd510e2..2214bb16a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,13 +2,14 @@ ## Unreleased +- Changes to SqlServerMemory + - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) + because the correct memory size was not being detected correctly on Azure VMs + ([issue #914](https://github.com/PowerShell/SqlServerDsc/issues/914)). + ## 12.3.0.0 - Changes to SqlServerDsc - - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) - because the correct memory size was not being detected correctly on Azure VMs - [issue #914](https://github.com/PowerShell/SqlServerDsc/issues/914) - [issue #1154](https://github.com/PowerShell/SqlServerDsc/issues/1154) - Reverting the change that was made as part of the [issue #1260](https://github.com/PowerShell/SqlServerDsc/issues/1260) in the previous release, as it only mitigated the issue, it did not From c139e99003f28d17e3e13a9fa3fd193999ca365f Mon Sep 17 00:00:00 2001 From: Johan Ljunggren Date: Mon, 25 Feb 2019 08:01:29 +0100 Subject: [PATCH 03/10] SqlSetup: Integration test is using SQL Server 2017 (#1103) - Changes to SqlSetup - Split integration tests into two jobs, one for running integration tests for SQL Server 2016 and another for running integration test for SQL Server 2017 (issue #858). - Localized messages for Master Data Services no longer start and end with single quote. - When installing features a verbose message is written if a feature is found to already be installed. It no longer quietly removes the feature from the `/FEATURES` argument. - Cleaned up a bit in the tests, removed excessive piping. - Fixed minor typo in examples. - A new optional parameter `FeatureFlag` parameter was added to control breaking changes. Functionality added under a feature flag can be toggled on or off, and could be changed later to be the default. This way we can also make more of the new functionalities the default in the same breaking change release (issue #1105). - Added a new way of detecting if the shared feature CONN (Client Tools Connectivity, and SQL Client Connectivity SDK), BC (Client Tools Backwards Compatibility), and SDK (Client Tools SDK) is installed or not. The new functionality is used when the parameter `FeatureFlag` is set to `'DetectionSharedFeatures'` (issue #1105). - Added a new helper function `Get-InstalledSharedFeatures` to move out some of the code from the `Get-TargetResource` to make unit testing easier and faster. --- CHANGELOG.md | 24 + DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 | 355 ++++++++--- .../MSFT_SqlSetup/MSFT_SqlSetup.schema.mof | 1 + .../en-US/MSFT_SqlSetup.strings.psd1 | 6 +- .../1-EnableDatabaseMail.ps1 | 2 +- .../2-DisableDatabaseMail.ps1 | 2 +- ...ServerFromUncPathUsingSourceCredential.ps1 | 2 +- ...amedInstanceInFailoverClusterFirstNode.ps1 | 2 +- ...medInstanceInFailoverClusterSecondNode.ps1 | 2 +- README.md | 19 + ...SFT_SqlAgentOperator.Integration.Tests.ps1 | 2 +- .../MSFT_SqlAgentOperator.config.ps1 | 2 +- ...T_SqlAlwaysOnService.Integration.Tests.ps1 | 2 +- .../MSFT_SqlAlwaysOnService.config.ps1 | 2 +- ...abaseDefaultLocation.Integration.Tests.ps1 | 2 +- ...MSFT_SqlDatabaseDefaultLocation.config.ps1 | 2 +- .../MSFT_SqlRS.Integration.Tests.ps1 | 2 +- Tests/Integration/MSFT_SqlRS.config.ps1 | 2 +- .../MSFT_SqlScript.Integration.Tests.ps1 | 2 +- Tests/Integration/MSFT_SqlScript.config.ps1 | 2 +- .../MSFT_SqlScriptQuery.Integration.Tests.ps1 | 2 +- .../MSFT_SqlScriptQuery.config.ps1 | 2 +- ...qlServerDatabaseMail.Integration.Tests.ps1 | 2 +- .../MSFT_SqlServerDatabaseMail.config.ps1 | 2 +- ...FT_SqlServerEndPoint.Integration.Tests.ps1 | 2 +- .../MSFT_SqlServerEndPoint.config.ps1 | 2 +- .../MSFT_SqlServerLogin.Integration.Tests.ps1 | 2 +- .../MSFT_SqlServerLogin.config.ps1 | 2 +- ...SFT_SqlServerNetwork.Integration.Tests.ps1 | 2 +- .../MSFT_SqlServerNetwork.config.ps1 | 2 +- .../MSFT_SqlServerRole.Integration.Tests.ps1 | 2 +- .../Integration/MSFT_SqlServerRole.config.ps1 | 2 +- ...rverSecureConnection.Integration.Tests.ps1 | 2 +- .../MSFT_SqlServerSecureConnection.config.ps1 | 2 +- ...FT_SqlServiceAccount.Integration.Tests.ps1 | 2 +- .../MSFT_SqlServiceAccount.config.ps1 | 2 +- ...FT_SqlSetup._SQL2017.Integration.Tests.ps1 | 603 ++++++++++++++++++ ...FT_SqlSetup_SQL2016.Integration.Tests.ps1} | 4 +- ...g.ps1 => MSFT_SqlSetup_SQL2016.config.ps1} | 6 +- .../MSFT_SqlSetup_SQL2017.config.ps1 | 448 +++++++++++++ Tests/Integration/README.md | 24 +- Tests/TestHelpers/CommonTestHelper.psm1 | 16 +- Tests/Unit/MSFT_SqlSetup.Tests.ps1 | 312 ++++++++- appveyor.yml | 8 +- 44 files changed, 1737 insertions(+), 151 deletions(-) create mode 100644 Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 rename Tests/Integration/{MSFT_SqlSetup.Integration.Tests.ps1 => MSFT_SqlSetup_SQL2016.Integration.Tests.ps1} (99%) rename Tests/Integration/{MSFT_SqlSetup.config.ps1 => MSFT_SqlSetup_SQL2016.config.ps1} (99%) create mode 100644 Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 diff --git a/CHANGELOG.md b/CHANGELOG.md index 2214bb16a..abe7546a7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,30 @@ - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) because the correct memory size was not being detected correctly on Azure VMs ([issue #914](https://github.com/PowerShell/SqlServerDsc/issues/914)). +- Changes to SqlSetup + - Split integration tests into two jobs, one for running integration tests + for SQL Server 2016 and another for running integration test for + SQL Server 2017 ([issue #858](https://github.com/PowerShell/SqlServerDsc/issues/858)). + - Localized messages for Master Data Services no longer start and end with + single quote. + - When installing features a verbose message is written if a feature is found + to already be installed. It no longer quietly removes the feature from the + `/FEATURES` argument. + - Cleaned up a bit in the tests, removed excessive piping. + - Fixed minor typo in examples. + - A new optional parameter `FeatureFlag` parameter was added to control + breaking changes. Functionality added under a feature flag can be + toggled on or off, and could be changed later to be the default. + This way we can also make more of the new functionalities the default + in the same breaking change release ([issue #1105](https://github.com/PowerShell/SqlServerDsc/issues/1105)). + - Added a new way of detecting if the shared feature CONN (Client Tools + Connectivity, and SQL Client Connectivity SDK), BC (Client Tools + Backwards Compatibility), and SDK (Client Tools SDK) is installed or + not. The new functionality is used when the parameter `FeatureFlag` + is set to `'DetectionSharedFeatures'` ([issue #1105](https://github.com/PowerShell/SqlServerDsc/issues/1105)). + - Added a new helper function `Get-InstalledSharedFeatures` to move out + some of the code from the `Get-TargetResource` to make unit testing + easier and faster. ## 12.3.0.0 diff --git a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 index 158748842..6f59e3715 100644 --- a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 +++ b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 @@ -31,6 +31,11 @@ $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlSetup' .PARAMETER FailoverClusterNetworkName Host name to be assigned to the clustered SQL Server instance. + + .PARAMETER FeatureFlag + Feature flags are used to toggle functionality on or off. See the + documentation for what additional functionality exist through a feature + flag. #> function Get-TargetResource { @@ -57,9 +62,18 @@ function Get-TargetResource [Parameter()] [System.String] - $FailoverClusterNetworkName + $FailoverClusterNetworkName, + + [Parameter()] + [System.String[]] + $FeatureFlag ) + if ($FeatureFlag) + { + Write-Verbose -Message ($script:localizedData.FeatureFlag -f ($FeatureFlag -join ''',''')) + } + if ($Action -in @('CompleteFailoverCluster','InstallFailoverCluster','Addnode')) { $sqlHostName = $FailoverClusterNetworkName @@ -153,22 +167,6 @@ function Get-TargetResource Write-Verbose -Message $script:localizedData.ReplicationFeatureNotFound } - # Check if Data Quality Client sub component is configured - $dataQualityClientRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\ConfigurationState" - - Write-Verbose -Message ($script:localizedData.EvaluateDataQualityClientFeature -f $dataQualityClientRegistryPath) - - $isDQCInstalled = (Get-ItemProperty -Path $dataQualityClientRegistryPath).SQL_DQ_CLIENT_Full - if ($isDQCInstalled -eq 1) - { - Write-Verbose -Message $script:localizedData.DataQualityClientFeatureFound - $features += 'DQC,' - } - else - { - Write-Verbose -Message $script:localizedData.DataQualityClientFeatureNotFound - } - # Check if Data Quality Services sub component is configured $dataQualityServicesRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\DQ\*" @@ -185,6 +183,25 @@ function Get-TargetResource Write-Verbose -Message $script:localizedData.DataQualityServicesFeatureNotFound } + if (-not (Test-FeatureFlag -FeatureFlag $FeatureFlag -TestFlag 'DetectionSharedFeatures')) + { + # Check if Data Quality Client sub component is configured + $dataQualityClientRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\ConfigurationState" + + Write-Verbose -Message ($script:localizedData.EvaluateDataQualityClientFeature -f $dataQualityClientRegistryPath) + + $isDQCInstalled = (Get-ItemProperty -Path $dataQualityClientRegistryPath).SQL_DQ_CLIENT_Full + if ($isDQCInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.DataQualityClientFeatureFound + $features += 'DQC,' + } + else + { + Write-Verbose -Message $script:localizedData.DataQualityClientFeatureNotFound + } + } + $instanceId = $fullInstanceId.Split('.')[1] $instanceDirectory = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$fullInstanceId\Setup" -Name 'SqlProgramDir').SqlProgramDir.Trim("\") @@ -199,20 +216,18 @@ function Get-TargetResource # Tempdb data files size $SqlTempdbFileSize = ($tempdbPrimaryFilegroup.Files.Size | Measure-Object -Average).Average / 1Kb - + # Tempdb data files growth $SqlTempdbFileGrowth = ($tempdbPrimaryFilegroup.Files.Growth | Measure-Object -Average).Average / 1Kb - + # Tempdb log file size $tempdbTempLog = ($databaseServer.Databases | Where-Object {$_.Name -eq 'tempdb'}).LogFiles | Where-Object {$_.Name -eq 'templog'} $SqlTempdbLogFileSize = $tempdbTempLog.Size / 1Kb - + # Tempdb log file growth $SqlTempdbLogFileGrowth = $tempdbTempLog.Growth / 1Kb } - - $sqlCollation = $databaseServer.Collation $sqlSystemAdminAccounts = @() @@ -362,71 +377,80 @@ function Get-TargetResource Write-Verbose -Message $script:localizedData.IntegrationServicesFeatureNotFound } - # Check if Documentation Components "BOL" is configured - $documentationComponentsRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\ConfigurationState" + if (-not (Test-FeatureFlag -FeatureFlag $FeatureFlag -TestFlag 'DetectionSharedFeatures')) + { + # Check if Documentation Components "BOL" is configured + $documentationComponentsRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\ConfigurationState" - Write-Verbose -Message ($script:localizedData.EvaluateDocumentationComponentsFeature -f $documentationComponentsRegistryPath) + Write-Verbose -Message ($script:localizedData.EvaluateDocumentationComponentsFeature -f $documentationComponentsRegistryPath) - $isBOLInstalled = (Get-ItemProperty -Path $documentationComponentsRegistryPath -ErrorAction SilentlyContinue).SQL_BOL_Components - if ($isBOLInstalled -eq 1) - { - Write-Verbose -Message $script:localizedData.DocumentationComponentsFeatureFound - $features += 'BOL,' - } - else - { - Write-Verbose -Message $script:localizedData.DocumentationComponentsFeatureNotFound - } + $isBOLInstalled = (Get-ItemProperty -Path $documentationComponentsRegistryPath -ErrorAction SilentlyContinue).SQL_BOL_Components + if ($isBOLInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.DocumentationComponentsFeatureFound + $features += 'BOL,' + } + else + { + Write-Verbose -Message $script:localizedData.DocumentationComponentsFeatureNotFound + } - $clientComponentsFullRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\Tools\Setup\Client_Components_Full" - $registryClientComponentsFullFeatureList = (Get-ItemProperty -Path $clientComponentsFullRegistryPath -ErrorAction SilentlyContinue).FeatureList + $clientComponentsFullRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\Tools\Setup\Client_Components_Full" + $registryClientComponentsFullFeatureList = (Get-ItemProperty -Path $clientComponentsFullRegistryPath -ErrorAction SilentlyContinue).FeatureList - Write-Verbose -Message ($script:localizedData.EvaluateClientConnectivityToolsFeature -f $clientComponentsFullRegistryPath) + Write-Verbose -Message ($script:localizedData.EvaluateClientConnectivityToolsFeature -f $clientComponentsFullRegistryPath) - if ($registryClientComponentsFullFeatureList -like '*Connectivity_FNS=3*') - { - Write-Verbose -Message $script:localizedData.ClientConnectivityToolsFeatureFound - $features += 'CONN,' - } - else - { - Write-Verbose -Message $script:localizedData.ClientConnectivityToolsFeatureNotFound - } + if ($registryClientComponentsFullFeatureList -like '*Connectivity_FNS=3*') + { + Write-Verbose -Message $script:localizedData.ClientConnectivityToolsFeatureFound + $features += 'CONN,' + } + else + { + Write-Verbose -Message $script:localizedData.ClientConnectivityToolsFeatureNotFound + } - Write-Verbose -Message ($script:localizedData.EvaluateClientConnectivityBackwardsCompatibilityToolsFeature -f $clientComponentsFullRegistryPath) - if ($registryClientComponentsFullFeatureList -like '*Tools_Legacy_FNS=3*') - { - Write-Verbose -Message $script:localizedData.ClientConnectivityBackwardsCompatibilityToolsFeatureFound - $features += 'BC,' - } - else - { - Write-Verbose -Message $script:localizedData.ClientConnectivityBackwardsCompatibilityToolsFeatureNotFound - } + Write-Verbose -Message ($script:localizedData.EvaluateClientConnectivityBackwardsCompatibilityToolsFeature -f $clientComponentsFullRegistryPath) + if ($registryClientComponentsFullFeatureList -like '*Tools_Legacy_FNS=3*') + { + Write-Verbose -Message $script:localizedData.ClientConnectivityBackwardsCompatibilityToolsFeatureFound + $features += 'BC,' + } + else + { + Write-Verbose -Message $script:localizedData.ClientConnectivityBackwardsCompatibilityToolsFeatureNotFound + } - Write-Verbose -Message ($script:localizedData.EvaluateClientToolsSdkFeature -f $clientComponentsFullRegistryPath) - if (($registryClientComponentsFullFeatureList -like '*SDK_Full=3*') -and ($registryClientComponentsFullFeatureList -like '*SDK_FNS=3*')) - { - Write-Verbose -Message $script:localizedData.ClientToolsSdkFeatureFound - $features += 'SDK,' - } - else - { - Write-Verbose -Message $script:localizedData.ClientToolsSdkFeatureNotFound - } + Write-Verbose -Message ($script:localizedData.EvaluateClientToolsSdkFeature -f $clientComponentsFullRegistryPath) + if (($registryClientComponentsFullFeatureList -like '*SDK_Full=3*') -and ($registryClientComponentsFullFeatureList -like '*SDK_FNS=3*')) + { + Write-Verbose -Message $script:localizedData.ClientToolsSdkFeatureFound + $features += 'SDK,' + } + else + { + Write-Verbose -Message $script:localizedData.ClientToolsSdkFeatureNotFound + } - # Check if MDS sub component is configured for this server - $masterDataServicesFullRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\ConfigurationState" - Write-Verbose -Message ($script:localizedData.EvaluateMasterDataServicesFeature -f $masterDataServicesFullRegistryPath) - $isMDSInstalled = (Get-ItemProperty -Path $masterDataServicesFullRegistryPath -ErrorAction SilentlyContinue).MDSCoreFeature - if ($isMDSInstalled -eq 1) - { - Write-Verbose -Message $script:localizedData.MasterDataServicesFeatureFound - $features += 'MDS,' + # Check if MDS sub component is configured for this server + $masterDataServicesFullRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($sqlVersion)0\ConfigurationState" + Write-Verbose -Message ($script:localizedData.EvaluateMasterDataServicesFeature -f $masterDataServicesFullRegistryPath) + $isMDSInstalled = (Get-ItemProperty -Path $masterDataServicesFullRegistryPath -ErrorAction SilentlyContinue).MDSCoreFeature + if ($isMDSInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.MasterDataServicesFeatureFound + $features += 'MDS,' + } + else + { + Write-Verbose -Message $script:localizedData.MasterDataServicesFeatureNotFound + } } - else + + if ((Test-FeatureFlag -FeatureFlag $FeatureFlag -TestFlag 'DetectionSharedFeatures')) { - Write-Verbose -Message $script:localizedData.MasterDataServicesFeatureNotFound + $installedSharedFeatures = Get-InstalledSharedFeatures -SqlVersion $sqlVersion + $features += '{0},' -f ($installedSharedFeatures -join ',') } $registryUninstallPath = 'HKLM:\Software\Microsoft\Windows\CurrentVersion\Uninstall' @@ -733,6 +757,11 @@ function Get-TargetResource .PARAMETER SetupProcessTimeout The timeout, in seconds, to wait for the setup process to finish. Default value is 7200 seconds (2 hours). If the setup process does not finish before this time, and error will be thrown. + + .PARAMETER FeatureFlag + Feature flags are used to toggle functionality on or off. See the + documentation for what additional functionality exist through a feature + flag. #> function Set-TargetResource { @@ -976,7 +1005,11 @@ function Set-TargetResource [Parameter()] [System.UInt32] - $SetupProcessTimeout = 7200 + $SetupProcessTimeout = 7200, + + [Parameter()] + [System.String[]] + $FeatureFlag ) $getTargetResourceParameters = @{ @@ -985,6 +1018,7 @@ function Set-TargetResource SourceCredential = $SourceCredential InstanceName = $InstanceName FailoverClusterNetworkName = $FailoverClusterNetworkName + FeatureFlag = $FeatureFlag } $getTargetResourceResult = Get-TargetResource @getTargetResourceParameters @@ -1082,6 +1116,10 @@ function Set-TargetResource { $featuresToInstall += "$feature," } + else + { + Write-Verbose -Message ($script:localizedData.FeatureAlreadyInstalled -f $feature) + } } $Features = $featuresToInstall.Trim(',') @@ -1365,31 +1403,31 @@ function Set-TargetResource 'SQLBackupDir' ) } - + # tempdb : define SqlTempdbFileCount if($PSBoundParameters.ContainsKey('SqlTempdbFileCount')) { $setupArguments += @{ SqlTempdbFileCount = $SqlTempdbFileCount } } - + # tempdb : define SqlTempdbFileSize if($PSBoundParameters.ContainsKey('SqlTempdbFileSize')) { $setupArguments += @{ SqlTempdbFileSize = $SqlTempdbFileSize } } - + # tempdb : define SqlTempdbFileGrowth if($PSBoundParameters.ContainsKey('SqlTempdbFileGrowth')) { $setupArguments += @{ SqlTempdbFileGrowth = $SqlTempdbFileGrowth } } - + # tempdb : define SqlTempdbLogFileSize if($PSBoundParameters.ContainsKey('SqlTempdbLogFileSize')) { $setupArguments += @{ SqlTempdbLogFileSize = $SqlTempdbLogFileSize } } - + # tempdb : define SqlTempdbLogFileGrowth if($PSBoundParameters.ContainsKey('SqlTempdbLogFileGrowth')) { @@ -1837,6 +1875,11 @@ function Set-TargetResource .PARAMETER SetupProcessTimeout The timeout, in seconds, to wait for the setup process to finish. Default value is 7200 seconds (2 hours). If the setup process does not finish before this time, and error will be thrown. + + .PARAMETER FeatureFlag + Feature flags are used to toggle functionality on or off. See the + documentation for what additional functionality exist through a feature + flag. #> function Test-TargetResource { @@ -2071,7 +2114,11 @@ function Test-TargetResource [Parameter()] [System.UInt32] - $SetupProcessTimeout = 7200 + $SetupProcessTimeout = 7200, + + [Parameter()] + [System.String[]] + $FeatureFlag ) $getTargetResourceParameters = @{ @@ -2080,6 +2127,7 @@ function Test-TargetResource SourceCredential = $SourceCredential InstanceName = $InstanceName FailoverClusterNetworkName = $FailoverClusterNetworkName + FeatureFlag = $FeatureFlag } $boundParameters = $PSBoundParameters @@ -2466,4 +2514,143 @@ function ConvertTo-StartupType return $StartMode } +<# + .SYNOPSIS + Returns an array of installed shared features. + + .PARAMETER SqlVersion + The major version of the SQL Server instance, i.e. 12, 13, or 14. +#> +function Get-InstalledSharedFeatures +{ + [CmdletBinding()] + [OutputType([System.String[]])] + param + ( + [Parameter(Mandatory = $true)] + [System.Int32] + $SqlVersion + ) + + $sharedFeatures = @() + + $configurationStateRegistryPath = "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($SqlVersion)0\ConfigurationState" + + # Check if Data Quality Client sub component is configured + Write-Verbose -Message ($script:localizedData.EvaluateDataQualityClientFeature -f $configurationStateRegistryPath) + + $isDQCInstalled = (Get-ItemProperty -Path $configurationStateRegistryPath -ErrorAction SilentlyContinue).SQL_DQ_CLIENT_Full + if ($isDQCInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.DataQualityClientFeatureFound + $sharedFeatures += 'DQC' + } + else + { + Write-Verbose -Message $script:localizedData.DataQualityClientFeatureNotFound + } + + # Check if Documentation Components "BOL" is configured + Write-Verbose -Message ($script:localizedData.EvaluateDocumentationComponentsFeature -f $configurationStateRegistryPath) + + $isBOLInstalled = (Get-ItemProperty -Path $configurationStateRegistryPath -ErrorAction SilentlyContinue).SQL_BOL_Components + if ($isBOLInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.DocumentationComponentsFeatureFound + $sharedFeatures += 'BOL' + } + else + { + Write-Verbose -Message $script:localizedData.DocumentationComponentsFeatureNotFound + } + + # Check if Client Tools Connectivity (and SQL Client Connectivity SDK) "CONN" is configured + Write-Verbose -Message ($script:localizedData.EvaluateDocumentationComponentsFeature -f $configurationStateRegistryPath) + + $isConnInstalled = (Get-ItemProperty -Path $configurationStateRegistryPath -ErrorAction SilentlyContinue).Connectivity_Full + if ($isConnInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.ClientConnectivityToolsFeatureFound + $sharedFeatures += 'CONN' + } + else + { + Write-Verbose -Message $script:localizedData.ClientConnectivityToolsFeatureNotFound + } + + # Check if Client Tools Backwards Compatibility "BC" is configured + Write-Verbose -Message ($script:localizedData.EvaluateDocumentationComponentsFeature -f $configurationStateRegistryPath) + + $isBcInstalled = (Get-ItemProperty -Path $configurationStateRegistryPath -ErrorAction SilentlyContinue).Tools_Legacy_Full + if ($isBcInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.ClientConnectivityBackwardsCompatibilityToolsFeatureFound + $sharedFeatures += 'BC' + } + else + { + Write-Verbose -Message $script:localizedData.ClientConnectivityBackwardsCompatibilityToolsFeatureNotFound + } + + # Check if Client Tools SDK "SDK" is configured + Write-Verbose -Message ($script:localizedData.EvaluateDocumentationComponentsFeature -f $configurationStateRegistryPath) + + $isSdkInstalled = (Get-ItemProperty -Path $configurationStateRegistryPath -ErrorAction SilentlyContinue).SDK_Full + if ($isSdkInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.ClientToolsSdkFeatureFound + $sharedFeatures += 'SDK' + } + else + { + Write-Verbose -Message $script:localizedData.ClientToolsSdkFeatureNotFound + } + + # Check if MDS sub component is configured for this server + Write-Verbose -Message ($script:localizedData.EvaluateMasterDataServicesFeature -f $configurationStateRegistryPath) + + $isMDSInstalled = (Get-ItemProperty -Path $configurationStateRegistryPath -ErrorAction SilentlyContinue).MDSCoreFeature + if ($isMDSInstalled -eq 1) + { + Write-Verbose -Message $script:localizedData.MasterDataServicesFeatureFound + $sharedFeatures += 'MDS' + } + else + { + Write-Verbose -Message $script:localizedData.MasterDataServicesFeatureNotFound + } + + return $sharedFeatures +} + +<# + .SYNOPSIS + Test if the specific feature flag should be enabled. + + .PARAMETER FeatureFlag + An array of feature flags that should be compared against. + + .PARAMETER TestFlag + The feature flag that is being check if it should be enabled. +#> +function Test-FeatureFlag +{ + [CmdletBinding()] + [OutputType([System.Boolean])] + param + ( + [Parameter()] + [System.String[]] + $FeatureFlag, + + [Parameter(Mandatory = $true)] + [System.String] + $TestFlag + ) + + $flagEnabled = $FeatureFlag -and ($FeatureFlag -and $FeatureFlag.Contains($TestFlag)) + + return $flagEnabled +} + Export-ModuleMember -Function *-TargetResource diff --git a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.schema.mof b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.schema.mof index 4e2d6f386..28548ec56 100644 --- a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.schema.mof +++ b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.schema.mof @@ -62,4 +62,5 @@ class MSFT_SqlSetup : OMI_BaseResource [Write, Description("Specifies the initial size of each tempdb log file in MB.")] UInt32 SqlTempdbLogFileSize; [Write, Description("Specifies the file growth increment of each tempdb data file in MB.")] UInt32 SqlTempdbLogFileGrowth; [Write, Description("The timeout, in seconds, to wait for the setup process to finish. Default value is 7200 seconds (2 hours). If the setup process does not finish before this time, and error will be thrown.")] UInt32 SetupProcessTimeout; + [Write, Description("Feature flags are used to toggle functionality on or off. See the documentation for what additional functionality exist through a feature flag.")] String FeatureFlag[]; }; diff --git a/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 b/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 index 9530fcfbf..ca1cd0878 100644 --- a/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 +++ b/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 @@ -72,6 +72,8 @@ ConvertFrom-StringData @' RobocopyAllFilesPresent = Robocopy reported that all files already present. StartSetupProcess = Started the process with id {0} using the path '{1}', and with a timeout value of {2} seconds. EvaluateMasterDataServicesFeature = Detecting Master Data Services (MDS) feature ({0}). - MasterDataServicesFeatureFound = 'Master Data Services (MDS) feature detected.' - MasterDataServicesFeatureNotFound = 'Master Data Services (MDS) feature not detected.' + MasterDataServicesFeatureFound = Master Data Services (MDS) feature detected. + MasterDataServicesFeatureNotFound = Master Data Services (MDS) feature not detected. + FeatureAlreadyInstalled = The feature '{0}' is already installed so it will not be installed again. + FeatureFlag = Using feature flag '{0}' '@ diff --git a/Examples/Resources/SqlServerDatabaseMail/1-EnableDatabaseMail.ps1 b/Examples/Resources/SqlServerDatabaseMail/1-EnableDatabaseMail.ps1 index d992d5c45..d5fa462d4 100644 --- a/Examples/Resources/SqlServerDatabaseMail/1-EnableDatabaseMail.ps1 +++ b/Examples/Resources/SqlServerDatabaseMail/1-EnableDatabaseMail.ps1 @@ -9,7 +9,7 @@ $ConfigurationData = @{ @{ NodeName = 'localhost' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' MailServerName = 'mail.company.local' AccountName = 'MyMail' diff --git a/Examples/Resources/SqlServerDatabaseMail/2-DisableDatabaseMail.ps1 b/Examples/Resources/SqlServerDatabaseMail/2-DisableDatabaseMail.ps1 index c90da0f66..ab58510a1 100644 --- a/Examples/Resources/SqlServerDatabaseMail/2-DisableDatabaseMail.ps1 +++ b/Examples/Resources/SqlServerDatabaseMail/2-DisableDatabaseMail.ps1 @@ -9,7 +9,7 @@ $ConfigurationData = @{ @{ NodeName = 'localhost' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' MailServerName = 'mail.company.local' AccountName = 'MyMail' diff --git a/Examples/Resources/SqlSetup/3-InstallNamedInstanceSingleServerFromUncPathUsingSourceCredential.ps1 b/Examples/Resources/SqlSetup/3-InstallNamedInstanceSingleServerFromUncPathUsingSourceCredential.ps1 index 2d640738d..cfb991b60 100644 --- a/Examples/Resources/SqlSetup/3-InstallNamedInstanceSingleServerFromUncPathUsingSourceCredential.ps1 +++ b/Examples/Resources/SqlSetup/3-InstallNamedInstanceSingleServerFromUncPathUsingSourceCredential.ps1 @@ -79,7 +79,7 @@ Configuration Example ASLogDir = 'C:\MSOLAP13.INST2016\Log' ASBackupDir = 'C:\MSOLAP13.INST2016\Backup' ASTempDir = 'C:\MSOLAP13.INST2016\Temp' - SourcePath = '\\fileserver.compant.local\images$\SQL2016RTM' + SourcePath = '\\fileserver.company.local\images$\SQL2016RTM' SourceCredential = $SqlInstallCredential UpdateEnabled = 'False' ForceReboot = $false diff --git a/Examples/Resources/SqlSetup/4-InstallNamedInstanceInFailoverClusterFirstNode.ps1 b/Examples/Resources/SqlSetup/4-InstallNamedInstanceInFailoverClusterFirstNode.ps1 index 598ce1f9b..089acb95b 100644 --- a/Examples/Resources/SqlSetup/4-InstallNamedInstanceInFailoverClusterFirstNode.ps1 +++ b/Examples/Resources/SqlSetup/4-InstallNamedInstanceInFailoverClusterFirstNode.ps1 @@ -80,7 +80,7 @@ Configuration Example Action = 'InstallFailoverCluster' ForceReboot = $false UpdateEnabled = 'False' - SourcePath = '\\fileserver.compant.local\images$\SQL2016RTM' + SourcePath = '\\fileserver.company.local\images$\SQL2016RTM' SourceCredential = $SqlInstallCredential InstanceName = 'INST2016' diff --git a/Examples/Resources/SqlSetup/5-InstallNamedInstanceInFailoverClusterSecondNode.ps1 b/Examples/Resources/SqlSetup/5-InstallNamedInstanceInFailoverClusterSecondNode.ps1 index 2929986a9..e114b7bd4 100644 --- a/Examples/Resources/SqlSetup/5-InstallNamedInstanceInFailoverClusterSecondNode.ps1 +++ b/Examples/Resources/SqlSetup/5-InstallNamedInstanceInFailoverClusterSecondNode.ps1 @@ -76,7 +76,7 @@ Configuration Example Action = 'AddNode' ForceReboot = $false UpdateEnabled = 'False' - SourcePath = '\\fileserver.compant.local\images$\SQL2016RTM' + SourcePath = '\\fileserver.company.local\images$\SQL2016RTM' SourceCredential = $SqlInstallCredential InstanceName = 'INST2016' diff --git a/README.md b/README.md index 7b2254141..e5b5bfd79 100644 --- a/README.md +++ b/README.md @@ -1619,6 +1619,22 @@ cluster. This is a limitation of SQL Server. See article [You cannot add or remove features to a SQL Server 2008, SQL Server 2008 R2, or SQL Server 2012 failover cluster](https://support.microsoft.com/en-us/help/2547273/you-cannot-add-or-remove-features-to-a-sql-server-2008,-sql-server-2008-r2,-or-sql-server-2012-failover-cluster). +#### Feature flags + +Feature flags are used to toggle functionality on or off. One or more +feature flags can be added to the parameter `FeatureFlag`, i.e. +`FeatureFlag = @('DetectionSharedFeatures')`. + +>**NOTE:** The functionality, exposed +with a feature flag, can be changed from one release to another, including +having breaking changes. + + +Feature flag | Description +--- | --- +DetectionSharedFeatures | A new way of detecting if the shared features is installed or not. This was implemented because the previous implementation did not work fully with SQL Server 2017. + + #### Credentials for running the resource ##### PsDscRunAsCredential @@ -1758,6 +1774,9 @@ need a '*SVCPASSWORD' argument in the setup arguments. * **`[UInt32]` SetupProcessTimeout** _(Write)_: The timeout, in seconds, to wait for the setup process to finish. Default value is 7200 seconds (2 hours). If the setup process does not finish before this time, and error will be thrown. +* **`[String[]]` FeatureFlag** _(Write)_: Feature flags are used to toggle + functionality on or off. See the documentation for what additional + functionality exist through a feature flag. #### Read-Only Properties from Get-TargetResource diff --git a/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 index f30b37ddf..a742fdee2 100644 --- a/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlAgentOperator.config.ps1 b/Tests/Integration/MSFT_SqlAgentOperator.config.ps1 index 2529d51c1..369111386 100644 --- a/Tests/Integration/MSFT_SqlAgentOperator.config.ps1 +++ b/Tests/Integration/MSFT_SqlAgentOperator.config.ps1 @@ -22,7 +22,7 @@ else Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' Name = 'MyOperator' EmailAddress = 'MyEmail@company.local' diff --git a/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 index 7db5b9f52..c100fffbe 100644 --- a/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlAlwaysOnService.config.ps1 b/Tests/Integration/MSFT_SqlAlwaysOnService.config.ps1 index 44c16a0ae..1690ee7aa 100644 --- a/Tests/Integration/MSFT_SqlAlwaysOnService.config.ps1 +++ b/Tests/Integration/MSFT_SqlAlwaysOnService.config.ps1 @@ -40,7 +40,7 @@ else @{ NodeName = 'localhost' ComputerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' RestartTimeout = 120 UserName = "$env:COMPUTERNAME\SqlInstall" diff --git a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 index 99b79fac8..a49b01f75 100644 --- a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 index 44a4d5742..92c84c78e 100644 --- a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 +++ b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 @@ -22,7 +22,7 @@ else Password = 'P@ssw0rd1' ComputerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' RestartTimeout = 120 DataFilePath = 'C:\SQLData\' diff --git a/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 index a357142aa..c072f9b99 100644 --- a/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category 'Integration_SQL2016') { return } diff --git a/Tests/Integration/MSFT_SqlRS.config.ps1 b/Tests/Integration/MSFT_SqlRS.config.ps1 index 5af8d7281..702715389 100644 --- a/Tests/Integration/MSFT_SqlRS.config.ps1 +++ b/Tests/Integration/MSFT_SqlRS.config.ps1 @@ -39,7 +39,7 @@ else DriveLetter = $mockIsoMediaDriveLetter DatabaseServerName = $env:COMPUTERNAME - DatabaseInstanceName = 'DSCSQL2016' + DatabaseInstanceName = 'DSCSQLTEST' CertificateFile = $env:DscPublicCertificatePath } diff --git a/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 index 3a5d6a879..ddaec72df 100644 --- a/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 @@ -9,7 +9,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlScript.config.ps1 b/Tests/Integration/MSFT_SqlScript.config.ps1 index 4954b8400..2f0bbb8de 100644 --- a/Tests/Integration/MSFT_SqlScript.config.ps1 +++ b/Tests/Integration/MSFT_SqlScript.config.ps1 @@ -24,7 +24,7 @@ else SqlLogin_Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' Database1Name = 'ScriptDatabase1' Database2Name = 'ScriptDatabase2' diff --git a/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 index 8047bb08f..8c4376419 100644 --- a/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 @@ -9,7 +9,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlScriptQuery.config.ps1 b/Tests/Integration/MSFT_SqlScriptQuery.config.ps1 index aa018fe3b..8c29ce94f 100644 --- a/Tests/Integration/MSFT_SqlScriptQuery.config.ps1 +++ b/Tests/Integration/MSFT_SqlScriptQuery.config.ps1 @@ -24,7 +24,7 @@ else SqlLogin_Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' Database1Name = 'ScriptDatabase3' Database2Name = 'ScriptDatabase4' diff --git a/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 index 60c9c5f9c..a10bfafc0 100644 --- a/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServerDatabaseMail.config.ps1 b/Tests/Integration/MSFT_SqlServerDatabaseMail.config.ps1 index 4a5e1161d..4083b1370 100644 --- a/Tests/Integration/MSFT_SqlServerDatabaseMail.config.ps1 +++ b/Tests/Integration/MSFT_SqlServerDatabaseMail.config.ps1 @@ -22,7 +22,7 @@ else Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' MailServerName = 'mail.company.local' AccountName = 'MyMail' diff --git a/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 index 09b7f2f2c..c15753605 100644 --- a/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServerEndPoint.config.ps1 b/Tests/Integration/MSFT_SqlServerEndPoint.config.ps1 index 9a3a715b2..bbcbe6cf0 100644 --- a/Tests/Integration/MSFT_SqlServerEndPoint.config.ps1 +++ b/Tests/Integration/MSFT_SqlServerEndPoint.config.ps1 @@ -18,7 +18,7 @@ else @{ NodeName = 'localhost' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' EndpointName = 'HADR' Port = 5023 diff --git a/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 index 832fc0dd0..d423666b6 100644 --- a/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServerLogin.config.ps1 b/Tests/Integration/MSFT_SqlServerLogin.config.ps1 index 1f97e9145..0c29798d5 100644 --- a/Tests/Integration/MSFT_SqlServerLogin.config.ps1 +++ b/Tests/Integration/MSFT_SqlServerLogin.config.ps1 @@ -22,7 +22,7 @@ else Admin_Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' DscUser1Name = ('{0}\{1}' -f $env:COMPUTERNAME, 'DscUser1') DscUser1Type = 'WindowsUser' diff --git a/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 index 20174790d..54da4494c 100644 --- a/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServerNetwork.config.ps1 b/Tests/Integration/MSFT_SqlServerNetwork.config.ps1 index f96852cdc..54fa9441e 100644 --- a/Tests/Integration/MSFT_SqlServerNetwork.config.ps1 +++ b/Tests/Integration/MSFT_SqlServerNetwork.config.ps1 @@ -22,7 +22,7 @@ else Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' ProtocolName = 'Tcp' Enabled = $true diff --git a/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 index 3b6fd0df0..a94d83f05 100644 --- a/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServerRole.config.ps1 b/Tests/Integration/MSFT_SqlServerRole.config.ps1 index f968e4e8c..113a06d7a 100644 --- a/Tests/Integration/MSFT_SqlServerRole.config.ps1 +++ b/Tests/Integration/MSFT_SqlServerRole.config.ps1 @@ -22,7 +22,7 @@ else Password = 'P@ssw0rd1' ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' Role1Name = 'DscServerRole1' Role2Name = 'DscServerRole2' diff --git a/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 index 039c09fe2..760d3369e 100644 --- a/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 @@ -9,7 +9,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServerSecureConnection.config.ps1 b/Tests/Integration/MSFT_SqlServerSecureConnection.config.ps1 index 364408296..110623d20 100644 --- a/Tests/Integration/MSFT_SqlServerSecureConnection.config.ps1 +++ b/Tests/Integration/MSFT_SqlServerSecureConnection.config.ps1 @@ -21,7 +21,7 @@ else ServiceAccount = "$env:COMPUTERNAME\svc-SqlPrimary" ServerName = $env:COMPUTERNAME - InstanceName = 'DSCSQL2016' + InstanceName = 'DSCSQLTEST' Thumbprint = $env:SqlCertificateThumbprint diff --git a/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 index 4dfdac1df..c3157e203 100644 --- a/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } diff --git a/Tests/Integration/MSFT_SqlServiceAccount.config.ps1 b/Tests/Integration/MSFT_SqlServiceAccount.config.ps1 index 66efad798..b02326b50 100644 --- a/Tests/Integration/MSFT_SqlServiceAccount.config.ps1 +++ b/Tests/Integration/MSFT_SqlServiceAccount.config.ps1 @@ -29,7 +29,7 @@ else ServerName = $env:COMPUTERNAME DefaultInstanceName = 'MSSQLSERVER' - NamedInstanceName = 'DSCSQL2016' + NamedInstanceName = 'DSCSQLTEST' ServiceTypeDatabaseEngine = 'DatabaseEngine' ServiceTypeSqlServerAgent = 'SqlServerAgent' diff --git a/Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 new file mode 100644 index 000000000..ef5da124c --- /dev/null +++ b/Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 @@ -0,0 +1,603 @@ +# This is used to make sure the integration test run in the correct order. +[Microsoft.DscResourceKit.IntegrationTest(OrderNumber = 1)] +param() + +Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') + +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2017')) +{ + return +} + +$script:dscModuleName = 'SqlServerDsc' +$script:dscResourceFriendlyName = 'SqlSetup' +$script:dscResourceName = "MSFT_$($script:dscResourceFriendlyName)" + +#region HEADER +# Integration Test Template Version: 1.3.2 +[String] $script:moduleRoot = Split-Path -Parent (Split-Path -Parent $PSScriptRoot) +if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests'))) -or ` + (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1'))) ) +{ + & git @('clone', 'https://github.com/PowerShell/DscResource.Tests.git', (Join-Path -Path $script:moduleRoot -ChildPath 'DscResource.Tests')) +} + +Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'DSCResource.Tests' -ChildPath 'TestHelper.psm1')) -Force +$TestEnvironment = Initialize-TestEnvironment ` + -DSCModuleName $script:dscModuleName ` + -DSCResourceName $script:dscResourceName ` + -TestType Integration +#endregion + +<# + .SYNOPSIS + This function will output the Setup Bootstrap Summary.txt log file. + + .DESCRIPTION + This function will output the Summary.txt log file, this is to be + able to debug any problems that potentially occurred during setup. + This will pick up the newest Summary.txt log file, so any + other log files will be ignored (AppVeyor build worker has + SQL Server instances installed by default). + This code is meant to work regardless what SQL Server + major version is used for the integration test. +#> +function Show-SqlBootstrapLog +{ + [CmdletBinding()] + param + ( + ) + + $summaryLogPath = Get-ChildItem -Path 'C:\Program Files\Microsoft SQL Server\**\Setup Bootstrap\Log\Summary.txt' | + Sort-Object -Property LastWriteTime -Descending | + Select-Object -First 1 + + $summaryLog = Get-Content $summaryLogPath + + Write-Verbose -Message $('-' * 80) -Verbose + Write-Verbose -Message 'Summary.txt' -Verbose + Write-Verbose -Message $('-' * 80) -Verbose + $summaryLog | ForEach-Object { + Write-Verbose $_ -Verbose + } + Write-Verbose -Message $('-' * 80) -Verbose +} + +<# + Workaround for issue #774. In the appveyor.yml file the folder + C:\Program Files (x86)\Microsoft SQL Server\**\Tools\PowerShell\Modules + was renamed to + C:\Program Files (x86)\Microsoft SQL Server\**\Tools\PowerShell\Modules.old + here we rename back the folder to the correct name. Only the version need + for our tests are renamed. +#> +$sqlModulePath = Get-ChildItem -Path 'C:\Program Files (x86)\Microsoft SQL Server\140\Tools\PowerShell\*.old' +$sqlModulePath | ForEach-Object -Process { + $newFolderName = (Split-Path -Path $_ -Leaf) -replace '\.old' + Write-Verbose ('Renaming ''{0}'' to ''..\{1}''' -f $_, $newFolderName) -Verbose + Rename-Item $_ -NewName $newFolderName -Force +} + +# Using try/finally to always cleanup. +try +{ + $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName)_SQL2017.config.ps1" + . $configFile + + $mockSourceMediaUrl = 'https://download.microsoft.com/download/E/F/2/EF23C21D-7860-4F05-88CE-39AA114B014B/SQLServer2017-x64-ENU.iso' + + # Download SQL Server media + if (-not (Test-Path -Path $ConfigurationData.AllNodes.ImagePath)) + { + # By switching to 'SilentlyContinue' should theoretically increase the download speed. + $previousProgressPreference = $ProgressPreference + $ProgressPreference = 'SilentlyContinue' + + Write-Verbose -Message "Start downloading the SQL Server media at $(Get-Date -Format 'yyyy-MM-dd hh:mm:ss')" -Verbose + + Invoke-WebRequest -Uri $mockSourceMediaUrl -OutFile $ConfigurationData.AllNodes.ImagePath + + Write-Verbose -Message ('SQL Server media file has SHA1 hash ''{0}''' -f (Get-FileHash -Path $ConfigurationData.AllNodes.ImagePath -Algorithm 'SHA1').Hash) -Verbose + + $ProgressPreference = $previousProgressPreference + + # Double check that the SQL media was downloaded. + if (-not (Test-Path -Path $ConfigurationData.AllNodes.ImagePath)) + { + Write-Warning -Message ('SQL media could not be downloaded, can not run the integration test.') + return + } + else + { + Write-Verbose -Message "Finished downloading the SQL Server media iso at $(Get-Date -Format 'yyyy-MM-dd hh:mm:ss')" -Verbose + } + } + else + { + Write-Verbose -Message 'SQL Server media is already downloaded' -Verbose + } + + Describe "$($script:dscResourceName)_Integration" { + BeforeAll { + $resourceId = "[$($script:dscResourceFriendlyName)]Integration_Test" + } + + $configurationName = "$($script:dscResourceName)_CreateDependencies_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + } + + $configurationName = "$($script:dscResourceName)_InstallDatabaseEngineNamedInstanceAsSystem_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } -ErrorVariable itBlockError + + # Check if previous It-block failed. If so output the SQL Server setup log file. + if ( $itBlockError.Count -ne 0 ) + { + Show-SqlBootstrapLog + } + + It 'Should be able to call Get-DscConfiguration without throwing' { + { + $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop + } | Should -Not -Throw + } + + It 'Should have set the resource and all the parameters should match' { + $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { + $_.ConfigurationName -eq $configurationName ` + -and $_.ResourceId -eq $resourceId + } + + $resourceCurrentState.Action | Should -BeNullOrEmpty + $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.AgtSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName -Leaf)) + $resourceCurrentState.AgtSvcStartupType | Should -Be 'Automatic' + $resourceCurrentState.ASServerMode | Should -Be $ConfigurationData.AllNodes.AnalysisServicesMultiServerMode + $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Backup") + $resourceCurrentState.ASCollation | Should -Be $ConfigurationData.AllNodes.Collation + $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Config") + $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Data") + $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Log") + $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Temp") + $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.ASSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) + $resourceCurrentState.AsSvcStartupType | Should -Be 'Automatic' + $resourceCurrentState.ASSysAdminAccounts | Should -Be @( + $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, + "NT SERVICE\SSASTELEMETRY`$$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)" + ) + $resourceCurrentState.BrowserSvcStartupType | Should -BeNullOrEmpty + $resourceCurrentState.ErrorReporting | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterGroupName | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterIPAddress | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterNetworkName | Should -BeNullOrEmpty + $resourceCurrentState.Features | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceFeatures + $resourceCurrentState.ForceReboot | Should -BeNullOrEmpty + $resourceCurrentState.FTSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir + $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir + $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL") + $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir + $resourceCurrentState.InstanceID | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName + $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName + $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.ISSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.ProductKey | Should -BeNullOrEmpty + $resourceCurrentState.RSSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.RSSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.SAPwd | Should -BeNullOrEmpty + $resourceCurrentState.SecurityMode | Should -Be 'SQL' + $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty + $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty + $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" + $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\Backup") + $resourceCurrentState.SQLCollation | Should -Be $ConfigurationData.AllNodes.Collation + $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.SQLSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) + $resourceCurrentState.SqlSvcStartupType | Should -Be 'Automatic' + $resourceCurrentState.SQLSysAdminAccounts | Should -Be @( + $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, + $ConfigurationData.AllNodes.SqlInstallAccountUserName, + "NT SERVICE\MSSQL`$$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)", + "NT SERVICE\SQLAgent`$$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)", + 'NT SERVICE\SQLWriter', + 'NT SERVICE\Winmgmt', + 'sa' + ) + $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty + $resourceCurrentState.SqlTempdbFileCount | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileCount + $resourceCurrentState.SqlTempdbFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileSize + $resourceCurrentState.SqlTempdbFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileGrowth + $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty + $resourceCurrentState.SqlTempdbLogFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempdbLogFileSize + $resourceCurrentState.SqlTempdbLogFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempdbLogFileGrowth + $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\DATA\") + $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\DATA\") + $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty + $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty + $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty + $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty + } + + It 'Should return $true when Test-DscConfiguration is run' { + Test-DscConfiguration -Verbose | Should -Be $true + } + } + + $configurationName = "$($script:dscResourceName)_StopServicesInstance_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + } + + $configurationName = "$($script:dscResourceName)_InstallDatabaseEngineDefaultInstanceAsUser_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } -ErrorVariable itBlockError + + # Check if previous It-block failed. If so output the SQL Server setup log file. + if ( $itBlockError.Count -ne 0 ) + { + Show-SqlBootstrapLog + } + + It 'Should be able to call Get-DscConfiguration without throwing' { + { + $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop + } | Should -Not -Throw + } + + It 'Should have set the resource and all the parameters should match' { + $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { + $_.ConfigurationName -eq $configurationName ` + -and $_.ResourceId -eq $resourceId + } + + $resourceCurrentState.Action | Should -BeNullOrEmpty + $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.AgtSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName -Leaf)) + $resourceCurrentState.ASServerMode | Should -BeNullOrEmpty + $resourceCurrentState.ASBackupDir | Should -BeNullOrEmpty + $resourceCurrentState.ASCollation | Should -BeNullOrEmpty + $resourceCurrentState.ASConfigDir | Should -BeNullOrEmpty + $resourceCurrentState.ASDataDir | Should -BeNullOrEmpty + $resourceCurrentState.ASLogDir | Should -BeNullOrEmpty + $resourceCurrentState.ASTempDir | Should -BeNullOrEmpty + $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.ASSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.ASSysAdminAccounts | Should -BeNullOrEmpty + $resourceCurrentState.BrowserSvcStartupType | Should -BeNullOrEmpty + $resourceCurrentState.ErrorReporting | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterGroupName | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterIPAddress | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterNetworkName | Should -BeNullOrEmpty + $resourceCurrentState.Features | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceFeatures + $resourceCurrentState.ForceReboot | Should -BeNullOrEmpty + $resourceCurrentState.FTSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir + $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir + $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL") + $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir + $resourceCurrentState.InstanceID | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName + $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName + $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.ISSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.ProductKey | Should -BeNullOrEmpty + $resourceCurrentState.RSSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.RSSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.SAPwd | Should -BeNullOrEmpty + $resourceCurrentState.SecurityMode | Should -Be 'Windows' + $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty + $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty + $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" + $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\Backup") + $resourceCurrentState.SQLCollation | Should -Be $ConfigurationData.AllNodes.Collation + $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.SQLSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) + $resourceCurrentState.SQLSysAdminAccounts | Should -Be @( + $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, + $ConfigurationData.AllNodes.SqlInstallAccountUserName, + "NT SERVICE\$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)", + "NT SERVICE\SQLSERVERAGENT", + 'NT SERVICE\SQLWriter', + 'NT SERVICE\Winmgmt', + 'sa' + ) + $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty + $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty + $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") + $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") + $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty + $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty + $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty + $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty + } + + It 'Should return $true when Test-DscConfiguration is run' { + Test-DscConfiguration -Verbose | Should -Be $true + } + } + + $configurationName = "$($script:dscResourceName)_StopSqlServerDefaultInstance_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + } + + $configurationName = "$($script:dscResourceName)_InstallTabularAnalysisServicesAsSystem_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } -ErrorVariable itBlockError + + # Check if previous It-block failed. If so output the SQL Server setup log file. + if ( $itBlockError.Count -ne 0 ) + { + Show-SqlBootstrapLog + } + + It 'Should be able to call Get-DscConfiguration without throwing' { + { + $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop + } | Should -Not -Throw + } + + It 'Should have set the resource and all the parameters should match' { + $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { + $_.ConfigurationName -eq $configurationName ` + -and $_.ResourceId -eq $resourceId + } + + $resourceCurrentState.Action | Should -BeNullOrEmpty + $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.AgtSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.ASServerMode | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularServerMode + $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Backup") + $resourceCurrentState.ASCollation | Should -Be $ConfigurationData.AllNodes.Collation + $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Config") + $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Data") + $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Log") + $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Temp") + $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.ASSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) + $resourceCurrentState.ASSysAdminAccounts | Should -Be @( + $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, + "NT SERVICE\SSASTELEMETRY`$$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)" + ) + $resourceCurrentState.BrowserSvcStartupType | Should -BeNullOrEmpty + $resourceCurrentState.ErrorReporting | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterGroupName | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterIPAddress | Should -BeNullOrEmpty + $resourceCurrentState.FailoverClusterNetworkName | Should -BeNullOrEmpty + $resourceCurrentState.Features | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularFeatures + $resourceCurrentState.ForceReboot | Should -BeNullOrEmpty + $resourceCurrentState.FTSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir + $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir + $resourceCurrentState.InstallSQLDataDir | Should -BeNullOrEmpty + $resourceCurrentState.InstanceDir | Should -BeNullOrEmpty + $resourceCurrentState.InstanceID | Should -BeNullOrEmpty + $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName + $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.ISSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.ProductKey | Should -BeNullOrEmpty + $resourceCurrentState.RSSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.RSSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.SAPwd | Should -BeNullOrEmpty + $resourceCurrentState.SecurityMode | Should -BeNullOrEmpty + $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty + $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty + $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" + $resourceCurrentState.SQLBackupDir | Should -BeNullOrEmpty + $resourceCurrentState.SQLCollation | Should -BeNullOrEmpty + $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty + $resourceCurrentState.SQLSvcAccountUsername | Should -BeNullOrEmpty + $resourceCurrentState.SQLSysAdminAccounts | Should -BeNullOrEmpty + $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty + $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty + $resourceCurrentState.SQLUserDBDir | Should -BeNullOrEmpty + $resourceCurrentState.SQLUserDBLogDir | Should -BeNullOrEmpty + $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty + $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty + $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty + $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty + } + + It 'Should return $true when Test-DscConfiguration is run' { + Test-DscConfiguration -Verbose | Should -Be $true + } + } + + $configurationName = "$($script:dscResourceName)_StopTabularAnalysisServices_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + } + + $configurationName = "$($script:dscResourceName)_StartServicesInstance_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + } + } +} +finally +{ + #region FOOTER + Restore-TestEnvironment -TestEnvironment $TestEnvironment + #endregion +} diff --git a/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 similarity index 99% rename from Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 rename to Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 index b4c88c344..f0cedc873 100644 --- a/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration') +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016')) { return } @@ -82,7 +82,7 @@ $sqlModulePath | ForEach-Object -Process { # Using try/finally to always cleanup. try { - $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName).config.ps1" + $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName)_SQL2016.config.ps1" . $configFile $mockSourceMediaUrl = 'https://download.microsoft.com/download/9/0/7/907AD35F-9F9C-43A5-9789-52470555DB90/ENU/SQLServer2016SP1-FullSlipstream-x64-ENU.iso' diff --git a/Tests/Integration/MSFT_SqlSetup.config.ps1 b/Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 similarity index 99% rename from Tests/Integration/MSFT_SqlSetup.config.ps1 rename to Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 index afd319576..4eaf3560e 100644 --- a/Tests/Integration/MSFT_SqlSetup.config.ps1 +++ b/Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 @@ -23,13 +23,13 @@ else NodeName = 'localhost' # Database Engine properties. - DatabaseEngineNamedInstanceName = 'DSCSQL2016' + DatabaseEngineNamedInstanceName = 'DSCSQLTEST' DatabaseEngineNamedInstanceFeatures = 'SQLENGINE,AS,CONN,BC,SDK' AnalysisServicesMultiServerMode = 'MULTIDIMENSIONAL' <# Analysis Services Tabular properties. - The features CONN,BC,SDK is installed with the DSCSQL2016 so those + The features CONN,BC,SDK is installed with the DSCSQLTEST so those features will found for DSCTABULAR instance as well. The features is added here so the same property can be used to evaluate the result in the test. @@ -40,7 +40,7 @@ else <# Database Engine default instance properties. - The features CONN,BC,SDK is installed with the DSCSQL2016 so those + The features CONN,BC,SDK is installed with the DSCSQLTEST so those features will found for DSCTABULAR instance as well. The features is added here so the same property can be used to evaluate the result in the test. diff --git a/Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 b/Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 new file mode 100644 index 000000000..00c00ee5c --- /dev/null +++ b/Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 @@ -0,0 +1,448 @@ +#region HEADER +# Integration Test Config Template Version: 1.2.0 +#endregion + +$configFile = [System.IO.Path]::ChangeExtension($MyInvocation.MyCommand.Path, 'json') +if (Test-Path -Path $configFile) +{ + <# + Allows reading the configuration data from a JSON file, + for real testing scenarios outside of the CI. + #> + $ConfigurationData = Get-Content -Path $configFile | ConvertFrom-Json +} +else +{ + # Get a spare drive letter + $mockLastDrive = ((Get-Volume).DriveLetter | Sort-Object | Select-Object -Last 1) + $mockIsoMediaDriveLetter = [char](([int][char]$mockLastDrive) + 1) + + $ConfigurationData = @{ + AllNodes = @( + @{ + NodeName = 'localhost' + + # Database Engine properties. + DatabaseEngineNamedInstanceName = 'DSCSQLTEST' + DatabaseEngineNamedInstanceFeatures = 'SQLENGINE,AS,CONN,BC,SDK' + AnalysisServicesMultiServerMode = 'MULTIDIMENSIONAL' + + <# + Analysis Services Tabular properties. + The features CONN,BC,SDK is installed with the DSCSQLTEST so those + features will found for DSCTABULAR instance as well. + The features is added here so the same property can be used to + evaluate the result in the test. + #> + AnalysisServicesTabularInstanceName = 'DSCTABULAR' + AnalysisServicesTabularFeatures = 'AS,CONN,BC,SDK' + AnalysisServicesTabularServerMode = 'TABULAR' + + <# + Database Engine default instance properties. + The features CONN,BC,SDK is installed with the DSCSQLTEST so those + features will found for DSCTABULAR instance as well. + The features is added here so the same property can be used to + evaluate the result in the test. + #> + DatabaseEngineDefaultInstanceName = 'MSSQLSERVER' + DatabaseEngineDefaultInstanceFeatures = 'SQLENGINE,CONN,BC,SDK' + + # General SqlSetup properties + Collation = 'Finnish_Swedish_CI_AS' + InstallSharedDir = 'C:\Program Files\Microsoft SQL Server' + InstallSharedWOWDir = 'C:\Program Files (x86)\Microsoft SQL Server' + UpdateEnabled = 'False' + SuppressReboot = $true # Make sure we don't reboot during testing. + ForceReboot = $false + + # Properties for mounting media + ImagePath = "$env:TEMP\SQL2017.iso" + DriveLetter = $mockIsoMediaDriveLetter + + # Parameters to configure Tempdb + SqlTempdbFileCount = '2' + SqlTempdbFileSize = '128' + SqlTempdbFileGrowth = '128' + SqlTempdbLogFileSize = '128' + SqlTempdbLogFileGrowth = '128' + + SqlInstallAccountUserName = "$env:COMPUTERNAME\SqlInstall" + SqlInstallAccountPassword = 'P@ssw0rd1' + SqlAdministratorAccountUserName = "$env:COMPUTERNAME\SqlAdmin" + SqlAdministratorAccountPassword = 'P@ssw0rd1' + SqlServicePrimaryAccountUserName = "$env:COMPUTERNAME\svc-SqlPrimary" + SqlServicePrimaryAccountPassword = 'yig-C^Equ3' + SqlAgentServicePrimaryAccountUserName = "$env:COMPUTERNAME\svc-SqlAgentPri" + SqlAgentServicePrimaryAccountPassword = 'yig-C^Equ3' + SqlServiceSecondaryAccountUserName = "$env:COMPUTERNAME\svc-SqlSecondary" + SqlServiceSecondaryAccountPassword = 'yig-C^Equ3' + SqlAgentServiceSecondaryAccountUserName = "$env:COMPUTERNAME\svc-SqlAgentSec" + SqlAgentServiceSecondaryAccountPassword = 'yig-C^Equ3' + + CertificateFile = $env:DscPublicCertificatePath + } + ) + } +} + +<# + Creating all the credential objects to save some repeating code. +#> + +$SqlInstallCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @($ConfigurationData.AllNodes.SqlInstallAccountUserName, + (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlInstallAccountPassword -AsPlainText -Force)) + +$SqlAdministratorCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @($ConfigurationData.AllNodes.SqlAdministratorAccountUserName, + (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlAdministratorAccountPassword -AsPlainText -Force)) + +$SqlServicePrimaryCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @($ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName, + (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlServicePrimaryAccountPassword -AsPlainText -Force)) + +$SqlAgentServicePrimaryCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @($ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName, + (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountPassword -AsPlainText -Force)) + +$SqlServiceSecondaryCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @( + $ConfigurationData.AllNodes.SqlServiceSecondaryAccountUserName, + (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlServiceSecondaryAccountPassword -AsPlainText -Force)) + +$SqlAgentServiceSecondaryCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @($ConfigurationData.AllNodes.SqlAgentServiceSecondaryAccountUserName, + (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlAgentServiceSecondaryAccountPassword -AsPlainText -Force)) + +<# + .SYNOPSIS + Setting up the dependencies to test installing SQL Server instances. +#> +Configuration MSFT_SqlSetup_CreateDependencies_Config +{ + Import-DscResource -ModuleName 'PSDscResources' + Import-DscResource -ModuleName 'StorageDsc' + + node $AllNodes.NodeName + { + MountImage 'MountIsoMedia' + { + ImagePath = $Node.ImagePath + DriveLetter = $Node.DriveLetter + Ensure = 'Present' + } + + WaitForVolume WaitForMountOfIsoMedia + { + DriveLetter = $Node.DriveLetter + RetryIntervalSec = 5 + RetryCount = 10 + } + + User 'CreateSqlServicePrimaryAccount' + { + Ensure = 'Present' + UserName = Split-Path -Path $SqlServicePrimaryCredential.UserName -Leaf + Password = $SqlServicePrimaryCredential + } + + User 'CreateSqlAgentServicePrimaryAccount' + { + Ensure = 'Present' + UserName = Split-Path -Path $SqlAgentServicePrimaryCredential.UserName -Leaf + Password = $SqlAgentServicePrimaryCredential + } + + User 'CreateSqlServiceSecondaryAccount' + { + Ensure = 'Present' + UserName = Split-Path -Path $SqlServiceSecondaryCredential.UserName -Leaf + Password = $SqlServicePrimaryCredential + } + + User 'CreateSqlAgentServiceSecondaryAccount' + { + Ensure = 'Present' + UserName = Split-Path -Path $SqlAgentServiceSecondaryCredential.UserName -Leaf + Password = $SqlAgentServicePrimaryCredential + } + + User 'CreateSqlInstallAccount' + { + Ensure = 'Present' + UserName = Split-Path -Path $SqlInstallCredential.UserName -Leaf + Password = $SqlInstallCredential + } + + Group 'AddSqlInstallAsAdministrator' + { + Ensure = 'Present' + GroupName = 'Administrators' + MembersToInclude = Split-Path -Path $SqlInstallCredential.UserName -Leaf + } + + User 'CreateSqlAdminAccount' + { + Ensure = 'Present' + UserName = Split-Path -Path $SqlAdministratorCredential.UserName -Leaf + Password = $SqlAdministratorCredential + } + + WindowsFeature 'NetFramework45' + { + Name = 'NET-Framework-45-Core' + Ensure = 'Present' + } + } +} + +<# + .SYNOPSIS + Installs a named instance of Database Engine and Analysis Services. + + .NOTES + This is the instance that is used for many of the other integration tests. +#> +Configuration MSFT_SqlSetup_InstallDatabaseEngineNamedInstanceAsSystem_Config +{ + Import-DscResource -ModuleName 'SqlServerDsc' + + node $AllNodes.NodeName + { + SqlSetup 'Integration_Test' + { + FeatureFlag = @('DetectionSharedFeatures') + + InstanceName = $Node.DatabaseEngineNamedInstanceName + Features = $Node.DatabaseEngineNamedInstanceFeatures + SourcePath = "$($Node.DriveLetter):\" + SqlSvcStartupType = 'Automatic' + AgtSvcStartupType = 'Automatic' + BrowserSvcStartupType = 'Automatic' + SecurityMode = 'SQL' + SAPwd = $SqlAdministratorCredential + SQLCollation = $Node.Collation + SQLSvcAccount = $SqlServicePrimaryCredential + AgtSvcAccount = $SqlAgentServicePrimaryCredential + ASServerMode = $Node.AnalysisServicesMultiServerMode + AsSvcStartupType = 'Automatic' + ASCollation = $Node.Collation + ASSvcAccount = $SqlServicePrimaryCredential + InstallSharedDir = $Node.InstallSharedDir + InstallSharedWOWDir = $Node.InstallSharedWOWDir + UpdateEnabled = $Node.UpdateEnabled + SuppressReboot = $Node.SuppressReboot + ForceReboot = $Node.ForceReboot + SqlTempdbFileCount = $Node.SqlTempdbFileCount + SqlTempdbFileSize = $Node.SqlTempdbFileSize + SqlTempdbFileGrowth = $Node.SqlTempdbFileGrowth + SqlTempdbLogFileSize = $Node.SqlTempdbLogFileSize + SqlTempdbLogFileGrowth = $Node.SqlTempdbLogFileGrowth + + # This must be set if using SYSTEM account to install. + SQLSysAdminAccounts = @( + Split-Path -Path $SqlAdministratorCredential.UserName -Leaf + <# + Must have permission to properties IsClustered and + IsHadrEnable for SqlAlwaysOnService. + #> + Split-Path -Path $SqlInstallCredential.UserName -Leaf + ) + + # This must be set if using SYSTEM account to install. + ASSysAdminAccounts = @( + Split-Path -Path $SqlAdministratorCredential.UserName -Leaf + ) + } + } +} + +<# + .SYNOPSIS + Stopping the named instance to save memory on the build worker. + + .NOTES + The named instance is restarted at the end of the tests. +#> +Configuration MSFT_SqlSetup_StopServicesInstance_Config +{ + Import-DscResource -ModuleName 'PSDscResources' + + node $AllNodes.NodeName + { + <# + Stopping the SQL Server Agent service for the named instance. + It will be restarted at the end of the tests. + #> + Service ('StopSqlServerAgentForInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) + { + Name = ('SQLAGENT${0}' -f $Node.DatabaseEngineNamedInstanceName) + State = 'Stopped' + } + + <# + Stopping the Database Engine named instance. It will be restarted + at the end of the tests. + #> + Service ('StopSqlServerInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) + { + Name = ('MSSQL${0}' -f $Node.DatabaseEngineNamedInstanceName) + State = 'Stopped' + } + + Service ('StopMultiAnalysisServicesInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) + { + Name = ('MSOLAP${0}' -f $Node.DatabaseEngineNamedInstanceName) + State = 'Stopped' + } + } +} + +<# + .SYNOPSIS + Installs a default instance of Database Engine. +#> +Configuration MSFT_SqlSetup_InstallDatabaseEngineDefaultInstanceAsUser_Config +{ + Import-DscResource -ModuleName 'SqlServerDsc' + + node $AllNodes.NodeName + { + SqlSetup 'Integration_Test' + { + FeatureFlag = @('DetectionSharedFeatures') + + InstanceName = $Node.DatabaseEngineDefaultInstanceName + Features = $Node.DatabaseEngineDefaultInstanceFeatures + SourcePath = "$($Node.DriveLetter):\" + SQLCollation = $Node.Collation + SQLSvcAccount = $SqlServicePrimaryCredential + AgtSvcAccount = $SqlAgentServicePrimaryCredential + InstallSharedDir = $Node.InstallSharedDir + InstallSharedWOWDir = $Node.InstallSharedWOWDir + UpdateEnabled = $Node.UpdateEnabled + SuppressReboot = $Node.SuppressReboot + ForceReboot = $Node.ForceReboot + SQLSysAdminAccounts = @( + Split-Path -Path $SqlAdministratorCredential.UserName -Leaf + ) + + PsDscRunAsCredential = $SqlInstallCredential + } + } +} + +<# + .SYNOPSIS + Stopping the default instance to save memory on the build worker. +#> +Configuration MSFT_SqlSetup_StopSqlServerDefaultInstance_Config +{ + Import-DscResource -ModuleName 'PSDscResources' + + node $AllNodes.NodeName + { + Service ('StopSqlServerAgentForInstance{0}' -f $Node.DatabaseEngineDefaultInstanceName) + { + Name = 'SQLSERVERAGENT' + State = 'Stopped' + } + + + Service ('StopSqlServerInstance{0}' -f $Node.DatabaseEngineDefaultInstanceName) + { + Name = $Node.DatabaseEngineDefaultInstanceName + State = 'Stopped' + } + } +} + +<# + .SYNOPSIS + Installs a named instance of Analysis Services in tabular mode. +#> +Configuration MSFT_SqlSetup_InstallTabularAnalysisServicesAsSystem_Config +{ + Import-DscResource -ModuleName 'SqlServerDsc' + + node $AllNodes.NodeName + { + SqlSetup 'Integration_Test' + { + FeatureFlag = @('DetectionSharedFeatures') + + InstanceName = $Node.AnalysisServicesTabularInstanceName + Features = $Node.AnalysisServicesTabularFeatures + SourcePath = "$($Node.DriveLetter):\" + ASServerMode = $Node.AnalysisServicesTabularServerMode + ASCollation = $Node.Collation + ASSvcAccount = $SqlServicePrimaryCredential + InstallSharedDir = $Node.InstallSharedDir + InstallSharedWOWDir = $Node.InstallSharedWOWDir + UpdateEnabled = $Node.UpdateEnabled + SuppressReboot = $Node.SuppressReboot + ForceReboot = $Node.ForceReboot + + # This must be set if using SYSTEM account to install. + ASSysAdminAccounts = @( + Split-Path -Path $SqlAdministratorCredential.UserName -Leaf + ) + } + } +} + +<# + .SYNOPSIS + Stopping the Analysis Services tabular named instance to save memory on + the build worker. +#> +Configuration MSFT_SqlSetup_StopTabularAnalysisServices_Config +{ + Import-DscResource -ModuleName 'PSDscResources' + + node $AllNodes.NodeName + { + Service ('StopTabularAnalysisServicesInstance{0}' -f $Node.AnalysisServicesTabularInstanceName) + { + Name = ('MSOLAP${0}' -f $Node.DatabaseEngineNamedInstanceName) + State = 'Stopped' + } + } +} + +<# + .SYNOPSIS + Restarting the Database Engine named instance. + + .NOTES + This is so that other integration tests are dependent on this + named instance. +#> +Configuration MSFT_SqlSetup_StartServicesInstance_Config +{ + Import-DscResource -ModuleName 'PSDscResources' + + node $AllNodes.NodeName + { + # Start the Database Engine named instance. + Service ('StartSqlServerInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) + { + Name = ('MSSQL${0}' -f $Node.DatabaseEngineNamedInstanceName) + State = 'Running' + } + + # Starting the SQL Server Agent service for the named instance. + Service ('StartSqlServerAgentForInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) + { + Name = ('SQLAGENT${0}' -f $Node.DatabaseEngineNamedInstanceName) + State = 'Running' + } + } +} diff --git a/Tests/Integration/README.md b/Tests/Integration/README.md index 346699af8..77b3c6e63 100644 --- a/Tests/Integration/README.md +++ b/Tests/Integration/README.md @@ -22,13 +22,13 @@ AppVeyor build worker for other integration tests to use. Instance | Feature | AS server mode | State --- | --- | --- | --- -DSCSQL2016 | SQLENGINE,AS,CONN,BC,SDK | MULTIDIMENSIONAL | Running +DSCSQLTEST | SQLENGINE,AS,CONN,BC,SDK | MULTIDIMENSIONAL | Running DSCTABULAR | AS,CONN,BC,SDK | TABULAR | Stopped MSSQLSERVER | SQLENGINE,CONN,BC,SDK | - | Stopped All running Database Engine instances also have a SQL Server Agent that is started. -The instance DSCSQL2016 support mixed authentication mode. +The instance DSCSQLTEST support mixed authentication mode. >**Note:** Some services are stopped to save memory on the build worker. See the >column *State*. @@ -48,13 +48,13 @@ be used by other integration tests. User | Password | Permission | Description --- | --- | --- | --- -.\SqlInstall | P@ssw0rd1 | Local Windows administrator. Administrator of Database Engine instance DSCSQL2016\*. | Runs Setup for the default instance. +.\SqlInstall | P@ssw0rd1 | Local Windows administrator. Administrator of Database Engine instance DSCSQLTEST\*. | Runs Setup for the default instance. .\SqlAdmin | P@ssw0rd1 | Administrator of all SQL Server instances. | .\svc-SqlPrimary | yig-C^Equ3 | Local user. | Runs the SQL Server Agent service. .\svc-SqlAgentPri | yig-C^Equ3 | Local user. | Runs the SQL Server Agent service. .\svc-SqlSecondary | yig-C^Equ3 | Local user. | Used by other tests, but created here. .\svc-SqlAgentSec | yig-C^Equ3 | Local user. | Used by other tests. -sa | P@ssw0rd1 | Administrator of the Database Engine instances DSCSQL2016. | +sa | P@ssw0rd1 | Administrator of the Database Engine instances DSCSQLTEST. | *\* This is due to that the integration tests runs the resource SqlAlwaysOnService with this user and that means that this user must have permission to access the @@ -104,7 +104,7 @@ DSCRS2016 | RS | The Reporting Services is initialized, and in a working state. - **InstallSharedDir:** C:\Program Files\Microsoft SQL Server - **InstallSharedWOWDir:** C:\Program Files (x86)\Microsoft SQL Server - **DatabaseServerName:** `$env:COMPUTERNAME` -- **DatabaseInstanceName:** DSCSQL2016 +- **DatabaseInstanceName:** DSCSQLTEST ## SqlDatabaseDefaultLocation @@ -113,7 +113,7 @@ DSCRS2016 | RS | The Reporting Services is initialized, and in a working state. **Depends on:** SqlSetup The integration test will change the data, log and backup path of instance -**DSCSQL2016** to the following. +**DSCSQLTEST** to the following. Data | Log | Backup --- | --- | --- @@ -142,7 +142,7 @@ Username | Members | Member of | Permission DscSqlUsers1 | DscUser1, DscUser2 | *None* | *None* The integration tests will leave the following logins on the SQL Server instance -**DSCSQL2016**. +**DSCSQLTEST**. Login | Type | Password | Permission --- | --- | --- | --- @@ -160,7 +160,7 @@ DscUser4 | SQL | P@ssw0rd1 | *None* **Depends on:** SqlSetup, SqlServerLogin The integration test will keep the following server roles on the SQL Server instance -**DSCSQL2016**. +**DSCSQLTEST**. Server Role | Members --- | --- @@ -174,21 +174,21 @@ DscServerRole2 | DscUser4 **Depends on:** SqlSetup The integration tests will leave the following logins on the SQL Server instance -**DSCSQL2016**. +**DSCSQLTEST**. Login | Type | Password | Permission --- | --- | --- | --- DscAdmin1 | SQL | P@ssw0rd1 | dbcreator The integration test will change the following server roles on the SQL Server instance -**DSCSQL2016**. +**DSCSQLTEST**. Server Role | Members --- | --- dbcreator | DscAdmin1 The integration test will leave the following databases on the SQL Server instance -**DSCSQL2016**. +**DSCSQLTEST**. Database name | Owner --- | --- @@ -202,7 +202,7 @@ ScriptDatabase2 | DscAdmin1 **Depends on:** SqlScript The integration test will leave the following databases on the SQL Server instance -**DSCSQL2016**. +**DSCSQLTEST**. Database name | Owner --- | --- diff --git a/Tests/TestHelpers/CommonTestHelper.psm1 b/Tests/TestHelpers/CommonTestHelper.psm1 index 1050060d4..f81ac7514 100644 --- a/Tests/TestHelpers/CommonTestHelper.psm1 +++ b/Tests/TestHelpers/CommonTestHelper.psm1 @@ -314,9 +314,19 @@ function Test-SkipContinuousIntegrationTask [Parameter(Mandatory = $true)] [ValidateSet('Unit', 'Integration')] [System.String] - $Type + $Type, + + [Parameter()] + [System.String[]] + $Category ) + # Support using only the Type parameter as category names. + if (-not $Category) + { + $Category = @($Type) + } + $result = $false if ($Type -eq 'Integration' -and -not $env:APPVEYOR -eq $true) @@ -325,9 +335,9 @@ function Test-SkipContinuousIntegrationTask $result = $true } - if ($env:APPVEYOR -eq $true -and $env:CONFIGURATION -ne $Type) + if ($env:APPVEYOR -eq $true -and $env:CONFIGURATION -notin $Category) { - Write-Verbose -Message ('{1} tests in {0} will be skipped unless $env:CONFIGURATION is set to ''{1}''.' -f $Name, $Type) -Verbose + Write-Verbose -Message ('{1} tests in {0} will be skipped unless $env:CONFIGURATION is set to ''{1}''.' -f $Name, ($Category -join ''', or ''')) -Verbose $result = $true } diff --git a/Tests/Unit/MSFT_SqlSetup.Tests.ps1 b/Tests/Unit/MSFT_SqlSetup.Tests.ps1 index 79b4f3abe..2280713c0 100644 --- a/Tests/Unit/MSFT_SqlSetup.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlSetup.Tests.ps1 @@ -967,7 +967,7 @@ try These are written with both lower-case and upper-case to make sure we support that. The feature list must be written in the order it is returned by the function Get-TargetResource. #> - $defaultFeatures = 'SQLEngine,Replication,Dqc,Dq,FullText,Rs,As,Is,Bol,Conn,Bc,Sdk,Mds,Ssms,Adv_Ssms' + $defaultFeatures = 'SQLEngine,Replication,Dq,Dqc,FullText,Rs,As,Is,Bol,Conn,Bc,Sdk,Mds,Ssms,Adv_Ssms' # Default parameters that are used for the It-blocks $mockDefaultParameters = @{ @@ -1195,6 +1195,220 @@ try } } + if ($mockSqlMajorVersion -in (14)) + { + Context 'When using feature flag DetectionSharedFeatures' { + Context "When SQL Server version is $mockSqlMajorVersion and the system is in the desired state for default instance" { + BeforeEach { + $testParameters.Remove('Features') + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + FeatureFlag = @('DetectionSharedFeatures') + } + + Mock -CommandName Get-InstalledSharedFeatures -MockWith { + return @( + 'DQC' + 'BOL' + 'CONN' + 'BC' + 'SDK' + 'MDS' + ) + } -Verifiable + + Mock -CommandName New-SmbMapping -Verifiable + Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable + + #region Mock Get-CimInstance + Mock -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" + } -MockWith $mockGetCimInstance_DefaultInstance_DatabaseService -Verifiable + + Mock -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" + } -MockWith $mockGetCimInstance_DefaultInstance_AgentService -Verifiable + + Mock -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" + } -MockWith $mockGetCimInstance_DefaultInstance_FullTextService -Verifiable + + Mock -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" + } -MockWith $mockGetCimInstance_DefaultInstance_ReportingService -Verifiable + + Mock -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" + } -MockWith $mockGetCimInstance_DefaultInstance_IntegrationService -Verifiable + + Mock -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" + } -MockWith $mockGetCimInstance_DefaultInstance_AnalysisService -Verifiable + + Mock -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" + } -MockWith $mockGetItemProperty_InstanceId_ConfigurationState -Verifiable + + # If Get-CimInstance is used in any other way than those mocks with a ParameterFilter, then throw and error + Mock -CommandName Get-CimInstance -MockWith { + throw "Mock Get-CimInstance was called with unexpected parameters. ClassName=$ClassName, Filter=$Filter" + } -Verifiable + #endregion Mock Get-CimInstance + + Mock -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" + } -MockWith $mockGetItemProperty_InstanceId_ConfigurationState -Verifiable + + Mock -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\DQ\*" + } -MockWith $mockGetItemProperty_DQFeature -Verifiable + + Mock -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' + } -MockWith $mockGetItemProperty_Setup -Verifiable + } + + It 'Should return the same values as passed as parameters' { + $result = Get-TargetResource @testParameters + $result.InstanceName | Should -Be $testParameters.InstanceName + + Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or + $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or + $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or + $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or + $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or + $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) + } -Exactly -Times 0 -Scope It + + #region Assert Get-CimInstance + Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" + } -Exactly -Times 1 -Scope It + + Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { + $ClassName -eq 'Win32_Service' -and + $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" + } -Exactly -Times 1 -Scope It + #endregion Assert Get-CimInstance + } + + It 'Should return correct names of installed features' { + $result = Get-TargetResource @testParameters + + $result.Features | Should -Be 'SQLENGINE,REPLICATION,DQ,FULLTEXT,RS,AS,IS,DQC,BOL,CONN,BC,SDK,MDS' + } + + It 'Should return the correct values in the hash table' { + $result = Get-TargetResource @testParameters + $result.SourcePath | Should -Be $mockSourcePath + $result.InstanceName | Should -Be $mockDefaultInstance_InstanceName + $result.InstanceID | Should -Be $mockDefaultInstance_InstanceName + $result.InstallSharedDir | Should -Be $mockSqlSharedDirectory + $result.InstallSharedWOWDir | Should -Be $mockSqlSharedWowDirectory + $result.SQLSvcAccountUsername | Should -Be $mockSqlServiceAccount + $result.AgtSvcAccountUsername | Should -Be $mockAgentServiceAccount + $result.SqlCollation | Should -Be $mockSqlCollation + $result.SQLSysAdminAccounts | Should -Be $mockSqlSystemAdministrator + $result.SecurityMode | Should -Be 'Windows' + $result.InstallSQLDataDir | Should -Be $mockSqlInstallPath + $result.SQLUserDBDir | Should -Be $mockSqlDefaultDatabaseFilePath + $result.SQLUserDBLogDir | Should -Be $mockSqlDefaultDatabaseLogPath + $result.SQLBackupDir | Should -Be $mockDynamicSqlBackupPath + $result.FTSvcAccountUsername | Should -Be $mockSqlServiceAccount + $result.RSSvcAccountUsername | Should -Be $mockSqlServiceAccount + $result.ASSvcAccountUsername | Should -Be $mockSqlServiceAccount + $result.ASCollation | Should -Be $mockSqlAnalysisCollation + $result.ASSysAdminAccounts | Should -Be $mockSqlAnalysisAdmins + $result.ASDataDir | Should -Be $mockSqlAnalysisDataDirectory + $result.ASLogDir | Should -Be $mockSqlAnalysisLogDirectory + $result.ASBackupDir | Should -Be $mockSqlAnalysisBackupDirectory + $result.ASTempDir | Should -Be $mockSqlAnalysisTempDirectory + $result.ASConfigDir | Should -Be $mockSqlAnalysisConfigDirectory + $result.ASServerMode | Should -Be 'MULTIDIMENSIONAL' + $result.ISSvcAccountUsername | Should -Be $mockSqlServiceAccount + } + + $mockDynamicAnalysisServerMode = 'POWERPIVOT' + + It 'Should return the correct values in the hash table' { + $result = Get-TargetResource @testParameters + $result.ASServerMode | Should -Be 'POWERPIVOT' + } + + $mockDynamicAnalysisServerMode = 'TABULAR' + + It 'Should return the correct values in the hash table' { + $result = Get-TargetResource @testParameters + $result.ASServerMode | Should -Be 'TABULAR' + } + + # Return the state to the default for all other tests. + $mockDynamicAnalysisServerMode = 'MULTIDIMENSIONAL' + + <# + This is a regression test for issue #691. + This sets administrators to only one for mock Connect-SQLAnalysis. + #> + + $mockSqlAnalysisSingleAdministrator = 'COMPANY\AnalysisAdmin' + $mockDynamicSqlAnalysisAdmins = $mockSqlAnalysisSingleAdministrator + + It 'Should return the correct type and value for property ASSysAdminAccounts' { + $result = Get-TargetResource @testParameters + Write-Output -NoEnumerate $result.ASSysAdminAccounts | Should -BeOfType [System.String[]] + $result.ASSysAdminAccounts | Should -Be $mockSqlAnalysisSingleAdministrator + } + + # Setting back the default administrators for mock Connect-SQLAnalysis. + $mockDynamicSqlAnalysisAdmins = $mockSqlAnalysisAdmins + } + } + } + Context "When using SourceCredential parameter and SQL Server version is $mockSqlMajorVersion and the system is not in the desired state for default instance" { BeforeEach { $testParameters.Remove('Features') @@ -1429,11 +1643,11 @@ try $result = Get-TargetResource @testParameters if ($mockSqlMajorVersion -in (13,14)) { - $result.Features | Should -Be 'SQLENGINE,REPLICATION,DQC,DQ,FULLTEXT,RS,AS,IS,BOL,MDS' + $result.Features | Should -Be 'SQLENGINE,REPLICATION,DQ,DQC,FULLTEXT,RS,AS,IS,BOL,MDS' } else { - $result.Features | Should -Be 'SQLENGINE,REPLICATION,DQC,DQ,FULLTEXT,RS,AS,IS,BOL,MDS,SSMS,ADV_SSMS' + $result.Features | Should -Be 'SQLENGINE,REPLICATION,DQ,DQC,FULLTEXT,RS,AS,IS,BOL,MDS,SSMS,ADV_SSMS' } } } @@ -1682,7 +1896,6 @@ try # Setting back the default administrators for mock Connect-SQLAnalysis. $mockDynamicSqlAnalysisAdmins = $mockSqlAnalysisAdmins - } Context "When using SourceCredential parameter and SQL Server version is $mockSqlMajorVersion and the system is in the desired state for default instance" { @@ -3043,9 +3256,8 @@ try } } - $testProductVersion | ForEach-Object -Process { - $mockSqlMajorVersion = $_ - + foreach ($mockSqlMajorVersion in $testProductVersion) + { $mockDefaultInstance_InstanceId = "$($mockSqlDatabaseEngineName)$($mockSqlMajorVersion).$($mockDefaultInstance_InstanceName)" $mockSqlInstallPath = "C:\Program Files\Microsoft SQL Server\$($mockDefaultInstance_InstanceId)\MSSQL" @@ -4824,9 +5036,8 @@ try New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList 'COMPANY\$sql.service', $mockServiceAccountPassword } - $serviceTypes | ForEach-Object { - $serviceType = $_ - + foreach ($serviceType in $serviceTypes ) + { Context "When service type is $serviceType" { $mockAccountArgumentName = ('{0}SVCACCOUNT' -f $serviceType) $mockPasswordArgumentName = ('{0}SVCPASSWORD' -f $serviceType) @@ -4877,7 +5088,7 @@ try } Context 'When using Get-TemporaryFolder' { - It 'Should return the correct temporary path.' { + It 'Should return the correct temporary path' { Get-TemporaryFolder | Should -BeExactly $mockExpectedTempPath } } @@ -4909,6 +5120,85 @@ try } } } + + Describe 'Get-InstalledSharedFeatures' -Tag 'Helper' { + Context 'When there are no shared features installed' { + BeforeAll { + $mockSqlMajorVersion = 14 + + Mock -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" + } -MockWith { + return @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'UnknownKey' -Value 1 -PassThru -Force + ) + ) + } -Verifiable + } + + It 'Should return an empty array' { + $getInstalledSharedFeaturesResult = Get-InstalledSharedFeatures -SqlVersion $mockSqlMajorVersion + + $getInstalledSharedFeaturesResult | Should -HaveCount 0 + } + } + + Context 'When there are shared features installed' { + BeforeAll { + $mockSqlMajorVersion = 14 + + Mock -CommandName Get-ItemProperty -ParameterFilter { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" + } -MockWith { + return @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'SQL_DQ_CLIENT_Full' -Value 1 -PassThru | + Add-Member -MemberType NoteProperty -Name 'SQL_BOL_Components' -Value 1 -PassThru | + Add-Member -MemberType NoteProperty -Name 'Connectivity_Full' -Value 1 -PassThru | + Add-Member -MemberType NoteProperty -Name 'Tools_Legacy_Full' -Value 1 -PassThru | + Add-Member -MemberType NoteProperty -Name 'SDK_Full' -Value 1 -PassThru | + Add-Member -MemberType NoteProperty -Name 'MDSCoreFeature' -Value 1 -PassThru -Force + ) + ) + } -Verifiable + } + + It 'Should return the correct array with installed shared features' { + $getInstalledSharedFeaturesResult = Get-InstalledSharedFeatures -SqlVersion $mockSqlMajorVersion + + $getInstalledSharedFeaturesResult | Should -HaveCount 6 + $getInstalledSharedFeaturesResult | Should -Contain 'DQC' + $getInstalledSharedFeaturesResult | Should -Contain 'BOL' + $getInstalledSharedFeaturesResult | Should -Contain 'CONN' + $getInstalledSharedFeaturesResult | Should -Contain 'BC' + $getInstalledSharedFeaturesResult | Should -Contain 'SDK' + $getInstalledSharedFeaturesResult | Should -Contain 'MDS' + } + } + } + + Describe 'Test-FeatureFlag' -Tag 'Helper' { + Context 'When no feature flags was provided' { + It 'Should return $false' { + Test-FeatureFlag -FeatureFlag $null -TestFlag 'MyFlag' | Should -Be $false + } + } + + Context 'When feature flags was provided' { + It 'Should return $true' { + Test-FeatureFlag -FeatureFlag @('FirstFlag','SecondFlag') -TestFlag 'SecondFlag' | Should -Be $true + } + } + + Context 'When feature flags was provided, but missing' { + It 'Should return $false' { + Test-FeatureFlag -FeatureFlag @('MyFlag2') -TestFlag 'MyFlag' | Should -Be $false + } + } + } } } finally diff --git a/appveyor.yml b/appveyor.yml index dc97e489d..7abad0b23 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -6,7 +6,7 @@ environment: gallery_api: secure: 9ekJzfsPCDBkyLrfmov83XbbhZ6E2N3z+B/Io8NbDetbHc6hWS19zsDmy7t0Vvxv # The job where deploy step should run (normally the last job) - DeployInJobNumber: 3 + DeployInJobNumber: 4 # The image that will be used when building the job matrix. image: Visual Studio 2017 @@ -18,7 +18,8 @@ image: Visual Studio 2017 configuration: - Meta - Unit - - Integration + - Integration_SQL2016 + - Integration_SQL2017 install: - git clone https://github.com/PowerShell/DscResource.Tests @@ -55,7 +56,8 @@ for: - matrix: only: - - configuration: Integration + - configuration: Integration_SQL2016 + - configuration: Integration_SQL2017 environment: SkipAllCommonTests: True From befa13f34959d326605eb7515141c28b13de6f46 Mon Sep 17 00:00:00 2001 From: nesith <47059791+nesith@users.noreply.github.com> Date: Sun, 3 Mar 2019 23:56:11 +1000 Subject: [PATCH 04/10] SqlSetup: Fixes issue #1254 (#1299) - Changes to SqlSetup - Changed the logic of 'Build the argument string to be passed to setup' to not quote the value if root directory is specified ([issue #1254](https://github.com/PowerShell/SqlServerDsc/issues/1254)). --- CHANGELOG.md | 3 +++ DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 | 11 ++++++++++- Tests/Unit/MSFT_SqlSetup.Tests.ps1 | 1 + 3 files changed, 14 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index abe7546a7..1bf3e0fa4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -30,6 +30,9 @@ - Added a new helper function `Get-InstalledSharedFeatures` to move out some of the code from the `Get-TargetResource` to make unit testing easier and faster. + - Changed the logic of 'Build the argument string to be passed to setup' to + not quote the value if root directory is specified + ([issue #1254](https://github.com/PowerShell/SqlServerDsc/issues/1254)). ## 12.3.0.0 diff --git a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 index 6f59e3715..e26c91724 100644 --- a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 +++ b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 @@ -1574,7 +1574,16 @@ function Set-TargetResource } else { - $setupArgumentValue = '"{0}"' -f $currentSetupArgument.Value + # Logic added as a fix for Issue#1254 SqlSetup:Fails when a root directory is specified + if($currentSetupArgument.Value -match '^[a-zA-Z]:\\$') + { + $setupArgumentValue = $currentSetupArgument.Value + } + else + { + $setupArgumentValue = '"{0}"' -f $currentSetupArgument.Value + } + } } diff --git a/Tests/Unit/MSFT_SqlSetup.Tests.ps1 b/Tests/Unit/MSFT_SqlSetup.Tests.ps1 index 2280713c0..84c79d68a 100644 --- a/Tests/Unit/MSFT_SqlSetup.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlSetup.Tests.ps1 @@ -3348,6 +3348,7 @@ try It 'Should set the system in the desired state when feature is SQLENGINE' { $testParameters = $mockDefaultParameters.Clone() + # This is also used to regression test issue #1254, SqlSetup fails when root directory is specified. $testParameters += @{ SQLSysAdminAccounts = 'COMPANY\User1','COMPANY\SQLAdmins' ASSysAdminAccounts = 'COMPANY\User1','COMPANY\SQLAdmins' From 9801548fb9e359c2ca6aba09c5c007a59bf907e5 Mon Sep 17 00:00:00 2001 From: Cody Konior Date: Tue, 5 Mar 2019 13:44:44 +0800 Subject: [PATCH 05/10] SqlAGDatabase: Fix impersonate permission test (#1295) - Changes to SqlAGDatabase - Fix MatchDatabaseOwner to check for CONTROL SERVER, IMPERSONATE LOGIN, or CONTROL LOGIN permission in addition to IMPERSONATE ANY LOGIN. - Update and fix MatchDatabaseOwner help text. - Changes to xSQLServerHelper - New-TerminatingError error text for a missing localized message now matches the output even if the "missing localized message" localized message is --- CHANGELOG.md | 8 + .../MSFT_SqlAGDatabase.psm1 | 88 ++++++--- .../MSFT_SqlAGDatabase.schema.mof | 2 +- .../en-US/MSFT_SqlAGDatabase.strings.psd1 | 2 +- .../en-US/about_SqlAGDatabase.help.txt | 7 +- README.md | 7 +- SqlServerDscHelper.psm1 | 156 +++++++++++++-- Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 | 53 ++--- Tests/Unit/SqlServerDSCHelper.Tests.ps1 | 183 +++++++++++++++--- Tests/Unit/Stubs/SMO.cs | 4 +- 10 files changed, 397 insertions(+), 113 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1bf3e0fa4..082487141 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -33,6 +33,14 @@ - Changed the logic of 'Build the argument string to be passed to setup' to not quote the value if root directory is specified ([issue #1254](https://github.com/PowerShell/SqlServerDsc/issues/1254)). +- Changes to SqlAGDatabase + - Fix MatchDatabaseOwner to check for CONTROL SERVER, IMPERSONATE LOGIN, or + CONTROL LOGIN permission in addition to IMPERSONATE ANY LOGIN. + - Update and fix MatchDatabaseOwner help text. +- Changes to xSQLServerHelper + - New-TerminatingError error text for a missing localized message now matches + the output even if the "missing localized message" localized message is + also missing. ## 12.3.0.0 diff --git a/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 b/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 index 6660f8fec..58381970c 100644 --- a/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 +++ b/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 @@ -138,11 +138,12 @@ function Get-TargetResource .PARAMETER MatchDatabaseOwner If set to $true, this ensures the database owner of the database on the primary replica is the owner of the database on all secondary replicas. This requires the database owner is available - as a login on all replicas and that the PSDscRunAsCredential has impersonate permissions. + as a login on all replicas and that the PsDscRunAsCredential has impersonate any login, control + server, impersonate login, or control login permissions. - If set to $false, the owner of the database will be the PSDscRunAsCredential. + If set to $false, the owner of the database will be the PsDscRunAsCredential. - The default is '$true'. + The default is '$false'. .PARAMETER ProcessOnlyOnActiveNode Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. @@ -232,27 +233,32 @@ function Set-TargetResource # Get only the secondary replicas. Some tests do not need to be performed on the primary replica $secondaryReplicas = $availabilityGroup.AvailabilityReplicas | Where-Object -FilterScript { $_.Role -ne 'Primary' } - # Ensure the appropriate permissions are in place on all the replicas - if ( $MatchDatabaseOwner ) + foreach ( $databaseToAddToAvailabilityGroup in $databasesToAddToAvailabilityGroup ) { - $impersonatePermissionsStatus = @{} + $databaseObject = $primaryServerObject.Databases[$databaseToAddToAvailabilityGroup] - foreach ( $availabilityGroupReplica in $secondaryReplicas ) + # Ensure the appropriate permissions are in place on all the replicas + if ( $MatchDatabaseOwner ) { - $currentAvailabilityGroupReplicaServerObject = Connect-SQL -ServerName $availabilityGroupReplica.Name - $impersonatePermissionsStatus.Add( - $availabilityGroupReplica.Name, - ( Test-ImpersonatePermissions -ServerObject $currentAvailabilityGroupReplicaServerObject ) - ) - } + $impersonatePermissionsStatus = @{} - if ( $impersonatePermissionsStatus.Values -contains $false ) - { - $impersonatePermissionsMissingParameters = @( - [System.Security.Principal.WindowsIdentity]::GetCurrent().Name, - ( ( $impersonatePermissionsStatus.GetEnumerator() | Where-Object -FilterScript { -not $_.Value } | Select-Object -ExpandProperty Key ) -join ', ' ) - ) - throw ($script:localizedData.ImpersonatePermissionsMissing -f $impersonatePermissionsMissingParameters ) + foreach ( $availabilityGroupReplica in $secondaryReplicas ) + { + $currentAvailabilityGroupReplicaServerObject = Connect-SQL -ServerName $availabilityGroupReplica.Name + $impersonatePermissionsStatus.Add( + $availabilityGroupReplica.Name, + ( Test-ImpersonatePermissions -ServerObject $currentAvailabilityGroupReplicaServerObject -Securable $databaseObject.Owner ) + ) + } + + if ( $impersonatePermissionsStatus.Values -contains $false ) + { + $impersonatePermissionsMissingParameters = @( + [System.Security.Principal.WindowsIdentity]::GetCurrent().Name, + ( ( $impersonatePermissionsStatus.GetEnumerator() | Where-Object -FilterScript { -not $_.Value } | Select-Object -ExpandProperty Key ) -join ', ' ) + ) + throw ($script:localizedData.ImpersonatePermissionsMissing -f $impersonatePermissionsMissingParameters ) + } } } @@ -481,15 +487,36 @@ function Set-TargetResource $restoreDatabaseQueryStringBuilder.Append($databaseFullBackupFile) | Out-Null $restoreDatabaseQueryStringBuilder.AppendLine('''') | Out-Null $restoreDatabaseQueryStringBuilder.Append('WITH NORECOVERY') | Out-Null + if ( $MatchDatabaseOwner ) + { + $restoreDatabaseQueryStringBuilder.AppendLine() | Out-Null + $restoreDatabaseQueryStringBuilder.Append('REVERT') | Out-Null + } $restoreDatabaseQueryString = $restoreDatabaseQueryStringBuilder.ToString() - # Build the parameters to restore the transaction log - $restoreSqlDatabaseLogParameters = @{ - Database = $databaseToAddToAvailabilityGroup - BackupFile = $databaseLogBackupFile - RestoreAction = 'Log' - NoRecovery = $true + # Need to restore the database with a query in order to impersonate the correct login + $restoreLogQueryStringBuilder = New-Object -TypeName System.Text.StringBuilder + + if ( $MatchDatabaseOwner ) + { + $restoreLogQueryStringBuilder.Append('EXECUTE AS LOGIN = ''') | Out-Null + $restoreLogQueryStringBuilder.Append($databaseObject.Owner) | Out-Null + $restoreLogQueryStringBuilder.AppendLine('''') | Out-Null + } + + $restoreLogQueryStringBuilder.Append('RESTORE DATABASE [') | Out-Null + $restoreLogQueryStringBuilder.Append($databaseToAddToAvailabilityGroup) | Out-Null + $restoreLogQueryStringBuilder.AppendLine(']') | Out-Null + $restoreLogQueryStringBuilder.Append('FROM DISK = ''') | Out-Null + $restoreLogQueryStringBuilder.Append($databaseLogBackupFile) | Out-Null + $restoreLogQueryStringBuilder.AppendLine('''') | Out-Null + $restoreLogQueryStringBuilder.Append('WITH NORECOVERY') | Out-Null + if ( $MatchDatabaseOwner ) + { + $restoreLogQueryStringBuilder.AppendLine() | Out-Null + $restoreLogQueryStringBuilder.Append('REVERT') | Out-Null } + $restoreLogQueryString = $restoreLogQueryStringBuilder.ToString() try { @@ -502,7 +529,7 @@ function Set-TargetResource # Restore the database Invoke-Query -SQLServer $currentAvailabilityGroupReplicaServerObject.NetName -SQLInstanceName $currentAvailabilityGroupReplicaServerObject.ServiceName -Database master -Query $restoreDatabaseQueryString - Restore-SqlDatabase -InputObject $currentAvailabilityGroupReplicaServerObject @restoreSqlDatabaseLogParameters + Invoke-Query -SQLServer $currentAvailabilityGroupReplicaServerObject.NetName -SQLInstanceName $currentAvailabilityGroupReplicaServerObject.ServiceName -Database master -Query $restoreLogQueryString # Add the database to the Availability Group Add-SqlAvailabilityDatabase -InputObject $currentReplicaAvailabilityGroupObject -Database $databaseToAddToAvailabilityGroup @@ -604,11 +631,12 @@ function Set-TargetResource .PARAMETER MatchDatabaseOwner If set to $true, this ensures the database owner of the database on the primary replica is the owner of the database on all secondary replicas. This requires the database owner is available - as a login on all replicas and that the PSDscRunAsCredential has impersonate permissions. + as a login on all replicas and that the PsDscRunAsCredential has impersonate any login, control + server, impersonate login, or control login permissions. - If set to $false, the owner of the database will be the PSDscRunAsCredential. + If set to $false, the owner of the database will be the PsDscRunAsCredential. - The default is '$true'. + The default is '$false'. .PARAMETER ProcessOnlyOnActiveNode Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. diff --git a/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.schema.mof b/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.schema.mof index a4c183253..d25a1bae3 100644 --- a/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.schema.mof +++ b/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.schema.mof @@ -8,7 +8,7 @@ class MSFT_SqlAGDatabase : OMI_BaseResource [Required, Description("The path used to seed the availability group replicas. This should be a path that is accessible by all of the replicas")] String BackupPath; [Write, Description("Specifies the membership of the database(s) in the availability group. The options are: Present: The defined database(s) are added to the availability group. All other databases that may be a member of the availability group are ignored. Absent: The defined database(s) are removed from the availability group. All other databases that may be a member of the availability group are ignored. The default is 'Present'."), ValueMap{"Present","Absent"}, Values{"Present","Absent"}] String Ensure; [Write, Description("When used with 'Ensure = 'Present'' it ensures the specified database(s) are the only databases that are a member of the specified Availability Group. This parameter is ignored when 'Ensure' is 'Absent'.")] Boolean Force; - [Write, Description("If set to $true, this ensures the database owner of the database on the primary replica is the owner of the database on all secondary replicas. This requires the database owner is available as a login on all replicas and that the PsDscRunAsCredential has impersonate permissions. If set to $false, the owner of the database will be the PsDscRunAsCredential. The default is '$true'")] Boolean MatchDatabaseOwner; + [Write, Description("If set to $true, this ensures the database owner of the database on the primary replica is the owner of the database on all secondary replicas. This requires the database owner is available as a login on all replicas and that the PsDscRunAsCredential has impersonate any login, control server, impersonate login, or control login permissions. If set to $false, the owner of the database will be the PsDscRunAsCredential. The default is '$false'")] Boolean MatchDatabaseOwner; [Write, Description("Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance.")] Boolean ProcessOnlyOnActiveNode; [Read, Description("Determines if the current node is actively hosting the SQL Server instance.")] Boolean IsActiveNode; }; diff --git a/DSCResources/MSFT_SqlAGDatabase/en-US/MSFT_SqlAGDatabase.strings.psd1 b/DSCResources/MSFT_SqlAGDatabase/en-US/MSFT_SqlAGDatabase.strings.psd1 index 112cf9e1e..6e08e9903 100644 --- a/DSCResources/MSFT_SqlAGDatabase/en-US/MSFT_SqlAGDatabase.strings.psd1 +++ b/DSCResources/MSFT_SqlAGDatabase/en-US/MSFT_SqlAGDatabase.strings.psd1 @@ -7,7 +7,7 @@ ConvertFrom-StringData @' DatabaseShouldBeMember = The following databases should be a member of the availability group '{0}': {1}. DatabaseShouldNotBeMember = The following databases should not be a member of the availability group '{0}': {1}. DatabasesNotFound = The following databases were not found in the instance: {0}. - ImpersonatePermissionsMissing = The login '{0}' is missing impersonate permissions in the instances '{1}'. + ImpersonatePermissionsMissing = The login '{0}' is missing impersonate any login, control server, impersonate login, or control login permissions in the instances '{1}'. NotActiveNode = The node '{0}' is not actively hosting the instance '{1}'. Exiting the test. ParameterNotOfType = The parameter '{0}' is not of the type '{1}'. ParameterNullOrEmpty = The parameter '{0}' is NULL or empty. diff --git a/DSCResources/MSFT_SqlAGDatabase/en-US/about_SqlAGDatabase.help.txt b/DSCResources/MSFT_SqlAGDatabase/en-US/about_SqlAGDatabase.help.txt index 706886d20..214e0e88c 100644 --- a/DSCResources/MSFT_SqlAGDatabase/en-US/about_SqlAGDatabase.help.txt +++ b/DSCResources/MSFT_SqlAGDatabase/en-US/about_SqlAGDatabase.help.txt @@ -40,11 +40,12 @@ PARAMETER Force PARAMETER MatchDatabaseOwner If set to $true, this ensures the database owner of the database on the primary replica is the owner of the database on all secondary replicas. This requires the database owner is available - as a login on all replicas and that the PSDscRunAsCredential has impersonate permissions. + as a login on all replicas and that the PsDscRunAsCredential has impersonate any login, control + server, impersonate login, or control login permissions. - If set to $false, the owner of the database will be the PSDscRunAsCredential. + If set to $false, the owner of the database will be the PsDscRunAsCredential. - The default is '$true'. + The default is '$false'. .PARAMETER ProcessOnlyOnActiveNode Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. diff --git a/README.md b/README.md index e5b5bfd79..9ce9bfccb 100644 --- a/README.md +++ b/README.md @@ -286,10 +286,11 @@ group. Availability Group. This parameter is ignored when 'Ensure' is 'Absent'. * **`[Boolean]` MatchDatabaseOwner** _(Write)_: If set to $true, this ensures the database owner of the database on the primary replica is the owner of the database - on all secondary replicas. This requires the database owner is available as a - login on all replicas and that the PsDscRunAsCredential has impersonate permissions. + on all secondary replicas. This requires the database owner is available + as a login on all replicas and that the PsDscRunAsCredential has impersonate any + login, control server, impersonate login, or control login permissions. If set to $false, the owner of the database will be the PsDscRunAsCredential. - The default is '$true'. + The default is '$false'. * **`[Boolean]` ProcessOnlyOnActiveNode** _(Write)_: Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. diff --git a/SqlServerDscHelper.psm1 b/SqlServerDscHelper.psm1 index 059a2d35f..8f3f682fc 100644 --- a/SqlServerDscHelper.psm1 +++ b/SqlServerDscHelper.psm1 @@ -289,7 +289,7 @@ function New-TerminatingError if (!$errorMessage) { - $errorMessage = ("No Localization key found for key: {0}" -f $ErrorType) + $errorMessage = ("No Localization key found for ErrorType: '{0}'." -f $ErrorType) } } @@ -1059,6 +1059,43 @@ function Update-AvailabilityGroupReplica } } +<# + .SYNOPSIS + Impersonates a login and determines whether required permissions are present. + + .PARAMETER SQLServer + String containing the host name of the SQL Server to connect to. + + .PARAMETER SQLInstanceName + String containing the SQL Server Database Engine instance to connect to. + + .PARAMETER LoginName + String containing the login (user) which should be checked for a permission. + + .PARAMETER Permissions + This is a list that represents a SQL Server set of database permissions. + + .PARAMETER SecurableClass + String containing the class of permissions to test. It can be: + SERVER: A permission that is applicable against server objects. + LOGIN: A permission that is applicable against login objects. + + Default is 'SERVER'. + + .PARAMETER SecurableName + String containing the name of the object against which permissions exist, e.g. if SecurableClass is LOGIN this is the name of a login permissions may exist against. + + Default is $null. + + .NOTES + These SecurableClass are not yet in this module yet and so are not implemented: + 'APPLICATION ROLE', 'ASSEMBLY', 'ASYMMETRIC KEY', 'CERTIFICATE', + 'CONTRACT', 'DATABASE', 'ENDPOINT', 'FULLTEXT CATALOG', + 'MESSAGE TYPE', 'OBJECT', 'REMOTE SERVICE BINDING', 'ROLE', + 'ROUTE', 'SCHEMA', 'SERVICE', 'SYMMETRIC KEY', 'TYPE', 'USER', + 'XML SCHEMA COLLECTION' + +#> function Test-LoginEffectivePermissions { param @@ -1079,7 +1116,16 @@ function Test-LoginEffectivePermissions [Parameter(Mandatory = $true)] [System.String[]] - $Permissions + $Permissions, + + [ValidateSet('SERVER', 'LOGIN')] + [Parameter()] + [System.String] + $SecurableClass = 'SERVER', + + [Parameter()] + [System.String] + $SecurableName ) # Assume the permissions are not present @@ -1092,12 +1138,24 @@ function Test-LoginEffectivePermissions WithResults = $true } - $queryToGetEffectivePermissionsForLogin = " - EXECUTE AS LOGIN = '$LoginName' - SELECT DISTINCT permission_name - FROM fn_my_permissions(null,'SERVER') - REVERT - " + if ( [System.String]::IsNullOrEmpty($SecurableName) ) + { + $queryToGetEffectivePermissionsForLogin = " + EXECUTE AS LOGIN = '$LoginName' + SELECT DISTINCT permission_name + FROM fn_my_permissions(null,'$SecurableClass') + REVERT + " + } + else + { + $queryToGetEffectivePermissionsForLogin = " + EXECUTE AS LOGIN = '$LoginName' + SELECT DISTINCT permission_name + FROM fn_my_permissions('$SecurableName','$SecurableClass') + REVERT + " + } Write-Verbose -Message ($script:localizedData.GetEffectivePermissionForLogin -f $LoginName, $sqlInstanceName) -Verbose @@ -1233,6 +1291,10 @@ function Get-PrimaryReplicaServerObject .PARAMETER ServerObject The server object on which to perform the test. + + .PARAMETER SecurableName + If set then impersonate permission on this specific securable (e.g. login) is also checked. + #> function Test-ImpersonatePermissions { @@ -1240,9 +1302,14 @@ function Test-ImpersonatePermissions ( [Parameter(Mandatory = $true)] [Microsoft.SqlServer.Management.Smo.Server] - $ServerObject + $ServerObject, + + [Parameter()] + [System.String] + $SecurableName ) + # The impersonate any login permission only exists in SQL 2014 and above $testLoginEffectivePermissionsParams = @{ SQLServer = $ServerObject.ComputerNamePhysicalNetBIOS SQLInstanceName = $ServerObject.ServiceName @@ -1251,12 +1318,77 @@ function Test-ImpersonatePermissions } $impersonatePermissionsPresent = Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams + if ($impersonatePermissionsPresent) + { + New-VerboseMessage -Message ( 'The login "{0}" has impersonate any login permissions on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) + return $impersonatePermissionsPresent + } + else + { + New-VerboseMessage -Message ( 'The login "{0}" does not have impersonate any login permissions on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) + } - if ( -not $impersonatePermissionsPresent ) + # Check for sysadmin / control server permission which allows impersonation + $testLoginEffectivePermissionsParams = @{ + SQLServer = $ServerObject.ComputerNamePhysicalNetBIOS + SQLInstanceName = $ServerObject.ServiceName + LoginName = $ServerObject.ConnectionContext.TrueLogin + Permissions = @('CONTROL SERVER') + } + $impersonatePermissionsPresent = Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams + if ($impersonatePermissionsPresent) + { + New-VerboseMessage -Message ( 'The login "{0}" has control server permissions on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) + return $impersonatePermissionsPresent + } + else { - New-VerboseMessage -Message ( 'The login "{0}" does not have impersonate permissions on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) + New-VerboseMessage -Message ( 'The login "{0}" does not have control server permissions on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) } + if ( -not [System.String]::IsNullOrEmpty($SecurableName) ) { + # Check for login-specific impersonation permissions + $testLoginEffectivePermissionsParams = @{ + SQLServer = $ServerObject.ComputerNamePhysicalNetBIOS + SQLInstanceName = $ServerObject.ServiceName + LoginName = $ServerObject.ConnectionContext.TrueLogin + Permissions = @('IMPERSONATE') + SecurableClass = 'LOGIN' + SecurableName = $SecurableName + } + $impersonatePermissionsPresent = Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams + if ($impersonatePermissionsPresent) + { + New-VerboseMessage -Message ( 'The login "{0}" has impersonate permissions on the instance "{1}\{2}" for the login "{3}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName, $SecurableName ) + return $impersonatePermissionsPresent + } + else + { + New-VerboseMessage -Message ( 'The login "{0}" does not have impersonate permissions on the instance "{1}\{2}" for the login "{3}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName, $SecurableName ) + } + + # Check for login-specific control permissions + $testLoginEffectivePermissionsParams = @{ + SQLServer = $ServerObject.ComputerNamePhysicalNetBIOS + SQLInstanceName = $ServerObject.ServiceName + LoginName = $ServerObject.ConnectionContext.TrueLogin + Permissions = @('CONTROL') + SecurableClass = 'LOGIN' + SecurableName = $SecurableName + } + $impersonatePermissionsPresent = Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams + if ($impersonatePermissionsPresent) + { + New-VerboseMessage -Message ( 'The login "{0}" has control permissions on the instance "{1}\{2}" for the login "{3}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName, $SecurableName ) + return $impersonatePermissionsPresent + } + else + { + New-VerboseMessage -Message ( 'The login "{0}" does not have control permissions on the instance "{1}\{2}" for the login "{3}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName, $SecurableName ) + } + } + + New-VerboseMessage -Message ( 'The login "{0}" does not have any impersonate permissions required on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) return $impersonatePermissionsPresent } @@ -1572,7 +1704,7 @@ function Get-ServiceAccount function Find-ExceptionByNumber { # Define parameters - param + param ( [Parameter(Mandatory = $true)] [System.Exception] diff --git a/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 b/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 index 43f35edb3..788ff9f74 100644 --- a/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 @@ -88,6 +88,8 @@ try $mockAvailabilityGroupObjectName = 'AvailabilityGroup1' $mockAvailabilityGroupWithoutDatabasesObjectName = 'AvailabilityGroupWithoutDatabases' $mockAvailabilityGroupObjectWithPrimaryReplicaOnAnotherServerName = 'AvailabilityGroup2' + $mockTrueLogin = 'Login1' + $mockDatabaseOwner = 'DatabaseOwner1' #endregion mock names @@ -279,6 +281,7 @@ try $newDatabaseObject.LogFiles = @{ FileName = ( [IO.Path]::Combine( $mockLogFilePath, "$($mockPresentDatabaseName).ldf" ) ) } + $newDatabaseObject.Owner = $mockDatabaseOwner # Add the database object to the database collection $mockDatabaseObjects.Add($newDatabaseObject) @@ -299,6 +302,7 @@ try $newDatabaseObject.LogFiles = @{ FileName = ( [IO.Path]::Combine( $mockLogFilePathIncorrect, "$($mockPresentDatabaseName).ldf" ) ) } + $newDatabaseObject.Owner = $mockDatabaseOwner # Add the database object to the database collection $mockDatabaseObjectsWithIncorrectFileNames.Add($newDatabaseObject) @@ -315,6 +319,9 @@ try $mockServerObject.AvailabilityGroups.Add($mockAvailabilityGroupObject.Clone()) $mockServerObject.AvailabilityGroups.Add($mockAvailabilityGroupWithoutDatabasesObject.Clone()) $mockServerObject.AvailabilityGroups.Add($mockAvailabilityGroupObjectWithPrimaryReplicaOnAnotherServer.Clone()) + $mockServerObject.ComputerNamePhysicalNetBIOS = $mockServerObjectDomainInstanceName + $mockServerObject.ConnectionContext = New-Object -TypeName Microsoft.SqlServer.Management.Smo.ServerConnection + $mockServerObject.ConnectionContext.TrueLogin = $mockTrueLogin $mockServerObject.Databases = $mockDatabaseObjects $mockServerObject.DomainInstanceName = $mockServerObjectDomainInstanceName $mockServerObject.NetName = $mockServerObjectDomainInstanceName @@ -328,6 +335,9 @@ try $mockServer2Object.AvailabilityGroups.Add($mockAvailabilityGroupObject.Clone()) $mockServer2Object.AvailabilityGroups.Add($mockAvailabilityGroupWithoutDatabasesObject.Clone()) $mockServer2Object.AvailabilityGroups.Add($mockAvailabilityGroupObjectWithPrimaryReplicaOnAnotherServer.Clone()) + $mockServer2Object.ComputerNamePhysicalNetBIOS = $mockPrimaryServerObjectDomainInstanceName + $mockServer2Object.ConnectionContext = New-Object -TypeName Microsoft.SqlServer.Management.Smo.ServerConnection + $mockServer2Object.ConnectionContext.TrueLogin = $mockTrueLogin $mockServer2Object.Databases = $mockDatabaseObjects $mockServer2Object.DomainInstanceName = $mockPrimaryServerObjectDomainInstanceName $mockServer2Object.NetName = $mockPrimaryServerObjectDomainInstanceName @@ -370,7 +380,8 @@ WITH NORECOVERY' $Query -like 'EXECUTE AS LOGIN = * RESTORE DATABASE * FROM DISK = * -WITH NORECOVERY' +WITH NORECOVERY* +REVERT' } #endregion Invoke Query Mock @@ -454,7 +465,6 @@ WITH NORECOVERY' Mock -CommandName Join-Path -MockWith { [IO.Path]::Combine($databaseMembershipClass.BackupPath,"$($database.Name)_Log_$(Get-Date -Format 'yyyyMMddhhmmss').trn") } -Verifiable -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Mock -CommandName New-TerminatingError { $ErrorType } -Verifiable Mock -CommandName Remove-Item -Verifiable - Mock -CommandName Restore-SqlDatabase -Verifiable } BeforeEach { @@ -501,13 +511,12 @@ WITH NORECOVERY' Assert-MockCalled -CommandName Import-SQLPSModule -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter { $Query -like 'EXEC master.dbo.xp_fileexist *' } Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Full_*.bak' } Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -530,13 +539,12 @@ WITH NORECOVERY' Assert-MockCalled -CommandName Import-SQLPSModule -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter { $Query -like 'EXEC master.dbo.xp_fileexist *' } Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Full_*.bak' } Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -565,7 +573,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 0 -Exactly } @@ -587,21 +594,20 @@ WITH NORECOVERY' Assert-MockCalled -CommandName Get-PrimaryReplicaServerObject -Scope It -Times 0 -Exactly -ParameterFilter { $AvailabilityGroup.PrimaryReplicaServerName -eq 'Server2' } Assert-MockCalled -CommandName Import-SQLPSModule -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter { $Query -like 'EXEC master.dbo.xp_fileexist *' } - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Full_*.bak' } Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 0 -Exactly } It 'Should throw the correct error when "MatchDatabaseOwner" is $true and the current login does not have impersonate permissions' { Mock -CommandName Test-ImpersonatePermissions -MockWith { $false } -Verifiable - { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw "The login '$([System.Security.Principal.WindowsIdentity]::GetCurrent().Name)' is missing impersonate permissions in the instances 'Server2'." + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw "The login '$([System.Security.Principal.WindowsIdentity]::GetCurrent().Name)' is missing impersonate any login, control server, impersonate login, or control login permissions in the instances 'Server2'." Assert-MockCalled -CommandName Add-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly -ParameterFilter { $InputObject.PrimaryReplicaServerName -eq 'Server1' -and $InputObject.LocalReplicaRole -eq 'Primary' } Assert-MockCalled -CommandName Add-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly -ParameterFilter { $InputObject.PrimaryReplicaServerName -eq 'Server1' -and $InputObject.LocalReplicaRole -eq 'Secondary' } @@ -623,7 +629,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -676,8 +681,7 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly $mockServerObject.Databases['DB1'].($prerequisiteCheck.Key) = $originalValue } @@ -708,7 +712,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -746,8 +749,7 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly $mockServerObject.Databases['DB1'].($filestreamProperty.Key) = $originalValue } @@ -779,7 +781,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly $mockServerObject.Databases['DB1'].ContainmentType = $originalValue @@ -812,7 +813,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly $mockServer2Object.Databases['DB1'].FileGroups.Files.FileName = $originalValue @@ -845,7 +845,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly $mockServer2Object.Databases['DB1'].LogFiles.FileName = $originalValue @@ -878,7 +877,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly $mockServerObject.Databases['DB1'].EncryptionEnabled = $false @@ -905,13 +903,12 @@ WITH NORECOVERY' Assert-MockCalled -CommandName Import-SQLPSModule -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter { $Query -like 'EXEC master.dbo.xp_fileexist *' } Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Full_*.bak' } Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -940,7 +937,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -969,7 +965,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -998,7 +993,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } @@ -1021,13 +1015,12 @@ WITH NORECOVERY' Assert-MockCalled -CommandName Import-SQLPSModule -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter { $Query -like 'EXEC master.dbo.xp_fileexist *' } Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Full_*.bak' } Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } } @@ -1060,7 +1053,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 2 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 0 -Exactly } @@ -1089,7 +1081,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 2 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 0 -Exactly } @@ -1118,7 +1109,6 @@ WITH NORECOVERY' Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 2 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 0 -Exactly } } @@ -1146,13 +1136,12 @@ WITH NORECOVERY' Assert-MockCalled -CommandName Import-SQLPSModule -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter { $Query -like 'EXEC master.dbo.xp_fileexist *' } Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabase - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 2 -Exactly -ParameterFilter $mockInvokeQueryParameterRestoreDatabaseWithExecuteAs Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Full_*.bak' } Assert-MockCalled -CommandName Join-Path -Scope It -Times 1 -Exactly -ParameterFilter { $ChildPath -like '*_Log_*.trn' } Assert-MockCalled -CommandName New-TerminatingError -Scope It -Times 0 -Exactly Assert-MockCalled -CommandName Remove-Item -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Remove-SqlAvailabilityDatabase -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Restore-SqlDatabase -Scope It -Times 1 -Exactly Assert-MockCalled -CommandName Test-ImpersonatePermissions -Scope It -Times 1 -Exactly } } diff --git a/Tests/Unit/SqlServerDSCHelper.Tests.ps1 b/Tests/Unit/SqlServerDSCHelper.Tests.ps1 index 31aaafb39..451d7e6ff 100644 --- a/Tests/Unit/SqlServerDSCHelper.Tests.ps1 +++ b/Tests/Unit/SqlServerDSCHelper.Tests.ps1 @@ -681,68 +681,111 @@ InModuleScope $script:helperModuleName { Describe "Testing Test-LoginEffectivePermissions" { - $mockAllPermissionsPresent = @( + $mockAllServerPermissionsPresent = @( 'Connect SQL', 'Alter Any Availability Group', 'View Server State' ) - $mockPermissionsMissing = @( + $mockServerPermissionsMissing = @( 'Connect SQL', 'View Server State' ) - $mockInvokeQueryClusterServicePermissionsSet = @() # Will be set dynamically in the check + $mockAllLoginPermissionsPresent = @( + 'View Definition', + 'Impersonate' + ) + + $mockLoginPermissionsMissing = @( + 'View Definition' + ) - $mockInvokeQueryClusterServicePermissionsResult = { + $mockInvokeQueryPermissionsSet = @() # Will be set dynamically in the check + + $mockInvokeQueryPermissionsResult = { return New-Object -TypeName PSObject -Property @{ Tables = @{ Rows = @{ - permission_name = $mockInvokeQueryClusterServicePermissionsSet + permission_name = $mockInvokeQueryPermissionsSet } } } } - $testLoginEffectivePermissionsParams = @{ + $testLoginEffectiveServerPermissionsParams = @{ SQLServer = 'Server1' SQLInstanceName = 'MSSQLSERVER' Login = 'NT SERVICE\ClusSvc' Permissions = @() } + $testLoginEffectiveLoginPermissionsParams = @{ + SQLServer = 'Server1' + SQLInstanceName = 'MSSQLSERVER' + Login = 'NT SERVICE\ClusSvc' + Permissions = @() + SecurableClass = 'LOGIN' + SecurableName = 'Login1' + } + BeforeEach { - Mock -CommandName Invoke-Query -MockWith $mockInvokeQueryClusterServicePermissionsResult -Verifiable + Mock -CommandName Invoke-Query -MockWith $mockInvokeQueryPermissionsResult -Verifiable } Context 'When all of the permissions are present' { + It 'Should return $true when the desired server permissions are present' { + $mockInvokeQueryPermissionsSet = $mockAllServerPermissionsPresent.Clone() + $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() + + Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $true + + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } - It 'Should return $true when the desired permissions are present' { - $mockInvokeQueryClusterServicePermissionsSet = $mockAllPermissionsPresent.Clone() - $testLoginEffectivePermissionsParams.Permissions = $mockAllPermissionsPresent.Clone() + It 'Should return $true when the desired login permissions are present' { + $mockInvokeQueryPermissionsSet = $mockAllLoginPermissionsPresent.Clone() + $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() - Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams | Should -Be $true + Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $true - Assert-MockCalled -CommandName Invoke-Query -Times 1 -Exactly + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } } Context 'When a permission is missing' { + It 'Should return $false when the desired server permissions are not present' { + $mockInvokeQueryPermissionsSet = $mockServerPermissionsMissing.Clone() + $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() - It 'Should return $false when the desired permissions are not present' { - $mockInvokeQueryClusterServicePermissionsSet = $mockPermissionsMissing.Clone() - $testLoginEffectivePermissionsParams.Permissions = $mockAllPermissionsPresent.Clone() + Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $false - Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams | Should -Be $false + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } + + It 'Should return $false when the specified login has no server permissions assigned' { + $mockInvokeQueryPermissionsSet = @() + $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() + + Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $false + + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } + + It 'Should return $false when the desired login permissions are not present' { + $mockInvokeQueryPermissionsSet = $mockLoginPermissionsMissing.Clone() + $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() + + Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $false Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } - It 'Should return $false when the specified login has no permissions assigned' { - $mockInvokeQueryClusterServicePermissionsSet = @() - $testLoginEffectivePermissionsParams.Permissions = $mockAllPermissionsPresent.Clone() + It 'Should return $false when the specified login has no login permissions assigned' { + $mockInvokeQueryPermissionsSet = @() + $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() - Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams | Should -Be $false + Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $false Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } @@ -1229,8 +1272,24 @@ InModuleScope $script:helperModuleName { } } + $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter = { + $Permissions -eq @('IMPERSONATE ANY LOGIN') + } + + $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter = { + $Permissions -eq @('CONTROL SERVER') + } + + $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter = { + $Permissions -eq @('IMPERSONATE') + } + + $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter = { + $Permissions -eq @('CONTROL') + } + Describe 'Testing Test-ImpersonatePermissions' { - $mockConnectionContextObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.ConnectionContext + $mockConnectionContextObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.ServerConnection $mockConnectionContextObject.TrueLogin = 'Login1' $mockServerObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.Server @@ -1238,23 +1297,60 @@ InModuleScope $script:helperModuleName { $mockServerObject.ServiceName = 'MSSQLSERVER' $mockServerObject.ConnectionContext = $mockConnectionContextObject + BeforeEach { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -MockWith { $false } -Verifiable + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -MockWith { $false } -Verifiable + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -MockWith { $false } -Verifiable + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -MockWith { $false } -Verifiable + } + Context 'When impersonate permissions are present for the login' { - Mock -CommandName Test-LoginEffectivePermissions -MockWith { $true } + It 'Should return true when the impersonate any login permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $true + + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly + } - It 'Should return true when the impersonate permissions are present for the login'{ + It 'Should return true when the control server permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -MockWith { $true } -Verifiable Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $true - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly + } + + It 'Should return true when the impersonate login permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $true + + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 1 -Exactly + } + + It 'Should return true when the control login permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $true + + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 1 -Exactly } } Context 'When impersonate permissions are missing for the login' { - Mock -CommandName Test-LoginEffectivePermissions -MockWith { $false } -Verifiable - - It 'Should return false when the impersonate permissions are missing for the login'{ + It 'Should return false when the server permissions are missing for the login' { Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $false - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 0 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 0 -Exactly + } + + It 'Should return false when the login permissions are missing for the login' { + Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $false + + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 1 -Exactly } } } @@ -1703,10 +1799,29 @@ InModuleScope $script:helperModuleName { Describe 'Testing New-TerminatingError' -Tag NewWarningMessage { Context -Name 'When building a localized error message' -Fixture { It 'Should return the correct error record with the correct error message' { - $errorRecord = New-TerminatingError -ErrorType 'NoKeyFound' -FormatArgs 'Dummy error' $errorRecord.Exception.Message | Should -Be 'No Localization key found for ErrorType: ''Dummy error''.' } + + It 'Should return the correct error record with the correct error message including InnerException' { + $errorRecord = New-TerminatingError -ErrorType 'NoKeyFound' -FormatArgs 'Dummy error' -InnerException 'Dummy exception' + $errorRecord.Exception.Message | Should -Be 'No Localization key found for ErrorType: ''Dummy error''. InnerException: Dummy exception' + } + + It 'Should return the correct error record with a matching FullyQualifiedErrorId' { + $errorRecord = New-TerminatingError -ErrorType 'NoKeyFound' -FormatArgs 'Dummy error' + $errorRecord.FullyQualifiedErrorId | Should -Be 'SqlServerDSCHelper.NoKeyFound' + } + + It 'Should return the correct error record with a matching FullyQualifiedErrorId when there is no calling module' { + Mock -CommandName Get-PSCallStack -MockWith { + , [PSCustomObject] @{ + ScriptName = '' + } + } + $errorRecord = New-TerminatingError -ErrorType 'NoKeyFound' -FormatArgs 'Dummy error' + $errorRecord.FullyQualifiedErrorId | Should -Be 'NoKeyFound' + } } Context -Name 'When building a localized error message that does not exists' -Fixture { @@ -1714,6 +1829,16 @@ InModuleScope $script:helperModuleName { $errorRecord = New-TerminatingError -ErrorType 'UnknownDummyMessage' -FormatArgs 'Dummy error' $errorRecord.Exception.Message | Should -Be 'No Localization key found for ErrorType: ''UnknownDummyMessage''.' } + + It 'Should return the correct error record with the correct error message even if the NoKeyFound message is missing' { + $noKeyFound = $script:localizedData.NoKeyFound + $script:localizedData.Remove('NoKeyFound') + + $errorRecord = New-TerminatingError -ErrorType 'UnknownDummyMessage' -FormatArgs 'Dummy error' + $errorRecord.Exception.Message | Should -Be 'No Localization key found for ErrorType: ''UnknownDummyMessage''.' + + $script:localizedData.NoKeyFound = $noKeyFound + } } Assert-VerifiableMock diff --git a/Tests/Unit/Stubs/SMO.cs b/Tests/Unit/Stubs/SMO.cs index 9d507d8e3..12744e18a 100644 --- a/Tests/Unit/Stubs/SMO.cs +++ b/Tests/Unit/Stubs/SMO.cs @@ -238,7 +238,7 @@ public class Server public string MockGranteeName; public AvailabilityGroupCollection AvailabilityGroups = new AvailabilityGroupCollection(); - public ConnectionContext ConnectionContext; + public ServerConnection ConnectionContext; public string ComputerNamePhysicalNetBIOS; public DatabaseCollection Databases = new DatabaseCollection(); public string DisplayName; @@ -783,7 +783,7 @@ public void Create() // TypeName: Microsoft.SqlServer.Management.Common.ServerConnection // Used by: // SqlAGDatabase - public class ConnectionContext + public class ServerConnection { public string TrueLogin; From 40e746cbffcba89237b9ada02de186b9baaf1bc3 Mon Sep 17 00:00:00 2001 From: Johan Ljunggren Date: Tue, 5 Mar 2019 17:18:33 +0100 Subject: [PATCH 06/10] SqlSetup: Join integration tests into the same script (#1296) --- ...s1 => MSFT_SqlSetup.Integration.Tests.ps1} | 82 ++- ...17.config.ps1 => MSFT_SqlSetup.config.ps1} | 62 +- ...SFT_SqlSetup_SQL2016.Integration.Tests.ps1 | 603 ------------------ .../MSFT_SqlSetup_SQL2016.config.ps1 | 442 ------------- Tests/TestHelpers/CommonTestHelper.psm1 | 35 + 5 files changed, 136 insertions(+), 1088 deletions(-) rename Tests/Integration/{MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 => MSFT_SqlSetup.Integration.Tests.ps1} (88%) rename Tests/Integration/{MSFT_SqlSetup_SQL2017.config.ps1 => MSFT_SqlSetup.config.ps1} (86%) delete mode 100644 Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 delete mode 100644 Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 diff --git a/Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 similarity index 88% rename from Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 rename to Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 index ef5da124c..eaf77e99e 100644 --- a/Tests/Integration/MSFT_SqlSetup._SQL2017.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 @@ -4,7 +4,7 @@ param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') -if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2017')) +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016','Integration_SQL2017')) { return } @@ -64,6 +64,25 @@ function Show-SqlBootstrapLog Write-Verbose -Message $('-' * 80) -Verbose } + +<# + This is used in both the configuration file and in this script file + to run the correct tests depending of what version of SQL Server is + being tested in the current job. +#> +if (Test-ContinuousIntegrationTaskCategory -Category 'Integration_SQL2017') +{ + $script:sqlVersion = '140' + $script:mockSourceMediaUrl = 'https://download.microsoft.com/download/E/F/2/EF23C21D-7860-4F05-88CE-39AA114B014B/SQLServer2017-x64-ENU.iso' +} +else +{ + $script:sqlVersion = '130' + $script:mockSourceMediaUrl = 'https://download.microsoft.com/download/9/0/7/907AD35F-9F9C-43A5-9789-52470555DB90/ENU/SQLServer2016SP1-FullSlipstream-x64-ENU.iso' +} + +Write-Verbose -Message ('Running integration tests for SQL Server version {0}' -f $script:sqlVersion) -Verbose + <# Workaround for issue #774. In the appveyor.yml file the folder C:\Program Files (x86)\Microsoft SQL Server\**\Tools\PowerShell\Modules @@ -72,7 +91,8 @@ function Show-SqlBootstrapLog here we rename back the folder to the correct name. Only the version need for our tests are renamed. #> -$sqlModulePath = Get-ChildItem -Path 'C:\Program Files (x86)\Microsoft SQL Server\140\Tools\PowerShell\*.old' +$sqlPsModulePath = 'C:\Program Files (x86)\Microsoft SQL Server\{0}\Tools\PowerShell\*.old' -f $script:sqlVersion +$sqlModulePath = Get-ChildItem -Path $sqlPsModulePath $sqlModulePath | ForEach-Object -Process { $newFolderName = (Split-Path -Path $_ -Leaf) -replace '\.old' Write-Verbose ('Renaming ''{0}'' to ''..\{1}''' -f $_, $newFolderName) -Verbose @@ -82,11 +102,9 @@ $sqlModulePath | ForEach-Object -Process { # Using try/finally to always cleanup. try { - $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName)_SQL2017.config.ps1" + $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName).config.ps1" . $configFile - $mockSourceMediaUrl = 'https://download.microsoft.com/download/E/F/2/EF23C21D-7860-4F05-88CE-39AA114B014B/SQLServer2017-x64-ENU.iso' - # Download SQL Server media if (-not (Test-Path -Path $ConfigurationData.AllNodes.ImagePath)) { @@ -96,7 +114,7 @@ try Write-Verbose -Message "Start downloading the SQL Server media at $(Get-Date -Format 'yyyy-MM-dd hh:mm:ss')" -Verbose - Invoke-WebRequest -Uri $mockSourceMediaUrl -OutFile $ConfigurationData.AllNodes.ImagePath + Invoke-WebRequest -Uri $script:mockSourceMediaUrl -OutFile $ConfigurationData.AllNodes.ImagePath Write-Verbose -Message ('SQL Server media file has SHA1 hash ''{0}''' -f (Get-FileHash -Path $ConfigurationData.AllNodes.ImagePath -Algorithm 'SHA1').Hash) -Verbose @@ -199,12 +217,12 @@ try $resourceCurrentState.AgtSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName -Leaf)) $resourceCurrentState.AgtSvcStartupType | Should -Be 'Automatic' $resourceCurrentState.ASServerMode | Should -Be $ConfigurationData.AllNodes.AnalysisServicesMultiServerMode - $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Backup") + $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Backup") $resourceCurrentState.ASCollation | Should -Be $ConfigurationData.AllNodes.Collation - $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Config") - $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Data") - $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Log") - $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Temp") + $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Config") + $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Data") + $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Log") + $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Temp") $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty $resourceCurrentState.ASSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) $resourceCurrentState.AsSvcStartupType | Should -Be 'Automatic' @@ -223,8 +241,8 @@ try $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir - $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL") - $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir + $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSQLDataDir -ChildPath "$($ConfigurationData.AllNodes.SqlServerInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL") + $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstanceDir $resourceCurrentState.InstanceID | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty @@ -237,7 +255,6 @@ try $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" - $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\Backup") $resourceCurrentState.SQLCollation | Should -Be $ConfigurationData.AllNodes.Collation $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty $resourceCurrentState.SQLSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) @@ -252,14 +269,15 @@ try 'sa' ) $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty - $resourceCurrentState.SqlTempdbFileCount | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileCount - $resourceCurrentState.SqlTempdbFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileSize - $resourceCurrentState.SqlTempdbFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileGrowth - $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty - $resourceCurrentState.SqlTempdbLogFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempdbLogFileSize - $resourceCurrentState.SqlTempdbLogFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempdbLogFileGrowth - $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\DATA\") - $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\DATA\") + $resourceCurrentState.SqlTempDbFileCount | Should -Be $ConfigurationData.AllNodes.SqlTempDbFileCount + $resourceCurrentState.SqlTempDbFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempDbFileSize + $resourceCurrentState.SqlTempDbFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempDbFileGrowth + $resourceCurrentState.SQLTempDbLogDir | Should -BeNullOrEmpty + $resourceCurrentState.SqlTempDbLogFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempDbLogFileSize + $resourceCurrentState.SqlTempDbLogFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempDbLogFileGrowth + $resourceCurrentState.SQLUserDBDir | Should -Be $ConfigurationData.AllNodes.SQLUserDBDir + $resourceCurrentState.SQLUserDBLogDir | Should -Be $ConfigurationData.AllNodes.SQLUserDBLogDir + $resourceCurrentState.SQLBackupDir | Should -Be $ConfigurationData.AllNodes.SQLBackupDir $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty @@ -366,7 +384,7 @@ try $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir - $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL") + $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.SqlServerInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL") $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir $resourceCurrentState.InstanceID | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName @@ -380,7 +398,7 @@ try $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" - $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\Backup") + $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.SqlServerInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\Backup") $resourceCurrentState.SQLCollation | Should -Be $ConfigurationData.AllNodes.Collation $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty $resourceCurrentState.SQLSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) @@ -395,12 +413,14 @@ try ) $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") - $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL14.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty + + # Regression test for issue #1287 + $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.SqlServerInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") + $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.SqlServerInstanceIdPrefix).$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") } It 'Should return $true when Test-DscConfiguration is run' { @@ -483,12 +503,12 @@ try $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty $resourceCurrentState.AgtSvcAccountUsername | Should -BeNullOrEmpty $resourceCurrentState.ASServerMode | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularServerMode - $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Backup") + $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Backup") $resourceCurrentState.ASCollation | Should -Be $ConfigurationData.AllNodes.Collation - $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Config") - $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Data") - $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Log") - $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS14.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Temp") + $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Config") + $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Data") + $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Log") + $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "$($ConfigurationData.AllNodes.AnalysisServiceInstanceIdPrefix).$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Temp") $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty $resourceCurrentState.ASSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) $resourceCurrentState.ASSysAdminAccounts | Should -Be @( diff --git a/Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 b/Tests/Integration/MSFT_SqlSetup.config.ps1 similarity index 86% rename from Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 rename to Tests/Integration/MSFT_SqlSetup.config.ps1 index 00c00ee5c..061131bba 100644 --- a/Tests/Integration/MSFT_SqlSetup_SQL2017.config.ps1 +++ b/Tests/Integration/MSFT_SqlSetup.config.ps1 @@ -17,11 +17,39 @@ else $mockLastDrive = ((Get-Volume).DriveLetter | Sort-Object | Select-Object -Last 1) $mockIsoMediaDriveLetter = [char](([int][char]$mockLastDrive) + 1) + <# + The variable $script:sqlVersion is set in the integration script file, + which is available once this script is dot-sourced. + #> + switch ($script:sqlVersion) + { + '140' + { + $versionSpecificData = @{ + SqlServerInstanceIdPrefix = 'MSSQL14' + AnalysisServiceInstanceIdPrefix = 'MSAS14' + IsoImageName = 'SQL2017.iso' + } + } + + '130' + { + $versionSpecificData = @{ + SqlServerInstanceIdPrefix = 'MSSQL13' + AnalysisServiceInstanceIdPrefix = 'MSAS13' + IsoImageName = 'SQL2016.iso' + } + } + } + $ConfigurationData = @{ AllNodes = @( @{ NodeName = 'localhost' + SqlServerInstanceIdPrefix = $versionSpecificData.SqlServerInstanceIdPrefix + AnalysisServiceInstanceIdPrefix = $versionSpecificData.AnalysisServiceInstanceIdPrefix + # Database Engine properties. DatabaseEngineNamedInstanceName = 'DSCSQLTEST' DatabaseEngineNamedInstanceFeatures = 'SQLENGINE,AS,CONN,BC,SDK' @@ -50,22 +78,27 @@ else # General SqlSetup properties Collation = 'Finnish_Swedish_CI_AS' + InstanceDir = 'C:\Program Files\Microsoft SQL Server' InstallSharedDir = 'C:\Program Files\Microsoft SQL Server' InstallSharedWOWDir = 'C:\Program Files (x86)\Microsoft SQL Server' + InstallSQLDataDir = "C:\Db\$($script:sqlVersion)\System" + SQLUserDBDir = "C:\Db\$($script:sqlVersion)\Data\" + SQLUserDBLogDir = "C:\Db\$($script:sqlVersion)\Log\" + SQLBackupDir = "C:\Db\$($script:sqlVersion)\Backup" UpdateEnabled = 'False' SuppressReboot = $true # Make sure we don't reboot during testing. ForceReboot = $false # Properties for mounting media - ImagePath = "$env:TEMP\SQL2017.iso" + ImagePath = Join-Path -Path $env:TEMP -ChildPath $versionSpecificData.IsoImageName DriveLetter = $mockIsoMediaDriveLetter - # Parameters to configure Tempdb - SqlTempdbFileCount = '2' - SqlTempdbFileSize = '128' - SqlTempdbFileGrowth = '128' - SqlTempdbLogFileSize = '128' - SqlTempdbLogFileGrowth = '128' + # Parameters to configure TempDb + SqlTempDbFileCount = '2' + SqlTempDbFileSize = '128' + SqlTempDbFileGrowth = '128' + SqlTempDbLogFileSize = '128' + SqlTempDbLogFileGrowth = '128' SqlInstallAccountUserName = "$env:COMPUTERNAME\SqlInstall" SqlInstallAccountPassword = 'P@ssw0rd1' @@ -235,16 +268,21 @@ Configuration MSFT_SqlSetup_InstallDatabaseEngineNamedInstanceAsSystem_Config AsSvcStartupType = 'Automatic' ASCollation = $Node.Collation ASSvcAccount = $SqlServicePrimaryCredential + InstanceDir = $Node.InstanceDir InstallSharedDir = $Node.InstallSharedDir InstallSharedWOWDir = $Node.InstallSharedWOWDir + InstallSQLDataDir = $Node.InstallSQLDataDir + SQLUserDBDir = $Node.SQLUserDBDir + SQLUserDBLogDir = $Node.SQLUserDBLogDir + SQLBackupDir = $Node.SQLBackupDir UpdateEnabled = $Node.UpdateEnabled SuppressReboot = $Node.SuppressReboot ForceReboot = $Node.ForceReboot - SqlTempdbFileCount = $Node.SqlTempdbFileCount - SqlTempdbFileSize = $Node.SqlTempdbFileSize - SqlTempdbFileGrowth = $Node.SqlTempdbFileGrowth - SqlTempdbLogFileSize = $Node.SqlTempdbLogFileSize - SqlTempdbLogFileGrowth = $Node.SqlTempdbLogFileGrowth + SqlTempDbFileCount = $Node.SqlTempDbFileCount + SqlTempDbFileSize = $Node.SqlTempDbFileSize + SqlTempDbFileGrowth = $Node.SqlTempDbFileGrowth + SqlTempDbLogFileSize = $Node.SqlTempDbLogFileSize + SqlTempDbLogFileGrowth = $Node.SqlTempDbLogFileGrowth # This must be set if using SYSTEM account to install. SQLSysAdminAccounts = @( diff --git a/Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 deleted file mode 100644 index f0cedc873..000000000 --- a/Tests/Integration/MSFT_SqlSetup_SQL2016.Integration.Tests.ps1 +++ /dev/null @@ -1,603 +0,0 @@ -# This is used to make sure the integration test run in the correct order. -[Microsoft.DscResourceKit.IntegrationTest(OrderNumber = 1)] -param() - -Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') - -if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2016')) -{ - return -} - -$script:dscModuleName = 'SqlServerDsc' -$script:dscResourceFriendlyName = 'SqlSetup' -$script:dscResourceName = "MSFT_$($script:dscResourceFriendlyName)" - -#region HEADER -# Integration Test Template Version: 1.3.2 -[String] $script:moduleRoot = Split-Path -Parent (Split-Path -Parent $PSScriptRoot) -if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests'))) -or ` - (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1'))) ) -{ - & git @('clone', 'https://github.com/PowerShell/DscResource.Tests.git', (Join-Path -Path $script:moduleRoot -ChildPath 'DscResource.Tests')) -} - -Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'DSCResource.Tests' -ChildPath 'TestHelper.psm1')) -Force -$TestEnvironment = Initialize-TestEnvironment ` - -DSCModuleName $script:dscModuleName ` - -DSCResourceName $script:dscResourceName ` - -TestType Integration -#endregion - -<# - .SYNOPSIS - This function will output the Setup Bootstrap Summary.txt log file. - - .DESCRIPTION - This function will output the Summary.txt log file, this is to be - able to debug any problems that potentially occurred during setup. - This will pick up the newest Summary.txt log file, so any - other log files will be ignored (AppVeyor build worker has - SQL Server instances installed by default). - This code is meant to work regardless what SQL Server - major version is used for the integration test. -#> -function Show-SqlBootstrapLog -{ - [CmdletBinding()] - param - ( - ) - - $summaryLogPath = Get-ChildItem -Path 'C:\Program Files\Microsoft SQL Server\**\Setup Bootstrap\Log\Summary.txt' | - Sort-Object -Property LastWriteTime -Descending | - Select-Object -First 1 - - $summaryLog = Get-Content $summaryLogPath - - Write-Verbose -Message $('-' * 80) -Verbose - Write-Verbose -Message 'Summary.txt' -Verbose - Write-Verbose -Message $('-' * 80) -Verbose - $summaryLog | ForEach-Object { - Write-Verbose $_ -Verbose - } - Write-Verbose -Message $('-' * 80) -Verbose -} - -<# - Workaround for issue #774. In the appveyor.yml file the folder - C:\Program Files (x86)\Microsoft SQL Server\**\Tools\PowerShell\Modules - was renamed to - C:\Program Files (x86)\Microsoft SQL Server\**\Tools\PowerShell\Modules.old - here we rename back the folder to the correct name. Only the version need - for our tests are renamed. -#> -$sqlModulePath = Get-ChildItem -Path 'C:\Program Files (x86)\Microsoft SQL Server\130\Tools\PowerShell\*.old' -$sqlModulePath | ForEach-Object -Process { - $newFolderName = (Split-Path -Path $_ -Leaf) -replace '\.old' - Write-Verbose ('Renaming ''{0}'' to ''..\{1}''' -f $_, $newFolderName) -Verbose - Rename-Item $_ -NewName $newFolderName -Force -} - -# Using try/finally to always cleanup. -try -{ - $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName)_SQL2016.config.ps1" - . $configFile - - $mockSourceMediaUrl = 'https://download.microsoft.com/download/9/0/7/907AD35F-9F9C-43A5-9789-52470555DB90/ENU/SQLServer2016SP1-FullSlipstream-x64-ENU.iso' - - # Download SQL Server media - if (-not (Test-Path -Path $ConfigurationData.AllNodes.ImagePath)) - { - # By switching to 'SilentlyContinue' should theoretically increase the download speed. - $previousProgressPreference = $ProgressPreference - $ProgressPreference = 'SilentlyContinue' - - Write-Verbose -Message "Start downloading the SQL Server media at $(Get-Date -Format 'yyyy-MM-dd hh:mm:ss')" -Verbose - - Invoke-WebRequest -Uri $mockSourceMediaUrl -OutFile $ConfigurationData.AllNodes.ImagePath - - Write-Verbose -Message ('SQL Server media file has SHA1 hash ''{0}''' -f (Get-FileHash -Path $ConfigurationData.AllNodes.ImagePath -Algorithm 'SHA1').Hash) -Verbose - - $ProgressPreference = $previousProgressPreference - - # Double check that the SQL media was downloaded. - if (-not (Test-Path -Path $ConfigurationData.AllNodes.ImagePath)) - { - Write-Warning -Message ('SQL media could not be downloaded, can not run the integration test.') - return - } - else - { - Write-Verbose -Message "Finished downloading the SQL Server media iso at $(Get-Date -Format 'yyyy-MM-dd hh:mm:ss')" -Verbose - } - } - else - { - Write-Verbose -Message 'SQL Server media is already downloaded' -Verbose - } - - Describe "$($script:dscResourceName)_Integration" { - BeforeAll { - $resourceId = "[$($script:dscResourceFriendlyName)]Integration_Test" - } - - $configurationName = "$($script:dscResourceName)_CreateDependencies_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } - } - - $configurationName = "$($script:dscResourceName)_InstallDatabaseEngineNamedInstanceAsSystem_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } -ErrorVariable itBlockError - - # Check if previous It-block failed. If so output the SQL Server setup log file. - if ( $itBlockError.Count -ne 0 ) - { - Show-SqlBootstrapLog - } - - It 'Should be able to call Get-DscConfiguration without throwing' { - { - $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop - } | Should -Not -Throw - } - - It 'Should have set the resource and all the parameters should match' { - $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { - $_.ConfigurationName -eq $configurationName ` - -and $_.ResourceId -eq $resourceId - } - - $resourceCurrentState.Action | Should -BeNullOrEmpty - $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.AgtSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName -Leaf)) - $resourceCurrentState.AgtSvcStartupType | Should -Be 'Automatic' - $resourceCurrentState.ASServerMode | Should -Be $ConfigurationData.AllNodes.AnalysisServicesMultiServerMode - $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Backup") - $resourceCurrentState.ASCollation | Should -Be $ConfigurationData.AllNodes.Collation - $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Config") - $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Data") - $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Log") - $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\OLAP\Temp") - $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.ASSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) - $resourceCurrentState.AsSvcStartupType | Should -Be 'Automatic' - $resourceCurrentState.ASSysAdminAccounts | Should -Be @( - $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, - "NT SERVICE\SSASTELEMETRY`$$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)" - ) - $resourceCurrentState.BrowserSvcStartupType | Should -BeNullOrEmpty - $resourceCurrentState.ErrorReporting | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterGroupName | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterIPAddress | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterNetworkName | Should -BeNullOrEmpty - $resourceCurrentState.Features | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceFeatures - $resourceCurrentState.ForceReboot | Should -BeNullOrEmpty - $resourceCurrentState.FTSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir - $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir - $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL") - $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir - $resourceCurrentState.InstanceID | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName - $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName - $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.ISSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.ProductKey | Should -BeNullOrEmpty - $resourceCurrentState.RSSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.RSSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.SAPwd | Should -BeNullOrEmpty - $resourceCurrentState.SecurityMode | Should -Be 'SQL' - $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty - $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty - $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" - $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\Backup") - $resourceCurrentState.SQLCollation | Should -Be $ConfigurationData.AllNodes.Collation - $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.SQLSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) - $resourceCurrentState.SqlSvcStartupType | Should -Be 'Automatic' - $resourceCurrentState.SQLSysAdminAccounts | Should -Be @( - $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, - $ConfigurationData.AllNodes.SqlInstallAccountUserName, - "NT SERVICE\MSSQL`$$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)", - "NT SERVICE\SQLAgent`$$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)", - 'NT SERVICE\SQLWriter', - 'NT SERVICE\Winmgmt', - 'sa' - ) - $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty - $resourceCurrentState.SqlTempdbFileCount | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileCount - $resourceCurrentState.SqlTempdbFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileSize - $resourceCurrentState.SqlTempdbFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempdbFileGrowth - $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty - $resourceCurrentState.SqlTempdbLogFileSize | Should -Be $ConfigurationData.AllNodes.SqlTempdbLogFileSize - $resourceCurrentState.SqlTempdbLogFileGrowth | Should -Be $ConfigurationData.AllNodes.SqlTempdbLogFileGrowth - $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\DATA\") - $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineNamedInstanceName)\MSSQL\DATA\") - $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty - $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty - $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty - $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty - } - - It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true - } - } - - $configurationName = "$($script:dscResourceName)_StopServicesInstance_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } - } - - $configurationName = "$($script:dscResourceName)_InstallDatabaseEngineDefaultInstanceAsUser_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } -ErrorVariable itBlockError - - # Check if previous It-block failed. If so output the SQL Server setup log file. - if ( $itBlockError.Count -ne 0 ) - { - Show-SqlBootstrapLog - } - - It 'Should be able to call Get-DscConfiguration without throwing' { - { - $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop - } | Should -Not -Throw - } - - It 'Should have set the resource and all the parameters should match' { - $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { - $_.ConfigurationName -eq $configurationName ` - -and $_.ResourceId -eq $resourceId - } - - $resourceCurrentState.Action | Should -BeNullOrEmpty - $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.AgtSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName -Leaf)) - $resourceCurrentState.ASServerMode | Should -BeNullOrEmpty - $resourceCurrentState.ASBackupDir | Should -BeNullOrEmpty - $resourceCurrentState.ASCollation | Should -BeNullOrEmpty - $resourceCurrentState.ASConfigDir | Should -BeNullOrEmpty - $resourceCurrentState.ASDataDir | Should -BeNullOrEmpty - $resourceCurrentState.ASLogDir | Should -BeNullOrEmpty - $resourceCurrentState.ASTempDir | Should -BeNullOrEmpty - $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.ASSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.ASSysAdminAccounts | Should -BeNullOrEmpty - $resourceCurrentState.BrowserSvcStartupType | Should -BeNullOrEmpty - $resourceCurrentState.ErrorReporting | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterGroupName | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterIPAddress | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterNetworkName | Should -BeNullOrEmpty - $resourceCurrentState.Features | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceFeatures - $resourceCurrentState.ForceReboot | Should -BeNullOrEmpty - $resourceCurrentState.FTSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir - $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir - $resourceCurrentState.InstallSQLDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL") - $resourceCurrentState.InstanceDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir - $resourceCurrentState.InstanceID | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName - $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName - $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.ISSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.ProductKey | Should -BeNullOrEmpty - $resourceCurrentState.RSSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.RSSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.SAPwd | Should -BeNullOrEmpty - $resourceCurrentState.SecurityMode | Should -Be 'Windows' - $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty - $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty - $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" - $resourceCurrentState.SQLBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\Backup") - $resourceCurrentState.SQLCollation | Should -Be $ConfigurationData.AllNodes.Collation - $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.SQLSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) - $resourceCurrentState.SQLSysAdminAccounts | Should -Be @( - $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, - $ConfigurationData.AllNodes.SqlInstallAccountUserName, - "NT SERVICE\$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)", - "NT SERVICE\SQLSERVERAGENT", - 'NT SERVICE\SQLWriter', - 'NT SERVICE\Winmgmt', - 'sa' - ) - $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLUserDBDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") - $resourceCurrentState.SQLUserDBLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSSQL13.$($ConfigurationData.AllNodes.DatabaseEngineDefaultInstanceName)\MSSQL\DATA\") - $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty - $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty - $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty - $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty - } - - It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true - } - } - - $configurationName = "$($script:dscResourceName)_StopSqlServerDefaultInstance_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } - } - - $configurationName = "$($script:dscResourceName)_InstallTabularAnalysisServicesAsSystem_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } -ErrorVariable itBlockError - - # Check if previous It-block failed. If so output the SQL Server setup log file. - if ( $itBlockError.Count -ne 0 ) - { - Show-SqlBootstrapLog - } - - It 'Should be able to call Get-DscConfiguration without throwing' { - { - $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop - } | Should -Not -Throw - } - - It 'Should have set the resource and all the parameters should match' { - $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { - $_.ConfigurationName -eq $configurationName ` - -and $_.ResourceId -eq $resourceId - } - - $resourceCurrentState.Action | Should -BeNullOrEmpty - $resourceCurrentState.AgtSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.AgtSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.ASServerMode | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularServerMode - $resourceCurrentState.ASBackupDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Backup") - $resourceCurrentState.ASCollation | Should -Be $ConfigurationData.AllNodes.Collation - $resourceCurrentState.ASConfigDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Config") - $resourceCurrentState.ASDataDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Data") - $resourceCurrentState.ASLogDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Log") - $resourceCurrentState.ASTempDir | Should -Be (Join-Path -Path $ConfigurationData.AllNodes.InstallSharedDir -ChildPath "MSAS13.$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)\OLAP\Temp") - $resourceCurrentState.ASSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.ASSvcAccountUsername | Should -Be ('.\{0}' -f (Split-Path -Path $ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName -Leaf)) - $resourceCurrentState.ASSysAdminAccounts | Should -Be @( - $ConfigurationData.AllNodes.SqlAdministratorAccountUserName, - "NT SERVICE\SSASTELEMETRY`$$($ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName)" - ) - $resourceCurrentState.BrowserSvcStartupType | Should -BeNullOrEmpty - $resourceCurrentState.ErrorReporting | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterGroupName | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterIPAddress | Should -BeNullOrEmpty - $resourceCurrentState.FailoverClusterNetworkName | Should -BeNullOrEmpty - $resourceCurrentState.Features | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularFeatures - $resourceCurrentState.ForceReboot | Should -BeNullOrEmpty - $resourceCurrentState.FTSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.FTSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.InstallSharedDir | Should -Be $ConfigurationData.AllNodes.InstallSharedDir - $resourceCurrentState.InstallSharedWOWDir | Should -Be $ConfigurationData.AllNodes.InstallSharedWOWDir - $resourceCurrentState.InstallSQLDataDir | Should -BeNullOrEmpty - $resourceCurrentState.InstanceDir | Should -BeNullOrEmpty - $resourceCurrentState.InstanceID | Should -BeNullOrEmpty - $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.AnalysisServicesTabularInstanceName - $resourceCurrentState.ISSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.ISSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.ProductKey | Should -BeNullOrEmpty - $resourceCurrentState.RSSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.RSSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.SAPwd | Should -BeNullOrEmpty - $resourceCurrentState.SecurityMode | Should -BeNullOrEmpty - $resourceCurrentState.SetupProcessTimeout | Should -BeNullOrEmpty - $resourceCurrentState.SourceCredential | Should -BeNullOrEmpty - $resourceCurrentState.SourcePath | Should -Be "$($ConfigurationData.AllNodes.DriveLetter):\" - $resourceCurrentState.SQLBackupDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLCollation | Should -BeNullOrEmpty - $resourceCurrentState.SQLSvcAccount | Should -BeNullOrEmpty - $resourceCurrentState.SQLSvcAccountUsername | Should -BeNullOrEmpty - $resourceCurrentState.SQLSysAdminAccounts | Should -BeNullOrEmpty - $resourceCurrentState.SQLTempDBDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLTempDBLogDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLUserDBDir | Should -BeNullOrEmpty - $resourceCurrentState.SQLUserDBLogDir | Should -BeNullOrEmpty - $resourceCurrentState.SQMReporting | Should -BeNullOrEmpty - $resourceCurrentState.SuppressReboot | Should -BeNullOrEmpty - $resourceCurrentState.UpdateEnabled | Should -BeNullOrEmpty - $resourceCurrentState.UpdateSource | Should -BeNullOrEmpty - } - - It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true - } - } - - $configurationName = "$($script:dscResourceName)_StopTabularAnalysisServices_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } - } - - $configurationName = "$($script:dscResourceName)_StartServicesInstance_Config" - - Context ('When using configuration {0}' -f $configurationName) { - It 'Should compile and apply the MOF without throwing' { - { - $configurationParameters = @{ - OutputPath = $TestDrive - # The variable $ConfigurationData was dot-sourced above. - ConfigurationData = $ConfigurationData - } - - & $configurationName @configurationParameters - - $startDscConfigurationParameters = @{ - Path = $TestDrive - ComputerName = 'localhost' - Wait = $true - Verbose = $true - Force = $true - ErrorAction = 'Stop' - } - - Start-DscConfiguration @startDscConfigurationParameters - } | Should -Not -Throw - } - } - } -} -finally -{ - #region FOOTER - Restore-TestEnvironment -TestEnvironment $TestEnvironment - #endregion -} diff --git a/Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 b/Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 deleted file mode 100644 index 4eaf3560e..000000000 --- a/Tests/Integration/MSFT_SqlSetup_SQL2016.config.ps1 +++ /dev/null @@ -1,442 +0,0 @@ -#region HEADER -# Integration Test Config Template Version: 1.2.0 -#endregion - -$configFile = [System.IO.Path]::ChangeExtension($MyInvocation.MyCommand.Path, 'json') -if (Test-Path -Path $configFile) -{ - <# - Allows reading the configuration data from a JSON file, - for real testing scenarios outside of the CI. - #> - $ConfigurationData = Get-Content -Path $configFile | ConvertFrom-Json -} -else -{ - # Get a spare drive letter - $mockLastDrive = ((Get-Volume).DriveLetter | Sort-Object | Select-Object -Last 1) - $mockIsoMediaDriveLetter = [char](([int][char]$mockLastDrive) + 1) - - $ConfigurationData = @{ - AllNodes = @( - @{ - NodeName = 'localhost' - - # Database Engine properties. - DatabaseEngineNamedInstanceName = 'DSCSQLTEST' - DatabaseEngineNamedInstanceFeatures = 'SQLENGINE,AS,CONN,BC,SDK' - AnalysisServicesMultiServerMode = 'MULTIDIMENSIONAL' - - <# - Analysis Services Tabular properties. - The features CONN,BC,SDK is installed with the DSCSQLTEST so those - features will found for DSCTABULAR instance as well. - The features is added here so the same property can be used to - evaluate the result in the test. - #> - AnalysisServicesTabularInstanceName = 'DSCTABULAR' - AnalysisServicesTabularFeatures = 'AS,CONN,BC,SDK' - AnalysisServicesTabularServerMode = 'TABULAR' - - <# - Database Engine default instance properties. - The features CONN,BC,SDK is installed with the DSCSQLTEST so those - features will found for DSCTABULAR instance as well. - The features is added here so the same property can be used to - evaluate the result in the test. - #> - DatabaseEngineDefaultInstanceName = 'MSSQLSERVER' - DatabaseEngineDefaultInstanceFeatures = 'SQLENGINE,CONN,BC,SDK' - - # General SqlSetup properties - Collation = 'Finnish_Swedish_CI_AS' - InstallSharedDir = 'C:\Program Files\Microsoft SQL Server' - InstallSharedWOWDir = 'C:\Program Files (x86)\Microsoft SQL Server' - UpdateEnabled = 'False' - SuppressReboot = $true # Make sure we don't reboot during testing. - ForceReboot = $false - - # Properties for mounting media - ImagePath = "$env:TEMP\SQL2016.iso" - DriveLetter = $mockIsoMediaDriveLetter - - # Parameters to configure Tempdb - SqlTempdbFileCount = '2' - SqlTempdbFileSize = '128' - SqlTempdbFileGrowth = '128' - SqlTempdbLogFileSize = '128' - SqlTempdbLogFileGrowth = '128' - - SqlInstallAccountUserName = "$env:COMPUTERNAME\SqlInstall" - SqlInstallAccountPassword = 'P@ssw0rd1' - SqlAdministratorAccountUserName = "$env:COMPUTERNAME\SqlAdmin" - SqlAdministratorAccountPassword = 'P@ssw0rd1' - SqlServicePrimaryAccountUserName = "$env:COMPUTERNAME\svc-SqlPrimary" - SqlServicePrimaryAccountPassword = 'yig-C^Equ3' - SqlAgentServicePrimaryAccountUserName = "$env:COMPUTERNAME\svc-SqlAgentPri" - SqlAgentServicePrimaryAccountPassword = 'yig-C^Equ3' - SqlServiceSecondaryAccountUserName = "$env:COMPUTERNAME\svc-SqlSecondary" - SqlServiceSecondaryAccountPassword = 'yig-C^Equ3' - SqlAgentServiceSecondaryAccountUserName = "$env:COMPUTERNAME\svc-SqlAgentSec" - SqlAgentServiceSecondaryAccountPassword = 'yig-C^Equ3' - - CertificateFile = $env:DscPublicCertificatePath - } - ) - } -} - -<# - Creating all the credential objects to save some repeating code. -#> - -$SqlInstallCredential = New-Object ` - -TypeName System.Management.Automation.PSCredential ` - -ArgumentList @($ConfigurationData.AllNodes.SqlInstallAccountUserName, - (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlInstallAccountPassword -AsPlainText -Force)) - -$SqlAdministratorCredential = New-Object ` - -TypeName System.Management.Automation.PSCredential ` - -ArgumentList @($ConfigurationData.AllNodes.SqlAdministratorAccountUserName, - (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlAdministratorAccountPassword -AsPlainText -Force)) - -$SqlServicePrimaryCredential = New-Object ` - -TypeName System.Management.Automation.PSCredential ` - -ArgumentList @($ConfigurationData.AllNodes.SqlServicePrimaryAccountUserName, - (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlServicePrimaryAccountPassword -AsPlainText -Force)) - -$SqlAgentServicePrimaryCredential = New-Object ` - -TypeName System.Management.Automation.PSCredential ` - -ArgumentList @($ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountUserName, - (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlAgentServicePrimaryAccountPassword -AsPlainText -Force)) - -$SqlServiceSecondaryCredential = New-Object ` - -TypeName System.Management.Automation.PSCredential ` - -ArgumentList @( - $ConfigurationData.AllNodes.SqlServiceSecondaryAccountUserName, - (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlServiceSecondaryAccountPassword -AsPlainText -Force)) - -$SqlAgentServiceSecondaryCredential = New-Object ` - -TypeName System.Management.Automation.PSCredential ` - -ArgumentList @($ConfigurationData.AllNodes.SqlAgentServiceSecondaryAccountUserName, - (ConvertTo-SecureString -String $ConfigurationData.AllNodes.SqlAgentServiceSecondaryAccountPassword -AsPlainText -Force)) - -<# - .SYNOPSIS - Setting up the dependencies to test installing SQL Server instances. -#> -Configuration MSFT_SqlSetup_CreateDependencies_Config -{ - Import-DscResource -ModuleName 'PSDscResources' - Import-DscResource -ModuleName 'StorageDsc' - - node $AllNodes.NodeName - { - MountImage 'MountIsoMedia' - { - ImagePath = $Node.ImagePath - DriveLetter = $Node.DriveLetter - Ensure = 'Present' - } - - WaitForVolume WaitForMountOfIsoMedia - { - DriveLetter = $Node.DriveLetter - RetryIntervalSec = 5 - RetryCount = 10 - } - - User 'CreateSqlServicePrimaryAccount' - { - Ensure = 'Present' - UserName = Split-Path -Path $SqlServicePrimaryCredential.UserName -Leaf - Password = $SqlServicePrimaryCredential - } - - User 'CreateSqlAgentServicePrimaryAccount' - { - Ensure = 'Present' - UserName = Split-Path -Path $SqlAgentServicePrimaryCredential.UserName -Leaf - Password = $SqlAgentServicePrimaryCredential - } - - User 'CreateSqlServiceSecondaryAccount' - { - Ensure = 'Present' - UserName = Split-Path -Path $SqlServiceSecondaryCredential.UserName -Leaf - Password = $SqlServicePrimaryCredential - } - - User 'CreateSqlAgentServiceSecondaryAccount' - { - Ensure = 'Present' - UserName = Split-Path -Path $SqlAgentServiceSecondaryCredential.UserName -Leaf - Password = $SqlAgentServicePrimaryCredential - } - - User 'CreateSqlInstallAccount' - { - Ensure = 'Present' - UserName = Split-Path -Path $SqlInstallCredential.UserName -Leaf - Password = $SqlInstallCredential - } - - Group 'AddSqlInstallAsAdministrator' - { - Ensure = 'Present' - GroupName = 'Administrators' - MembersToInclude = Split-Path -Path $SqlInstallCredential.UserName -Leaf - } - - User 'CreateSqlAdminAccount' - { - Ensure = 'Present' - UserName = Split-Path -Path $SqlAdministratorCredential.UserName -Leaf - Password = $SqlAdministratorCredential - } - - WindowsFeature 'NetFramework45' - { - Name = 'NET-Framework-45-Core' - Ensure = 'Present' - } - } -} - -<# - .SYNOPSIS - Installs a named instance of Database Engine and Analysis Services. - - .NOTES - This is the instance that is used for many of the other integration tests. -#> -Configuration MSFT_SqlSetup_InstallDatabaseEngineNamedInstanceAsSystem_Config -{ - Import-DscResource -ModuleName 'SqlServerDsc' - - node $AllNodes.NodeName - { - SqlSetup 'Integration_Test' - { - InstanceName = $Node.DatabaseEngineNamedInstanceName - Features = $Node.DatabaseEngineNamedInstanceFeatures - SourcePath = "$($Node.DriveLetter):\" - SqlSvcStartupType = 'Automatic' - AgtSvcStartupType = 'Automatic' - BrowserSvcStartupType = 'Automatic' - SecurityMode = 'SQL' - SAPwd = $SqlAdministratorCredential - SQLCollation = $Node.Collation - SQLSvcAccount = $SqlServicePrimaryCredential - AgtSvcAccount = $SqlAgentServicePrimaryCredential - ASServerMode = $Node.AnalysisServicesMultiServerMode - AsSvcStartupType = 'Automatic' - ASCollation = $Node.Collation - ASSvcAccount = $SqlServicePrimaryCredential - InstallSharedDir = $Node.InstallSharedDir - InstallSharedWOWDir = $Node.InstallSharedWOWDir - UpdateEnabled = $Node.UpdateEnabled - SuppressReboot = $Node.SuppressReboot - ForceReboot = $Node.ForceReboot - SqlTempdbFileCount = $Node.SqlTempdbFileCount - SqlTempdbFileSize = $Node.SqlTempdbFileSize - SqlTempdbFileGrowth = $Node.SqlTempdbFileGrowth - SqlTempdbLogFileSize = $Node.SqlTempdbLogFileSize - SqlTempdbLogFileGrowth = $Node.SqlTempdbLogFileGrowth - - # This must be set if using SYSTEM account to install. - SQLSysAdminAccounts = @( - Split-Path -Path $SqlAdministratorCredential.UserName -Leaf - <# - Must have permission to properties IsClustered and - IsHadrEnable for SqlAlwaysOnService. - #> - Split-Path -Path $SqlInstallCredential.UserName -Leaf - ) - - # This must be set if using SYSTEM account to install. - ASSysAdminAccounts = @( - Split-Path -Path $SqlAdministratorCredential.UserName -Leaf - ) - } - } -} - -<# - .SYNOPSIS - Stopping the named instance to save memory on the build worker. - - .NOTES - The named instance is restarted at the end of the tests. -#> -Configuration MSFT_SqlSetup_StopServicesInstance_Config -{ - Import-DscResource -ModuleName 'PSDscResources' - - node $AllNodes.NodeName - { - <# - Stopping the SQL Server Agent service for the named instance. - It will be restarted at the end of the tests. - #> - Service ('StopSqlServerAgentForInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) - { - Name = ('SQLAGENT${0}' -f $Node.DatabaseEngineNamedInstanceName) - State = 'Stopped' - } - - <# - Stopping the Database Engine named instance. It will be restarted - at the end of the tests. - #> - Service ('StopSqlServerInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) - { - Name = ('MSSQL${0}' -f $Node.DatabaseEngineNamedInstanceName) - State = 'Stopped' - } - - Service ('StopMultiAnalysisServicesInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) - { - Name = ('MSOLAP${0}' -f $Node.DatabaseEngineNamedInstanceName) - State = 'Stopped' - } - } -} - -<# - .SYNOPSIS - Installs a default instance of Database Engine. -#> -Configuration MSFT_SqlSetup_InstallDatabaseEngineDefaultInstanceAsUser_Config -{ - Import-DscResource -ModuleName 'SqlServerDsc' - - node $AllNodes.NodeName - { - SqlSetup 'Integration_Test' - { - InstanceName = $Node.DatabaseEngineDefaultInstanceName - Features = $Node.DatabaseEngineDefaultInstanceFeatures - SourcePath = "$($Node.DriveLetter):\" - SQLCollation = $Node.Collation - SQLSvcAccount = $SqlServicePrimaryCredential - AgtSvcAccount = $SqlAgentServicePrimaryCredential - InstallSharedDir = $Node.InstallSharedDir - InstallSharedWOWDir = $Node.InstallSharedWOWDir - UpdateEnabled = $Node.UpdateEnabled - SuppressReboot = $Node.SuppressReboot - ForceReboot = $Node.ForceReboot - SQLSysAdminAccounts = @( - Split-Path -Path $SqlAdministratorCredential.UserName -Leaf - ) - - PsDscRunAsCredential = $SqlInstallCredential - } - } -} - -<# - .SYNOPSIS - Stopping the default instance to save memory on the build worker. -#> -Configuration MSFT_SqlSetup_StopSqlServerDefaultInstance_Config -{ - Import-DscResource -ModuleName 'PSDscResources' - - node $AllNodes.NodeName - { - Service ('StopSqlServerAgentForInstance{0}' -f $Node.DatabaseEngineDefaultInstanceName) - { - Name = 'SQLSERVERAGENT' - State = 'Stopped' - } - - - Service ('StopSqlServerInstance{0}' -f $Node.DatabaseEngineDefaultInstanceName) - { - Name = $Node.DatabaseEngineDefaultInstanceName - State = 'Stopped' - } - } -} - -<# - .SYNOPSIS - Installs a named instance of Analysis Services in tabular mode. -#> -Configuration MSFT_SqlSetup_InstallTabularAnalysisServicesAsSystem_Config -{ - Import-DscResource -ModuleName 'SqlServerDsc' - - node $AllNodes.NodeName - { - SqlSetup 'Integration_Test' - { - InstanceName = $Node.AnalysisServicesTabularInstanceName - Features = $Node.AnalysisServicesTabularFeatures - SourcePath = "$($Node.DriveLetter):\" - ASServerMode = $Node.AnalysisServicesTabularServerMode - ASCollation = $Node.Collation - ASSvcAccount = $SqlServicePrimaryCredential - InstallSharedDir = $Node.InstallSharedDir - InstallSharedWOWDir = $Node.InstallSharedWOWDir - UpdateEnabled = $Node.UpdateEnabled - SuppressReboot = $Node.SuppressReboot - ForceReboot = $Node.ForceReboot - - # This must be set if using SYSTEM account to install. - ASSysAdminAccounts = @( - Split-Path -Path $SqlAdministratorCredential.UserName -Leaf - ) - } - } -} - -<# - .SYNOPSIS - Stopping the Analysis Services tabular named instance to save memory on - the build worker. -#> -Configuration MSFT_SqlSetup_StopTabularAnalysisServices_Config -{ - Import-DscResource -ModuleName 'PSDscResources' - - node $AllNodes.NodeName - { - Service ('StopTabularAnalysisServicesInstance{0}' -f $Node.AnalysisServicesTabularInstanceName) - { - Name = ('MSOLAP${0}' -f $Node.DatabaseEngineNamedInstanceName) - State = 'Stopped' - } - } -} - -<# - .SYNOPSIS - Restarting the Database Engine named instance. - - .NOTES - This is so that other integration tests are dependent on this - named instance. -#> -Configuration MSFT_SqlSetup_StartServicesInstance_Config -{ - Import-DscResource -ModuleName 'PSDscResources' - - node $AllNodes.NodeName - { - # Start the Database Engine named instance. - Service ('StartSqlServerInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) - { - Name = ('MSSQL${0}' -f $Node.DatabaseEngineNamedInstanceName) - State = 'Running' - } - - # Starting the SQL Server Agent service for the named instance. - Service ('StartSqlServerAgentForInstance{0}' -f $Node.DatabaseEngineNamedInstanceName) - { - Name = ('SQLAGENT${0}' -f $Node.DatabaseEngineNamedInstanceName) - State = 'Running' - } - } -} diff --git a/Tests/TestHelpers/CommonTestHelper.psm1 b/Tests/TestHelpers/CommonTestHelper.psm1 index f81ac7514..6eee6a16a 100644 --- a/Tests/TestHelpers/CommonTestHelper.psm1 +++ b/Tests/TestHelpers/CommonTestHelper.psm1 @@ -299,6 +299,11 @@ function New-SQLSelfSignedCertificate .PARAMETER Type Type of tests in the test file. Can be set to Unit or Integration. + + .PARAMETER Category + Optional. One or more categories to check if they are set in + $env:CONFIGURATION. If this are not set, the parameter Type + is used as category. #> function Test-SkipContinuousIntegrationTask { @@ -343,3 +348,33 @@ function Test-SkipContinuousIntegrationTask return $result } + +<# + .SYNOPSIS + Returns $true if the the environment variable APPVEYOR is set to $true, + and the environment variable CONFIGURATION is set to the value passed + in the parameter Type. + + .PARAMETER Category + One or more categories to check if they are set in $env:CONFIGURATION. +#> +function Test-ContinuousIntegrationTaskCategory +{ + [OutputType([System.Boolean])] + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [System.String[]] + $Category + ) + + $result = $false + + if ($env:APPVEYOR -eq $true -and $env:CONFIGURATION -in $Category) + { + $result = $true + } + + return $result +} From d36a171abc7b4ab6cba1f49c47742d7a496e9eae Mon Sep 17 00:00:00 2001 From: Cody Konior Date: Sun, 10 Mar 2019 19:49:31 +0800 Subject: [PATCH 07/10] SqlAGReplica: Fix read-only routing lists and some defaults (#1302) - Changes to SqlAG - Updated documentation on the behavior of defaults as they only apply when creating a group. - Changes to SqlAGReplica - AvailabilityMode, BackupPriority, and FailoverMode defaults only apply when creating a replica not when making changes to an existing replica. Explicit parameters will still change existing replicas (issue #1244). - ReadOnlyRoutingList now gets updated without throwing an error on the first run (issue #518). --- CHANGELOG.md | 9 +++ DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 | 36 ++++++---- DSCResources/MSFT_SqlAG/MSFT_SqlAG.schema.mof | 12 ++-- .../MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 | 70 ++++++++++--------- .../MSFT_SqlAGReplica.schema.mof | 6 +- README.md | 22 +++--- Tests/Unit/Stubs/SMO.cs | 2 +- 7 files changed, 90 insertions(+), 67 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 082487141..c11d4eed3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -41,6 +41,15 @@ - New-TerminatingError error text for a missing localized message now matches the output even if the "missing localized message" localized message is also missing. +- Changes to SqlAG + - Updated documentation on the behaviour of defaults as they only apply when + creating a group. +- Changes to SqlAGReplica + - AvailabilityMode, BackupPriority, and FailoverMode defaults only apply when + creating a replica not when making changes to an existing replica. Explicit + parameters will still change existing replicas ([issue #1244](https://github.com/PowerShell/SqlServerDsc/issues/1244)). + - ReadOnlyRoutingList now gets updated without throwing an error on the first + run ([issue #518](https://github.com/PowerShell/SqlServerDsc/issues/518)). ## 12.3.0.0 diff --git a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 index f03d37eaf..f58e05efd 100644 --- a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 +++ b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 @@ -108,13 +108,13 @@ function Get-TargetResource Specifies if the availability group should be present or absent. Default is Present. .PARAMETER AutomatedBackupPreference - Specifies the automated backup preference for the availability group. + Specifies the automated backup preference for the availability group. When creating a group the default is 'None'. .PARAMETER AvailabilityMode - Specifies the replica availability mode. Default is 'AsynchronousCommit'. + Specifies the replica availability mode. When creating a group the default is 'AsynchronousCommit'. .PARAMETER BackupPriority - Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. Default is 50. + Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. When creating a group the efault is 50. .PARAMETER BasicAvailabilityGroup Specifies the type of availability group is Basic. This is only available is SQL Server 2016 and later and is ignored when applied to previous versions. @@ -132,13 +132,16 @@ function Get-TargetResource Specifies how the availability replica handles connections when in the secondary role. .PARAMETER EndpointHostName - Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name. + Specifies the hostname or IP address of the availability group replica endpoint. When creating a group the default is the instance network name. .PARAMETER FailureConditionLevel Specifies the automatic failover behavior of the availability group. + .PARAMETER FailoverMode + Specifies the failover mode. When creating a group the default is 'Manual'. + .PARAMETER HealthCheckTimeout - Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive server to be unhealthy. Default is 30,000. + Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive server to be unhealthy. When creating a group the default is 30,000. .PARAMETER ProcessOnlyOnActiveNode Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. @@ -285,14 +288,14 @@ function Set-TargetResource throw New-TerminatingError -ErrorType DatabaseMirroringEndpointNotFound -FormatArgs $ServerName, $InstanceName -ErrorCategory ObjectNotFound } - if ( -not $EndpointHostName ) - { - $EndpointHostName = $serverObject.NetName - } - # If the availability group does not exist, create it if ( -not $availabilityGroup ) { + if ( -not $EndpointHostName ) + { + $EndpointHostName = $serverObject.NetName + } + # Set up the parameters to create the AG Replica $newReplicaParams = @{ Name = $serverObject.DomainInstanceName @@ -485,13 +488,13 @@ function Set-TargetResource Specifies if the availability group should be present or absent. Default is Present. .PARAMETER AutomatedBackupPreference - Specifies the automated backup preference for the availability group. + Specifies the automated backup preference for the availability group. When creating a group the default is 'None'. .PARAMETER AvailabilityMode - Specifies the replica availability mode. Default is 'AsynchronousCommit'. + Specifies the replica availability mode. When creating a group the default is 'AsynchronousCommit'. .PARAMETER BackupPriority - Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. Default is 50. + Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. When creating a group the default is 50. .PARAMETER BasicAvailabilityGroup Specifies the type of availability group is Basic. This is only available is SQL Server 2016 and later and is ignored when applied to previous versions. @@ -509,13 +512,16 @@ function Set-TargetResource Specifies how the availability replica handles connections when in the secondary role. .PARAMETER EndpointHostName - Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name. + Specifies the hostname or IP address of the availability group replica endpoint. When creating a group the default is the instance network name. .PARAMETER FailureConditionLevel Specifies the automatic failover behavior of the availability group. + .PARAMETER FailoverMode + Specifies the failover mode. When creating a group the default is 'Manual'. + .PARAMETER HealthCheckTimeout - Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive server to be unhealthy. Default is 30,000. + Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive server to be unhealthy. When creating a group the default is 30,000. .PARAMETER ProcessOnlyOnActiveNode Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. diff --git a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.schema.mof b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.schema.mof index 60031896c..c1ff4ca25 100644 --- a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.schema.mof +++ b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.schema.mof @@ -5,18 +5,18 @@ class MSFT_SqlAG : OMI_BaseResource [Required, Description("Hostname of the SQL Server to be configured.")] String ServerName; [Key, Description("Name of the SQL instance to be configured.")] String InstanceName; [Write, Description("Specifies if the availability group should be present or absent. Default is Present."), ValueMap{"Present","Absent"}, Values{"Present","Absent"}] String Ensure; - [Write, Description("Specifies the automated backup preference for the availability group. Default is None"), ValueMap{"Primary","SecondaryOnly","Secondary","None"}, Values{"Primary","SecondaryOnly","Secondary","None"}] String AutomatedBackupPreference; - [Write, Description("Specifies the replica availability mode. Default is 'AsynchronousCommit'."), ValueMap{"AsynchronousCommit","SynchronousCommit"}, Values{"AsynchronousCommit","SynchronousCommit"}] String AvailabilityMode; - [Write, Description("Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are: integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. Default is 50.")] UInt32 BackupPriority; + [Write, Description("Specifies the automated backup preference for the availability group. When creating a group the default is None"), ValueMap{"Primary","SecondaryOnly","Secondary","None"}, Values{"Primary","SecondaryOnly","Secondary","None"}] String AutomatedBackupPreference; + [Write, Description("Specifies the replica availability mode. When creating a group the default is 'AsynchronousCommit'."), ValueMap{"AsynchronousCommit","SynchronousCommit"}, Values{"AsynchronousCommit","SynchronousCommit"}] String AvailabilityMode; + [Write, Description("Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are: integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. When creating a group the default is 50.")] UInt32 BackupPriority; [Write, Description("Specifies the type of availability group is Basic. This is only available is SQL Server 2016 and later and is ignored when applied to previous versions.")] Boolean BasicAvailabilityGroup; [Write, Description("Specifies if the option Database Level Health Detection is enabled. This is only available is SQL Server 2016 and later and is ignored when applied to previous versions.")] Boolean DatabaseHealthTrigger; [Write, Description("Specifies if the option Database DTC Support is enabled. This is only available is SQL Server 2016 and later and is ignored when applied to previous versions. This can't be altered once the Availability Group is created and is ignored if it is the case.")] Boolean DtcSupportEnabled; [Write, Description("Specifies how the availability replica handles connections when in the primary role."), ValueMap{"AllowAllConnections","AllowReadWriteConnections"}, Values{"AllowAllConnections","AllowReadWriteConnections"}] String ConnectionModeInPrimaryRole; [Write, Description("Specifies how the availability replica handles connections when in the secondary role."), ValueMap{"AllowNoConnections","AllowReadIntentConnectionsOnly","AllowAllConnections"}, Values{"AllowNoConnections","AllowReadIntentConnectionsOnly","AllowAllConnections"}] String ConnectionModeInSecondaryRole; - [Write, Description("Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name.")] String EndpointHostName; + [Write, Description("Specifies the hostname or IP address of the availability group replica endpoint. When creating a group the default is the instance network name.")] String EndpointHostName; [Write, Description("Specifies the automatic failover behavior of the availability group."), ValueMap{"OnServerDown","OnServerUnresponsive","OnCriticalServerErrors","OnModerateServerErrors","OnAnyQualifiedFailureCondition"}, Values{"OnServerDown","OnServerUnresponsive","OnCriticalServerErrors","OnModerateServerErrors","OnAnyQualifiedFailureCondition"}] String FailureConditionLevel; - [Write, Description("Specifies the failover mode. Default is 'Manual'."), ValueMap{"Automatic","Manual"}, Values{"Automatic","Manual"}] String FailoverMode; - [Write, Description("Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive server to be unhealthy. Default is 30000.")] UInt32 HealthCheckTimeout; + [Write, Description("Specifies the failover mode. When creating a group the default is 'Manual'."), ValueMap{"Automatic","Manual"}, Values{"Automatic","Manual"}] String FailoverMode; + [Write, Description("Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive server to be unhealthy. When creating a group the default is 30000.")] UInt32 HealthCheckTimeout; [Write, Description("Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance.")] Boolean ProcessOnlyOnActiveNode; [Read, Description("Gets the Endpoint URL of the availability group replica endpoint.")] String EndpointUrl; [Read, Description("Gets the port the database mirroring endpoint is listening on.")] UInt32 EndpointPort; diff --git a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 index f47eac2e5..37089e4d7 100644 --- a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 +++ b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 @@ -132,10 +132,10 @@ function Get-TargetResource Specifies if the availability group should be present or absent. Default is Present. .PARAMETER AvailabilityMode - Specifies the replica availability mode. Default is 'AsynchronousCommit'. + Specifies the replica availability mode. Default when creating a replica is 'AsynchronousCommit'. .PARAMETER BackupPriority - Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. Default is 50. + Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. When creating a replica the default is 50. .PARAMETER ConnectionModeInPrimaryRole Specifies how the availability replica handles connections when in the primary role. @@ -144,10 +144,10 @@ function Get-TargetResource Specifies how the availability replica handles connections when in the secondary role. .PARAMETER EndpointHostName - Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name which is set in the code because the value can only be determined when connected to the SQL Instance. + Specifies the hostname or IP address of the availability group replica endpoint. When creating a group the default is the instance network name which is set in the code because the value can only be determined when connected to the SQL Instance. .PARAMETER FailoverMode - Specifies the failover mode. Default is Manual. + Specifies the failover mode. When creating a replica the default is 'Manual'. .PARAMETER ReadOnlyRoutingConnectionUrl Specifies the fully-qualified domain name (FQDN) and port to use when routing to the replica for read only connections. @@ -196,12 +196,12 @@ function Set-TargetResource [Parameter()] [ValidateSet('AsynchronousCommit', 'SynchronousCommit')] [System.String] - $AvailabilityMode = 'AsynchronousCommit', + $AvailabilityMode, [Parameter()] [ValidateRange(0, 100)] [System.UInt32] - $BackupPriority = 50, + $BackupPriority, [Parameter()] [ValidateSet('AllowAllConnections', 'AllowReadWriteConnections')] @@ -220,7 +220,7 @@ function Set-TargetResource [Parameter()] [ValidateSet('Automatic', 'Manual')] [System.String] - $FailoverMode = 'Manual', + $FailoverMode, [Parameter()] [System.String] @@ -309,13 +309,16 @@ function Set-TargetResource $availabilityGroupReplica = $availabilityGroup.AvailabilityReplicas[$Name] if ( $availabilityGroupReplica ) { - if ( $AvailabilityMode -ne $availabilityGroupReplica.AvailabilityMode ) + # Get the parameters that were submitted to the function + [System.Array] $submittedParameters = $PSBoundParameters.Keys + + if ( ( $submittedParameters -contains 'AvailabilityMode' ) -and ( $AvailabilityMode -ne $availabilityGroupReplica.AvailabilityMode ) ) { $availabilityGroupReplica.AvailabilityMode = $AvailabilityMode Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica } - if ( $BackupPriority -ne $availabilityGroupReplica.BackupPriority ) + if ( ( $submittedParameters -contains 'BackupPriority' ) -and ( $BackupPriority -ne $availabilityGroupReplica.BackupPriority ) ) { $availabilityGroupReplica.BackupPriority = $BackupPriority Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica @@ -345,7 +348,7 @@ function Set-TargetResource Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica } - if ( $EndpointHostName -ne $currentEndpointHostName ) + if ( ( $submittedParameters -contains 'EndpointHostName' ) -and ( $EndpointHostName -ne $currentEndpointHostName ) ) { $newEndpointUrl = $availabilityGroupReplica.EndpointUrl.Replace($currentEndpointHostName, $EndpointHostName) $availabilityGroupReplica.EndpointUrl = $newEndpointUrl @@ -359,21 +362,25 @@ function Set-TargetResource Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica } - if ( $FailoverMode -ne $availabilityGroupReplica.FailoverMode ) + if ( ( $submittedParameters -contains 'FailoverMode' ) -and ( $FailoverMode -ne $availabilityGroupReplica.FailoverMode ) ) { $availabilityGroupReplica.FailoverMode = $FailoverMode Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica } - if ( $ReadOnlyRoutingConnectionUrl -ne $availabilityGroupReplica.ReadOnlyRoutingConnectionUrl ) + if ( ( $submittedParameters -contains 'ReadOnlyRoutingConnectionUrl' ) -and ( $ReadOnlyRoutingConnectionUrl -ne $availabilityGroupReplica.ReadOnlyRoutingConnectionUrl ) ) { $availabilityGroupReplica.ReadOnlyRoutingConnectionUrl = $ReadOnlyRoutingConnectionUrl Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica } - if ( $ReadOnlyRoutingList -ne $availabilityGroupReplica.ReadOnlyRoutingList ) + if ( ( $submittedParameters -contains 'ReadOnlyRoutingList' ) -and ( ( $ReadOnlyRoutingList -join ',' ) -ne ( $availabilityGroupReplica.ReadOnlyRoutingList -join ',' ) ) ) { - $availabilityGroupReplica.ReadOnlyRoutingList = $ReadOnlyRoutingList + $availabilityGroupReplica.ReadOnlyRoutingList.Clear() + foreach ( $readOnlyRoutingListEntry in $ReadOnlyRoutingList ) + { + $availabilityGroupReplica.ReadOnlyRoutingList.Add($readOnlyRoutingListEntry) | Out-Null + } Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityGroupReplica } } @@ -489,10 +496,10 @@ function Set-TargetResource Specifies if the availability group should be present or absent. Default is Present. .PARAMETER AvailabilityMode - Specifies the replica availability mode. Default is 'AsynchronousCommit'. + Specifies the replica availability mode. Default when creating a replica is 'AsynchronousCommit'. .PARAMETER BackupPriority - Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. Default is 50. + Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. When creating a replica the default is 50. .PARAMETER ConnectionModeInPrimaryRole Specifies how the availability replica handles connections when in the primary role. @@ -501,10 +508,10 @@ function Set-TargetResource Specifies how the availability replica handles connections when in the secondary role. .PARAMETER EndpointHostName - Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name which is set in the code because the value can only be determined when connected to the SQL Instance. + Specifies the hostname or IP address of the availability group replica endpoint. wWhen creating a group the default is the instance network name which is set in the code because the value can only be determined when connected to the SQL Instance. .PARAMETER FailoverMode - Specifies the failover mode. Default is Manual. + Specifies the failover mode. When creating a replica the default is 'Manual'. .PARAMETER ReadOnlyRoutingConnectionUrl Specifies the fully-qualified domain name (FQDN) and port to use when routing to the replica for read only connections. @@ -553,12 +560,12 @@ function Test-TargetResource [Parameter()] [ValidateSet('AsynchronousCommit', 'SynchronousCommit')] [System.String] - $AvailabilityMode = 'AsynchronousCommit', + $AvailabilityMode, [Parameter()] [ValidateRange(0, 100)] [System.UInt32] - $BackupPriority = 50, + $BackupPriority, [Parameter()] [ValidateSet('AllowAllConnections', 'AllowReadWriteConnections')] @@ -577,7 +584,7 @@ function Test-TargetResource [Parameter()] [ValidateSet('Automatic', 'Manual')] [System.String] - $FailoverMode = 'Manual', + $FailoverMode, [Parameter()] [System.String] @@ -648,25 +655,22 @@ function Test-TargetResource if ( $getTargetResourceResult.Ensure -eq 'Present' ) { - # PsBoundParameters won't work here because it doesn't account for default values - foreach ( $parameter in $MyInvocation.MyCommand.Parameters.GetEnumerator() ) + foreach ( $parameter in $PSBoundParameters.GetEnumerator() ) { $parameterName = $parameter.Key $parameterValue = Get-Variable -Name $parameterName -ErrorAction SilentlyContinue | Select-Object -ExpandProperty Value # Make sure we don't try to validate a common parameter - if ( $parametersToCheck -contains $parameterName ) + if ( $parametersToCheck -notcontains $parameterName ) { - # If the parameter is Null, a value wasn't provided - if ( -not [System.String]::IsNullOrEmpty($parameterValue) ) - { - if ( $getTargetResourceResult.($parameterName) -ne $parameterValue ) - { - New-VerboseMessage -Message "'$($parameterName)' should be '$($parameterValue)' but is '$($getTargetResourceResult.($parameterName))'" + continue + } - $result = $false - } - } + if ( $getTargetResourceResult.($parameterName) -ne $parameterValue ) + { + New-VerboseMessage -Message "'$($parameterName)' should be '$($parameterValue)' but is '$($getTargetResourceResult.($parameterName))'" + + $result = $False } } diff --git a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.schema.mof b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.schema.mof index 43d1a02b9..43b6375cb 100644 --- a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.schema.mof +++ b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.schema.mof @@ -8,12 +8,12 @@ class MSFT_SqlAGReplica : OMI_BaseResource [Write, Description("Hostname of the SQL Server where the primary replica is expected to be active. If the primary replica is not found here, the resource will attempt to find the host that holds the primary replica and connect to it.")] String PrimaryReplicaServerName; [Write, Description("Name of the SQL instance where the primary replica lives.")] String PrimaryReplicaInstanceName; [Write, Description("Specifies if the availability group replica should be present or absent. Default is Present."), ValueMap{"Present","Absent"}, Values{"Present","Absent"}] String Ensure; - [Write, Description("Specifies the replica availability mode. Default is 'AsynchronousCommit'."), ValueMap{"AsynchronousCommit","SynchronousCommit"}, Values{"AsynchronousCommit","SynchronousCommit"}] String AvailabilityMode; + [Write, Description("Specifies the replica availability mode. Default when creating a replica is 'AsynchronousCommit'."), ValueMap{"AsynchronousCommit","SynchronousCommit"}, Values{"AsynchronousCommit","SynchronousCommit"}] String AvailabilityMode; [Write, Description("Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are: integers from 0 through 100. Of the set of replicas which are online and available, the replica that has the highest priority performs the backup. Default is 50.")] UInt32 BackupPriority; [Write, Description("Specifies how the availability replica handles connections when in the primary role."), ValueMap{"AllowAllConnections","AllowReadWriteConnections"}, Values{"AllowAllConnections","AllowReadWriteConnections"}] String ConnectionModeInPrimaryRole; [Write, Description("Specifies how the availability replica handles connections when in the secondary role."), ValueMap{"AllowNoConnections","AllowReadIntentConnectionsOnly","AllowAllConnections"}, Values{"AllowNoConnections","AllowReadIntentConnectionsOnly","AllowAllConnections"}] String ConnectionModeInSecondaryRole; - [Write, Description("Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name which is set in the code because the value can only be determined when connected to the SQL Instance.")] String EndpointHostName; - [Write, Description("Specifies the failover mode. Default is 'Manual'."), ValueMap{"Automatic","Manual"}, Values{"Automatic","Manual"}] String FailoverMode; + [Write, Description("Specifies the hostname or IP address of the availability group replica endpoint. When creating a group the default is the instance network name which is set in the code because the value can only be determined when connected to the SQL Instance.")] String EndpointHostName; + [Write, Description("Specifies the failover mode. Default when creating a replica is 'Manual'."), ValueMap{"Automatic","Manual"}, Values{"Automatic","Manual"}] String FailoverMode; [Write, Description("Specifies the fully-qualified domain name (FQDN) and port to use when routing to the replica for read only connections.")] String ReadOnlyRoutingConnectionUrl; [Write, Description("Specifies an ordered list of replica server names that represent the probe sequence for connection director to use when redirecting read-only connections through this availability replica. This parameter applies if the availability replica is the current primary replica of the availability group.")] String ReadOnlyRoutingList[]; [Write, Description("Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server instance.")] Boolean ProcessOnlyOnActiveNode; diff --git a/README.md b/README.md index 9ce9bfccb..54f363864 100644 --- a/README.md +++ b/README.md @@ -184,14 +184,16 @@ It will also manage the Availability Group replica on the specified node. * **`[String]` Ensure** _(Write)_: Specifies if the availability group should be present or absent. Default is Present. { *Present* | Absent } * **`[String]` AutomatedBackupPreference** _(Write)_: Specifies the automated backup - preference for the availability group. Default is None. + preference for the availability group. When creating a group the default is 'None'. { Primary | SecondaryOnly | Secondary | *None* } * **`[String]` AvailabilityMode** _(Write)_: Specifies the replica availability mode. - Default is 'AsynchronousCommit'. { *AsynchronousCommit* | SynchronousCommit } + Default when creating a group is 'AsynchronousCommit'. + { *AsynchronousCommit* | SynchronousCommit } * **`[UInt32]` BackupPriority** _(Write)_: Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are: integers from 0 through 100. Of the set of replicas which are online and available, - the replica that has the highest priority performs the backup. Default is 50. + the replica that has the highest priority performs the backup. When creating + a group the default is 50. * **`[Boolean]` BasicAvailabilityGroup** _(Write)_: Specifies the type of availability group is Basic. This is only available is SQL Server 2016 and later and is ignored when applied to previous versions. @@ -215,10 +217,10 @@ It will also manage the Availability Group replica on the specified node. OnCriticalServerErrors | OnModerateServerErrors | OnAnyQualifiedFailureCondition } * **`[String]` FailoverMode** _(Write)_: Specifies the failover mode. - Default is 'Manual'. { Automatic | *Manual* } + When creating a group the default is 'Manual'. { Automatic | *Manual* } * **`[UInt32]` HealthCheckTimeout** _(Write)_: Specifies the length of time, in milliseconds, after which AlwaysOn availability groups declare an unresponsive - server to be unhealthy. Default is 30000. + server to be unhealthy. When creating a group the default is 30000. * **`[Boolean]` ProcessOnlyOnActiveNode** _(Write)_: Specifies that the resource will only determine if a change is needed if the target node is the active host of the SQL Server Instance. @@ -414,11 +416,13 @@ Always On Availability Group Replica. * **`[String]` Ensure** _(Write)_: Specifies if the availability group replica should be present or absent. Default is Present. { *Present* | Absent } * **`[String]` AvailabilityMode** _(Write)_: Specifies the replica availability mode. - Default is 'AsynchronousCommit'. { *AsynchronousCommit* | SynchronousCommit } + When creating a replica the default is 'AsynchronousCommit'. + { *AsynchronousCommit* | SynchronousCommit } * **`[UInt32]` BackupPriority** _(Write)_: Specifies the desired priority of the replicas in performing backups. The acceptable values for this parameter are: integers from 0 through 100. Of the set of replicas which are online and available, - the replica that has the highest priority performs the backup. Default is 50. + the replica that has the highest priority performs the backup. When creating a + replica the default is 50. * **`[String]` ConnectionModeInPrimaryRole** _(Write)_: Specifies how the availability replica handles connections when in the primary role. { AllowAllConnections | AllowReadWriteConnections } @@ -427,8 +431,8 @@ Always On Availability Group Replica. { AllowNoConnections | AllowReadIntentConnectionsOnly | AllowAllConnections } * **`[String]` EndpointHostName** _(Write)_: Specifies the hostname or IP address of the availability group replica endpoint. Default is the instance network name. -* **`[String]` FailoverMode** _(Write)_: Specifies the failover mode. Default is - 'Manual'. { Automatic | *Manual* } +* **`[String]` FailoverMode** _(Write)_: Specifies the failover mode. + When creating a replica the default is 'Manual'. { Automatic | *Manual* } * **`[String]` ReadOnlyRoutingConnectionUrl** _(Write)_: Specifies the fully-qualified domain name (FQDN) and port to use when routing to the replica for read only connections. diff --git a/Tests/Unit/Stubs/SMO.cs b/Tests/Unit/Stubs/SMO.cs index 12744e18a..02ba17e86 100644 --- a/Tests/Unit/Stubs/SMO.cs +++ b/Tests/Unit/Stubs/SMO.cs @@ -765,7 +765,7 @@ public AvailabilityReplica( AvailabilityGroup availabilityGroup, string name ) public string FailoverMode; public string Name; public string ReadOnlyRoutingConnectionUrl; - public string[] ReadOnlyRoutingList; + public System.Collections.Specialized.StringCollection ReadOnlyRoutingList; public string Role = "Secondary"; public void Alter() From 98eaa8460af0099cdbc757cd8213541cba49a086 Mon Sep 17 00:00:00 2001 From: Cody Konior Date: Wed, 20 Mar 2019 21:50:09 +0800 Subject: [PATCH 08/10] SqlAGReplica: Fix read-only routing list reaching desired state (#1312) - Changes to SqlAGReplica - Test-Resource fixed to report whether ReadOnlyRoutingList desired state has been reached correctly (issue #1305). --- CHANGELOG.md | 2 ++ DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 | 2 +- .../MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 | 13 +++++++++++-- .../MSFT_SqlServerEndpointPermission.psm1 | 4 ++-- .../3-CreateAvailabilityGroupDetailed.ps1 | 6 +++--- Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 | 19 +++++++++++++------ Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 | 8 ++++---- .../MSFT_SqlServerSecureConnection.Tests.ps1 | 4 ++-- 8 files changed, 38 insertions(+), 20 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index c11d4eed3..38e706e14 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -50,6 +50,8 @@ parameters will still change existing replicas ([issue #1244](https://github.com/PowerShell/SqlServerDsc/issues/1244)). - ReadOnlyRoutingList now gets updated without throwing an error on the first run ([issue #518](https://github.com/PowerShell/SqlServerDsc/issues/518)). + - Test-Resource fixed to report whether ReadOnlyRoutingList desired state + has been reached correctly ([issue #1305](https://github.com/PowerShell/SqlServerDsc/issues/1305)). ## 12.3.0.0 diff --git a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 index f58e05efd..54120a639 100644 --- a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 +++ b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 @@ -695,7 +695,7 @@ function Test-TargetResource { New-VerboseMessage -Message "'$($parameterName)' should be '$($parameterValue)' but is '$($getTargetResourceResult.($parameterName))'" - $result = $False + $result = $false } } diff --git a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 index 37089e4d7..6c129416c 100644 --- a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 +++ b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 @@ -666,11 +666,20 @@ function Test-TargetResource continue } - if ( $getTargetResourceResult.($parameterName) -ne $parameterValue ) + if ( $parameterName -eq 'ReadOnlyRoutingList' ) + { + $different = ( $getTargetResourceResult.($parameterName) -join ',' ) -ne ( $parameterValue -join ',' ) + } + else + { + $different = $getTargetResourceResult.($parameterName) -ne $parameterValue + } + + if ( $different ) { New-VerboseMessage -Message "'$($parameterName)' should be '$($parameterValue)' but is '$($getTargetResourceResult.($parameterName))'" - $result = $False + $result = $false } } diff --git a/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 b/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 index 16bb3ae3c..92f3b5af2 100644 --- a/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 +++ b/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 @@ -49,7 +49,7 @@ function Get-TargetResource { New-VerboseMessage -Message "Enumerating permissions for endpoint $Name" - $permissionSet = New-Object -Property @{ Connect = $True } -TypeName Microsoft.SqlServer.Management.Smo.ObjectPermissionSet + $permissionSet = New-Object -Property @{ Connect = $true } -TypeName Microsoft.SqlServer.Management.Smo.ObjectPermissionSet $endpointPermission = $endpointObject.EnumObjectPermissions( $permissionSet ) | Where-Object { $_.PermissionState -eq "Grant" -and $_.Grantee -eq $Principal } if ($endpointPermission.Count -ne 0) @@ -152,7 +152,7 @@ function Set-TargetResource $endpointObject = $sqlServerObject.Endpoints[$Name] if ($null -ne $endpointObject) { - $permissionSet = New-Object -Property @{ Connect = $True } -TypeName Microsoft.SqlServer.Management.Smo.ObjectPermissionSet + $permissionSet = New-Object -Property @{ Connect = $true } -TypeName Microsoft.SqlServer.Management.Smo.ObjectPermissionSet if ($Ensure -eq 'Present') { diff --git a/Examples/Resources/SqlAG/3-CreateAvailabilityGroupDetailed.ps1 b/Examples/Resources/SqlAG/3-CreateAvailabilityGroupDetailed.ps1 index 1b026e1b9..b0b2b252a 100644 --- a/Examples/Resources/SqlAG/3-CreateAvailabilityGroupDetailed.ps1 +++ b/Examples/Resources/SqlAG/3-CreateAvailabilityGroupDetailed.ps1 @@ -23,9 +23,9 @@ $ConfigurationData = @{ FailoverMode = 'Automatic' HealthCheckTimeout = 15000 - BasicAvailabilityGroup = $False - DatabaseHealthTrigger = $True - DtcSupportEnabled = $True + BasicAvailabilityGroup = $false + DatabaseHealthTrigger = $true + DtcSupportEnabled = $true }, @{ diff --git a/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 b/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 index 53fe6f5a3..6233f9b98 100644 --- a/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 @@ -71,7 +71,7 @@ try $mockEndpointHostName = $mockServerName $mockFailoverMode = 'Manual' $mockReadOnlyRoutingConnectionUrl = "TCP://$($mockServerName).domain.com:1433" - $mockReadOnlyRoutingList = @($mockServerName) + $mockReadOnlyRoutingList = @('Server1', 'Server2') $mockProcessOnlyOnActiveNode = $false #endregion @@ -153,7 +153,7 @@ try $mockAvailabilityGroupReplica1EndpointUrl = "$($mockAvailabilityGroupReplica1EndpointProtocol)://$($mockServer1Name):$($mockAvailabilityGroupReplica1EndpointPort)" $mockAvailabilityGroupReplica1FailoverMode = 'Manual' $mockAvailabilityGroupReplica1ReadOnlyRoutingConnectionUrl = "TCP://$($mockServer1Name).domain.com:1433" - $mockAvailabilityGroupReplica1ReadOnlyRoutingList = @($mockServer1Name) + $mockAvailabilityGroupReplica1ReadOnlyRoutingList = $mockReadOnlyRoutingList $mockAvailabilityGroupReplica2Name = $mockServer2Name $mockAvailabilityGroupReplica2AvailabilityMode = 'AsynchronousCommit' @@ -165,7 +165,7 @@ try $mockAvailabilityGroupReplica2EndpointUrl = "$($mockAvailabilityGroupReplica2EndpointProtocol)://$($mockServer2Name):$($mockAvailabilityGroupReplica2EndpointPort)" $mockAvailabilityGroupReplica2FailoverMode = 'Manual' $mockAvailabilityGroupReplica2ReadOnlyRoutingConnectionUrl = "TCP://$($mockServer2Name).domain.com:1433" - $mockAvailabilityGroupReplica2ReadOnlyRoutingList = @($mockServer2Name) + $mockAvailabilityGroupReplica2ReadOnlyRoutingList = $mockReadOnlyRoutingList $mockAvailabilityGroupReplica3Name = $mockServer3Name $mockAvailabilityGroupReplica3AvailabilityMode = 'AsynchronousCommit' @@ -177,7 +177,7 @@ try $mockAvailabilityGroupReplica3EndpointUrl = "$($mockAvailabilityGroupReplica3EndpointProtocol)://$($mockServer3Name):$($mockAvailabilityGroupReplica3EndpointPort)" $mockAvailabilityGroupReplica3FailoverMode = 'Manual' $mockAvailabilityGroupReplica3ReadOnlyRoutingConnectionUrl = "TCP://$($mockServer3Name).domain.com:1433" - $mockAvailabilityGroupReplica3ReadOnlyRoutingList = @($mockServer3Name) + $mockAvailabilityGroupReplica3ReadOnlyRoutingList = $mockReadOnlyRoutingList #endregion @@ -612,7 +612,7 @@ try $getTargetResourceResult.FailoverMode | Should -Be $mockFailoverMode $getTargetResourceResult.Name | Should -Be $mockServerName $getTargetResourceResult.ReadOnlyRoutingConnectionUrl | Should -Be $mockReadOnlyRoutingConnectionUrl - $getTargetResourceResult.ReadOnlyRoutingList | Should -Be $mockServerName + $getTargetResourceResult.ReadOnlyRoutingList | Should -Be $mockReadOnlyRoutingList $getTargetResourceResult.ServerName | Should -Be $mockServerName $getTargetResourceResult.InstanceName | Should -Be $mockInstanceName $getTargetResourceResult.EndpointHostName | Should -Be $mockServerName @@ -1083,7 +1083,7 @@ try ConnectionModeInSecondaryRole = 'AllowReadIntentConnectionsOnly' FailoverMode = 'Automatic' ReadOnlyRoutingConnectionUrl = 'TCP://TestHost.domain.com:1433' - ReadOnlyRoutingList = @('Server1', 'Server2') + ReadOnlyRoutingList = @('Server2', 'Server1') } } @@ -1355,6 +1355,13 @@ try } } + It "Should return $true when the Availability Replica is present all properties are in the desired state" { + Test-TargetResource @testTargetResourceParameters | Should -Be $true + + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-ActiveNode -Scope It -Times 1 -Exactly + } + It 'Should return $false when the Availability Replica is absent' { $testTargetResourceParameters.Name = $mockAvailabilityGroupReplica2Name diff --git a/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 b/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 index 9daa8758e..4ed04687e 100644 --- a/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 @@ -78,7 +78,7 @@ try $mockObjectSmoServer.Name = "$mockServerName\$mockInstanceName" $mockObjectSmoServer.DisplayName = $mockInstanceName $mockObjectSmoServer.InstanceName = $mockInstanceName - $mockObjectSmoServer.IsHadrEnabled = $False + $mockObjectSmoServer.IsHadrEnabled = $false $mockObjectSmoServer.MockGranteeName = $mockPrincipal return $mockObjectSmoServer @@ -204,7 +204,7 @@ try $mockObjectSmoServer.Name = "$mockServerName\$mockInstanceName" $mockObjectSmoServer.DisplayName = $mockInstanceName $mockObjectSmoServer.InstanceName = $mockInstanceName - $mockObjectSmoServer.IsHadrEnabled = $False + $mockObjectSmoServer.IsHadrEnabled = $false $mockObjectSmoServer.MockGranteeName = $mockPrincipal return $mockObjectSmoServer @@ -272,7 +272,7 @@ try $mockObjectSmoServer.Name = "$mockServerName\$mockInstanceName" $mockObjectSmoServer.DisplayName = $mockInstanceName $mockObjectSmoServer.InstanceName = $mockInstanceName - $mockObjectSmoServer.IsHadrEnabled = $False + $mockObjectSmoServer.IsHadrEnabled = $false $mockObjectSmoServer.MockGranteeName = $mockPrincipal return $mockObjectSmoServer @@ -329,7 +329,7 @@ try $mockObjectSmoServer.Name = "$mockServerName\$mockInstanceName" $mockObjectSmoServer.DisplayName = $mockInstanceName $mockObjectSmoServer.InstanceName = $mockInstanceName - $mockObjectSmoServer.IsHadrEnabled = $False + $mockObjectSmoServer.IsHadrEnabled = $false # This make the SMO Server object mock to throw when Grant() method is called. $mockObjectSmoServer.MockGranteeName = $mockOtherPrincipal diff --git a/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 b/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 index 70045cd9f..30dbcc64d 100644 --- a/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 @@ -638,7 +638,7 @@ try It 'Should return False when no permissions were found' { $result = Test-CertificatePermission -Thumbprint '12345678' -ServiceAccount 'Everyone' - $result | Should -be $False + $result | Should -be $false Assert-VerifiableMock } } @@ -654,7 +654,7 @@ try It 'Should return False when the wrong permissions are added' { $result = Test-CertificatePermission -Thumbprint '12345678' -ServiceAccount 'Everyone' - $result | Should -be $False + $result | Should -be $false Assert-VerifiableMock } } From f13c0014c79ad8b848f5cd33bc81daa151810821 Mon Sep 17 00:00:00 2001 From: Johan Ljunggren Date: Wed, 20 Mar 2019 15:38:06 +0100 Subject: [PATCH 09/10] SqlServerDsc: New resource SqlRSSetup, plus other changes (#1308) - Changes to SqlServerDsc - Added new resources. - SqlRSSetup - Added helper module DscResource.Common from the repository DscResource.Template. - Moved all helper functions from SqlServerDscHelper.psm1 to DscResource.Common. - Renamed Test-SqlDscParameterState to Test-DscParameterState. - Added helper module DscResource.LocalizationHelper from the repository DscResource.Template, this replaces the helper module CommonResourceHelper.psm1. - Cleaned up unit tests, mostly around loading cmdlet stubs and loading classes stubs, but also some tests that were using some odd variants. - Fix all integration tests according to issue [PowerShell/DscResource.Template#14](https://github.com/PowerShell/DscResource.Template/issues/14). - Changes to SqlSetup - Moved some resource specific helper functions to the new helper module DscResource.Common so they can be shared with the new resource SqlRSSetup. - Improved verbose messages in Test-TargetResource function to more clearly tell if features are already installed or not. - Refactored unit tests for the functions Test-TargetResource and Set-TargetResource to improve testing speed. - Modified the Test-TargetResource and Set-TargetResource to not be case-sensitive when comparing feature names. *This was handled correctly in real-world scenarios, but failed when running the unit tests (and testing casing).* - Changes to SqlDatabaseDefaultLocation - No longer does the Test-TargetResource fail on the second test run when the backup file path was changed, and the path was ending with a backslash (issue #1307). --- CHANGELOG.md | 33 +- DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 | 11 +- .../MSFT_SqlAGDatabase.psm1 | 12 +- .../MSFT_SqlAGListener.psm1 | 12 +- .../MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 | 11 +- .../MSFT_SqlAgentOperator.psm1 | 12 +- DSCResources/MSFT_SqlAlias/MSFT_SqlAlias.psm1 | 9 +- .../MSFT_SqlAlwaysOnService.psm1 | 11 +- .../MSFT_SqlDatabase/MSFT_SqlDatabase.psm1 | 12 +- .../MSFT_SqlDatabaseDefaultLocation.psm1 | 24 +- .../MSFT_SqlDatabaseOwner.psm1 | 14 +- .../MSFT_SqlDatabasePermission.psm1 | 14 +- .../MSFT_SqlDatabaseRecoveryModel.psm1 | 14 +- .../MSFT_SqlDatabaseRole.psm1 | 11 +- DSCResources/MSFT_SqlRS/MSFT_SqlRS.psm1 | 11 +- .../MSFT_SqlRSSetup/MSFT_SqlRSSetup.psm1 | 801 ++++ .../MSFT_SqlRSSetup.schema.mof | 21 + .../en-US/MSFT_SqlRSSetup.strings.psd1 | 25 + .../MSFT_SqlScript/MSFT_SqlScript.psm1 | 10 +- .../MSFT_SqlScriptQuery.psm1 | 10 +- .../MSFT_SqlServerConfiguration.psm1 | 12 +- .../MSFT_SqlServerDatabaseMail.psm1 | 14 +- .../MSFT_SqlServerEndpoint.psm1 | 12 +- .../MSFT_SqlServerEndpointPermission.psm1 | 12 +- .../MSFT_SqlServerEndpointState.psm1 | 12 +- .../MSFT_SqlServerLogin.psm1 | 21 +- .../MSFT_SqlServerMaxDop.psm1 | 12 +- .../MSFT_SqlServerMemory.psm1 | 9 +- .../MSFT_SqlServerNetwork.psm1 | 12 +- .../MSFT_SqlServerPermission.psm1 | 12 +- .../MSFT_SqlServerRole.psm1 | 11 +- .../MSFT_SqlServerSecureConnection.psm1 | 10 +- .../MSFT_SqlServiceAccount.psm1 | 11 +- DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 | 261 +- .../en-US/MSFT_SqlSetup.strings.psd1 | 13 +- .../MSFT_SqlWaitForAG/MSFT_SqlWaitForAG.psm1 | 11 +- .../MSFT_SqlWindowsFirewall.psm1 | 10 +- Examples/README.md | 1 + .../SqlRSSetup/1-InstallReportingServices.ps1 | 31 + .../2-UninstallReportingServices.ps1 | 33 + .../DscResource.Common.psm1 | 818 +++- .../en-US/DscResource.Common.strings.psd1 | 27 +- .../sv-SE/DscResource.Common.strings.psd1 | 2 +- .../DscResource.LocalizationHelper.psm1 | 133 +- README.md | 102 + ...SFT_SqlAgentOperator.Integration.Tests.ps1 | 4 +- ...T_SqlAlwaysOnService.Integration.Tests.ps1 | 4 +- ...abaseDefaultLocation.Integration.Tests.ps1 | 13 +- ...MSFT_SqlDatabaseDefaultLocation.config.ps1 | 2 + .../MSFT_SqlRS.Integration.Tests.ps1 | 8 +- .../MSFT_SqlRSSetup.Integration.Tests.ps1 | 161 + Tests/Integration/MSFT_SqlRSSetup.config.ps1 | 90 + .../MSFT_SqlScript.Integration.Tests.ps1 | 4 +- .../MSFT_SqlScriptQuery.Integration.Tests.ps1 | 4 +- ...qlServerDatabaseMail.Integration.Tests.ps1 | 4 +- ...FT_SqlServerEndPoint.Integration.Tests.ps1 | 4 +- .../MSFT_SqlServerLogin.Integration.Tests.ps1 | 12 +- ...SFT_SqlServerNetwork.Integration.Tests.ps1 | 4 +- .../MSFT_SqlServerRole.Integration.Tests.ps1 | 14 +- ...rverSecureConnection.Integration.Tests.ps1 | 4 +- ...FT_SqlServiceAccount.Integration.Tests.ps1 | 16 +- .../MSFT_SqlSetup.Integration.Tests.ps1 | 6 +- Tests/Integration/README.md | 98 +- Tests/Unit/CommonResourceHelper.Tests.ps1 | 210 -- ...Tests.ps1 => DscResource.Common.Tests.ps1} | 3344 ++++++++++------- .../DscResource.LocalizationHelper.Tests.ps1 | 203 + Tests/Unit/MSFT_SqlAG.Tests.ps1 | 6 +- Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 | 10 +- Tests/Unit/MSFT_SqlAGListener.Tests.ps1 | 4 +- Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 | 7 +- Tests/Unit/MSFT_SqlDatabase.Tests.ps1 | 1 - .../MSFT_SqlDatabaseDefaultLocation.Tests.ps1 | 103 +- .../Unit/MSFT_SqlDatabasePermission.Tests.ps1 | 2 +- Tests/Unit/MSFT_SqlDatabaseRole.Tests.ps1 | 2 +- Tests/Unit/MSFT_SqlRS.Tests.ps1 | 3 +- Tests/Unit/MSFT_SqlRSSetup.Tests.ps1 | 958 +++++ Tests/Unit/MSFT_SqlScript.Tests.ps1 | 249 +- Tests/Unit/MSFT_SqlScriptQuery.Tests.ps1 | 253 +- Tests/Unit/MSFT_SqlServerEndpoint.Tests.ps1 | 2 +- ...MSFT_SqlServerEndpointPermission.Tests.ps1 | 2 +- .../MSFT_SqlServerEndpointState.Tests.ps1 | 4 +- Tests/Unit/MSFT_SqlServerLogin.Tests.ps1 | 7 +- Tests/Unit/MSFT_SqlServerMaxDop.Tests.ps1 | 5 +- Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 | 5 +- Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 | 2 +- .../Unit/MSFT_SqlServerReplication.Tests.ps1 | 3 +- .../MSFT_SqlServerSecureConnection.Tests.ps1 | 3 +- Tests/Unit/MSFT_SqlServiceAccount.Tests.ps1 | 4 +- Tests/Unit/MSFT_SqlSetup.Tests.ps1 | 1607 ++------ 89 files changed, 6192 insertions(+), 3939 deletions(-) create mode 100644 DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.psm1 create mode 100644 DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.schema.mof create mode 100644 DSCResources/MSFT_SqlRSSetup/en-US/MSFT_SqlRSSetup.strings.psd1 create mode 100644 Examples/Resources/SqlRSSetup/1-InstallReportingServices.ps1 create mode 100644 Examples/Resources/SqlRSSetup/2-UninstallReportingServices.ps1 rename SqlServerDscHelper.psm1 => Modules/DscResource.Common/DscResource.Common.psm1 (79%) rename en-US/SqlServerDscHelper.strings.psd1 => Modules/DscResource.Common/en-US/DscResource.Common.strings.psd1 (89%) rename sv-SE/SqlServerDscHelper.strings.psd1 => Modules/DscResource.Common/sv-SE/DscResource.Common.strings.psd1 (99%) rename DSCResources/CommonResourceHelper.psm1 => Modules/DscResource.LocalizationHelper/DscResource.LocalizationHelper.psm1 (89%) create mode 100644 Tests/Integration/MSFT_SqlRSSetup.Integration.Tests.ps1 create mode 100644 Tests/Integration/MSFT_SqlRSSetup.config.ps1 delete mode 100644 Tests/Unit/CommonResourceHelper.Tests.ps1 rename Tests/Unit/{SqlServerDSCHelper.Tests.ps1 => DscResource.Common.Tests.ps1} (64%) create mode 100644 Tests/Unit/DscResource.LocalizationHelper.Tests.ps1 create mode 100644 Tests/Unit/MSFT_SqlRSSetup.Tests.ps1 diff --git a/CHANGELOG.md b/CHANGELOG.md index 38e706e14..827851b12 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,21 @@ ## Unreleased +- Changes to SqlServerDsc + - Added new resources. + - SqlRSSetup + - Added helper module DscResource.Common from the repository + DscResource.Template. + - Moved all helper functions from SqlServerDscHelper.psm1 to DscResource.Common. + - Renamed Test-SqlDscParameterState to Test-DscParameterState. + - New-TerminatingError error text for a missing localized message now matches + the output even if the "missing localized message" localized message is + also missing. + - Added helper module DscResource.LocalizationHelper from the repository + DscResource.Template, this replaces the helper module CommonResourceHelper.psm1. + - Cleaned up unit tests, mostly around loading cmdlet stubs and loading + classes stubs, but also some tests that were using some odd variants. + - Fix all integration tests according to issue [PowerShell/DscResource.Template#14](https://github.com/PowerShell/DscResource.Template/issues/14). - Changes to SqlServerMemory - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) because the correct memory size was not being detected correctly on Azure VMs @@ -33,14 +48,20 @@ - Changed the logic of 'Build the argument string to be passed to setup' to not quote the value if root directory is specified ([issue #1254](https://github.com/PowerShell/SqlServerDsc/issues/1254)). + - Moved some resource specific helper functions to the new helper module + DscResource.Common so they can be shared with the new resource SqlRSSetup. + - Improved verbose messages in Test-TargetResource function to more + clearly tell if features are already installed or not. + - Refactored unit tests for the functions Test-TargetResource and + Set-TargetResource to improve testing speed. + - Modified the Test-TargetResource and Set-TargetResource to not be + case-sensitive when comparing feature names. *This was handled + correctly in real-world scenarios, but failed when running the unit + tests (and testing casing).* - Changes to SqlAGDatabase - Fix MatchDatabaseOwner to check for CONTROL SERVER, IMPERSONATE LOGIN, or CONTROL LOGIN permission in addition to IMPERSONATE ANY LOGIN. - Update and fix MatchDatabaseOwner help text. -- Changes to xSQLServerHelper - - New-TerminatingError error text for a missing localized message now matches - the output even if the "missing localized message" localized message is - also missing. - Changes to SqlAG - Updated documentation on the behaviour of defaults as they only apply when creating a group. @@ -52,6 +73,10 @@ run ([issue #518](https://github.com/PowerShell/SqlServerDsc/issues/518)). - Test-Resource fixed to report whether ReadOnlyRoutingList desired state has been reached correctly ([issue #1305](https://github.com/PowerShell/SqlServerDsc/issues/1305)). +- Changes to SqlDatabaseDefaultLocation + - No longer does the Test-TargetResource fail on the second test run + when the backup file path was changed, and the path was ending with + a backslash ([issue #1307](https://github.com/PowerShell/SqlServerDsc/issues/1307)). ## 12.3.0.0 diff --git a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 index 54120a639..fb3ec33e0 100644 --- a/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 +++ b/DSCResources/MSFT_SqlAG/MSFT_SqlAG.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 b/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 index 58381970c..507532df4 100644 --- a/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 +++ b/DSCResources/MSFT_SqlAGDatabase/MSFT_SqlAGDatabase.psm1 @@ -1,9 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlAGDatabase' diff --git a/DSCResources/MSFT_SqlAGListener/MSFT_SqlAGListener.psm1 b/DSCResources/MSFT_SqlAGListener/MSFT_SqlAGListener.psm1 index 69a01d975..2ca4e66df 100644 --- a/DSCResources/MSFT_SqlAGListener/MSFT_SqlAGListener.psm1 +++ b/DSCResources/MSFT_SqlAGListener/MSFT_SqlAGListener.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS Returns the current state of the Availability Group listener. diff --git a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 index 6c129416c..12808d29d 100644 --- a/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 +++ b/DSCResources/MSFT_SqlAGReplica/MSFT_SqlAGReplica.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlAgentOperator/MSFT_SqlAgentOperator.psm1 b/DSCResources/MSFT_SqlAgentOperator/MSFT_SqlAgentOperator.psm1 index f1c0d0047..aa292485b 100644 --- a/DSCResources/MSFT_SqlAgentOperator/MSFT_SqlAgentOperator.psm1 +++ b/DSCResources/MSFT_SqlAgentOperator/MSFT_SqlAgentOperator.psm1 @@ -1,9 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlAgentOperator' diff --git a/DSCResources/MSFT_SqlAlias/MSFT_SqlAlias.psm1 b/DSCResources/MSFT_SqlAlias/MSFT_SqlAlias.psm1 index 02fc47d98..de21afa39 100644 --- a/DSCResources/MSFT_SqlAlias/MSFT_SqlAlias.psm1 +++ b/DSCResources/MSFT_SqlAlias/MSFT_SqlAlias.psm1 @@ -1,4 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) -ChildPath 'SqlServerDscHelper.psm1') -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') function Get-TargetResource { diff --git a/DSCResources/MSFT_SqlAlwaysOnService/MSFT_SqlAlwaysOnService.psm1 b/DSCResources/MSFT_SqlAlwaysOnService/MSFT_SqlAlwaysOnService.psm1 index 4f6aa384e..5c7683c38 100644 --- a/DSCResources/MSFT_SqlAlwaysOnService/MSFT_SqlAlwaysOnService.psm1 +++ b/DSCResources/MSFT_SqlAlwaysOnService/MSFT_SqlAlwaysOnService.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlDatabase/MSFT_SqlDatabase.psm1 b/DSCResources/MSFT_SqlDatabase/MSFT_SqlDatabase.psm1 index 0f3bd5150..afa5a205b 100644 --- a/DSCResources/MSFT_SqlDatabase/MSFT_SqlDatabase.psm1 +++ b/DSCResources/MSFT_SqlDatabase/MSFT_SqlDatabase.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS This function gets the sql database. diff --git a/DSCResources/MSFT_SqlDatabaseDefaultLocation/MSFT_SqlDatabaseDefaultLocation.psm1 b/DSCResources/MSFT_SqlDatabaseDefaultLocation/MSFT_SqlDatabaseDefaultLocation.psm1 index cd02b9b39..d6f908048 100644 --- a/DSCResources/MSFT_SqlDatabaseDefaultLocation/MSFT_SqlDatabaseDefaultLocation.psm1 +++ b/DSCResources/MSFT_SqlDatabaseDefaultLocation/MSFT_SqlDatabaseDefaultLocation.psm1 @@ -1,9 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlDatabaseDefaultLocation' @@ -145,6 +147,12 @@ Function Set-TargetResource $ProcessOnlyOnActiveNode ) + if ($Type -eq 'Backup') + { + # Ending backslash is removed because of issue #1307. + $Path = $Path.TrimEnd('\') + } + # Make sure the Path exists, needs to be cluster aware as well for this check if (-Not (Test-Path $Path)) { @@ -263,6 +271,12 @@ function Test-TargetResource Write-Verbose -Message ($script:localizedData.TestingCurrentPath -f $Type) + if ($Type -eq 'Backup') + { + # Ending backslash is removed because of issue #1307. + $Path = $Path.TrimEnd('\') + } + $getTargetResourceParameters = @{ InstanceName = $InstanceName ServerName = $ServerName diff --git a/DSCResources/MSFT_SqlDatabaseOwner/MSFT_SqlDatabaseOwner.psm1 b/DSCResources/MSFT_SqlDatabaseOwner/MSFT_SqlDatabaseOwner.psm1 index ce7268158..c1005ebce 100644 --- a/DSCResources/MSFT_SqlDatabaseOwner/MSFT_SqlDatabaseOwner.psm1 +++ b/DSCResources/MSFT_SqlDatabaseOwner/MSFT_SqlDatabaseOwner.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS This function gets the owner of the desired sql database. @@ -200,7 +206,7 @@ function Test-TargetResource Write-Verbose -Message "Testing owner $Name of database $Database" $currentValues = Get-TargetResource @PSBoundParameters - return Test-SQLDscParameterState -CurrentValues $CurrentValues ` + return Test-DscParameterState -CurrentValues $CurrentValues ` -DesiredValues $PSBoundParameters ` -ValuesToCheck @('Name', 'Database') } diff --git a/DSCResources/MSFT_SqlDatabasePermission/MSFT_SqlDatabasePermission.psm1 b/DSCResources/MSFT_SqlDatabasePermission/MSFT_SqlDatabasePermission.psm1 index 178180639..6904f0bb4 100644 --- a/DSCResources/MSFT_SqlDatabasePermission/MSFT_SqlDatabasePermission.psm1 +++ b/DSCResources/MSFT_SqlDatabasePermission/MSFT_SqlDatabasePermission.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS Returns the current permissions for the user in the database @@ -402,7 +408,7 @@ function Test-TargetResource the value 'Present' for the Ensure parameter, otherwise Ensure will have the value 'Absent'. #> - return Test-SQLDscParameterState -CurrentValues $getTargetResourceResult ` + return Test-DscParameterState -CurrentValues $getTargetResourceResult ` -DesiredValues $PSBoundParameters ` -ValuesToCheck @('Name', 'Ensure', 'PermissionState') } diff --git a/DSCResources/MSFT_SqlDatabaseRecoveryModel/MSFT_SqlDatabaseRecoveryModel.psm1 b/DSCResources/MSFT_SqlDatabaseRecoveryModel/MSFT_SqlDatabaseRecoveryModel.psm1 index b5d698536..4d2c52792 100644 --- a/DSCResources/MSFT_SqlDatabaseRecoveryModel/MSFT_SqlDatabaseRecoveryModel.psm1 +++ b/DSCResources/MSFT_SqlDatabaseRecoveryModel/MSFT_SqlDatabaseRecoveryModel.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS This function gets all Key properties defined in the resource schema file @@ -191,7 +197,7 @@ function Test-TargetResource $currentValues = Get-TargetResource @PSBoundParameters - return Test-SQLDscParameterState -CurrentValues $currentValues ` + return Test-DscParameterState -CurrentValues $currentValues ` -DesiredValues $PSBoundParameters ` -ValuesToCheck @('Name', 'RecoveryModel') } diff --git a/DSCResources/MSFT_SqlDatabaseRole/MSFT_SqlDatabaseRole.psm1 b/DSCResources/MSFT_SqlDatabaseRole/MSFT_SqlDatabaseRole.psm1 index df2e34ffd..8ddbb884d 100644 --- a/DSCResources/MSFT_SqlDatabaseRole/MSFT_SqlDatabaseRole.psm1 +++ b/DSCResources/MSFT_SqlDatabaseRole/MSFT_SqlDatabaseRole.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlRS/MSFT_SqlRS.psm1 b/DSCResources/MSFT_SqlRS/MSFT_SqlRS.psm1 index 227dfa040..2f96ed6d2 100644 --- a/DSCResources/MSFT_SqlRS/MSFT_SqlRS.psm1 +++ b/DSCResources/MSFT_SqlRS/MSFT_SqlRS.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.psm1 b/DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.psm1 new file mode 100644 index 000000000..5db49aeeb --- /dev/null +++ b/DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.psm1 @@ -0,0 +1,801 @@ +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + +$script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlRSSetup' + +<# + .SYNOPSIS + Returns the current state of the Microsoft SQL Server Reporting Service + instance. + + .PARAMETER InstanceName + Name of the Microsoft SQL Server Reporting Service instance to installed. + This can only be set to 'SSRS'. { 'SSRS' } + + .PARAMETER IAcceptLicenseTerms + Accept licens terms. This must be set to 'Yes'. { 'Yes' } + + .PARAMETER SourcePath + The path to the installation media file to be used for installation, + e.g an UNC path to a shared resource. Environment variables can be used + in the path. + + .NOTES + The following properties are always returning $null because it's currently + unknown how to return that information. + - ProductKey + - Edition + + The following properties always return $null on purpose. This could be + changed in the future. + - Action + - SourceCredential + - ForceRestart + - EditionUpgrade + - VersionUpgrade + - LogPath + +#> +function Get-TargetResource +{ + [CmdletBinding()] + [OutputType([System.Collections.Hashtable])] + param + ( + [Parameter(Mandatory = $true)] + [ValidateSet('SSRS')] + [System.String] + $InstanceName, + + [Parameter(Mandatory = $true)] + [ValidateSet('Yes')] + [System.String] + $IAcceptLicenseTerms, + + [Parameter(Mandatory = $true)] + [System.String] + $SourcePath + ) + + $returnObject = @{ + InstanceName = $null + IAcceptLicenseTerms = $IAcceptLicenseTerms + SourcePath = $SourcePath + Action = $null + SourceCredential = $null + ProductKey = $null + ForceRestart = $false + EditionUpgrade = $false + VersionUpgrade = $false + Edition = $null + LogPath = $null + InstallFolder = $null + ErrorDumpDirectory = $null + CurrentVersion = $null + ServiceName = $null + } + + $InstanceName = $InstanceName.ToUpper() + + $getRegistryPropertyValueParameters = @{ + Path = 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\RS' + Name = $InstanceName + } + + $reportingServiceInstanceId = Get-RegistryPropertyValue @getRegistryPropertyValueParameters + if ($reportingServiceInstanceId) + { + Write-Verbose -Message ( + $script:localizedData.FoundInstance -f $InstanceName + ) + + # InstanceName + $returnObject['InstanceName'] = $InstanceName + + # InstallFolder + $getRegistryPropertyValueParameters = @{ + Path = 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\SSRS\Setup' + Name = 'InstallRootDirectory' + } + + $returnObject['InstallFolder'] = Get-RegistryPropertyValue @getRegistryPropertyValueParameters + + # ServiceName + $getRegistryPropertyValueParameters['Name'] = 'ServiceName' + + $returnObject['ServiceName'] = Get-RegistryPropertyValue @getRegistryPropertyValueParameters + + # ErrorDumpDirectory + $getRegistryPropertyValueParameters = @{ + Path = 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\SSRS\CPE' + Name = 'ErrorDumpDir' + } + + $returnObject['ErrorDumpDirectory'] = Get-RegistryPropertyValue @getRegistryPropertyValueParameters + + # CurrentVersion + $getPackageParameters = @{ + Name = 'Microsoft SQL Server Reporting Services' + ProviderName = 'Programs' + ErrorAction = 'SilentlyContinue' + # Get-Package returns a lot of excessive information that we don't need. + Verbose = $false + } + + $reportingServicesPackage = Get-Package @getPackageParameters + if ($reportingServicesPackage) + { + Write-Verbose -Message ( + $script:localizedData.VersionFound -f $reportingServicesPackage.Version + ) + + $returnObject['CurrentVersion'] = $reportingServicesPackage.Version + } + else + { + Write-Warning -Message $script:localizedData.PackageNotFound + } + } + else + { + Write-Verbose -Message ( + $script:localizedData.InstanceNotFound -f $InstanceName + ) + } + + return $returnObject +} + +<# + .SYNOPSIS + Installs the the Microsoft SQL Server Reporting Service instance. + + .PARAMETER InstanceName + Name of the Microsoft SQL Server Reporting Service instance to installed. + This can only be set to 'SSRS'. { 'SSRS' } + + .PARAMETER IAcceptLicenseTerms + Accept licens terms. This must be set to 'Yes'. { 'Yes' } + + .PARAMETER SourcePath + The path to the installation media file to be used for installation, + e.g an UNC path to a shared resource. Environment variables can be used + in the path. + + .PARAMETER Action + The action to be performed. Default value is 'Install' which performs + either install or upgrade. + { *Install* | Uninstall } + + .PARAMETER SourceCredential + Credentials used to access the path set in the parameter 'SourcePath'. + + .PARAMETER SuppressRestart + Suppresses any attempts to restart. + + .PARAMETER ProductKey + Sets the custom license key, e.g. '12345-12345-12345-12345-12345'. + + .PARAMETER ForceRestart + Forces a restart after installation is finished. + + .PARAMETER EditionUpgrade + Upgrades the edition of the installed product. Requires that either the + ProductKey or the Edition parameter is also assigned. Default is $false. + + .PARAMETER VersionUpgrade + Upgrades installed product version, if the major product version of the + source executable is higher than the major current version. Requires that + either the ProductKey or the Edition parameter is also assigned. Default + is $false. + + Not used in Set-TargetResource. The default is that the installation + does upgrade. This variable is only used in Test-TargetResource to return + false if the major version is different. + + .PARAMETER Edition + Sets the custom free edition. + { 'Development' | 'Evaluation' | 'ExpressAdvanced' } + + .PARAMETER LogPath + Specifies the setup log file location, e.g. 'log.txt'. By default, log + files are created under %TEMP%. + + .PARAMETER InstallFolder + Sets the install folder, e.g. 'C:\Program Files\SSRS'. Default value is + 'C:\Program Files\Microsoft SQL Server Reporting Services'. + + .PARAMETER SetupProcessTimeout + The timeout, in seconds, to wait for the setup process to finish. + Default value is 7200 seconds (2 hours). If the setup process does not + finish before this time, and error will be thrown. +#> +function Set-TargetResource +{ + <# + Suppressing this rule because $global:DSCMachineStatus is used to trigger + a Restart, either by force or when there are pending changes. + #> + [System.Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidGlobalVars', '')] + <# + Suppressing this rule because $global:DSCMachineStatus is only set, + never used (by design of Desired State Configuration). + #> + [System.Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSUseDeclaredVarsMoreThanAssignments', '', Scope='Function', Target='DSCMachineStatus')] + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [ValidateSet('SSRS')] + [System.String] + $InstanceName, + + [Parameter(Mandatory = $true)] + [ValidateSet('Yes')] + [System.String] + $IAcceptLicenseTerms, + + [Parameter(Mandatory = $true)] + [System.String] + $SourcePath, + + [Parameter()] + [ValidateSet('Install','Uninstall')] + [System.String] + $Action = 'Install', + + [Parameter()] + [System.Management.Automation.PSCredential] + $SourceCredential, + + [Parameter()] + [System.Boolean] + $SuppressRestart, + + [Parameter()] + [System.String] + $ProductKey, + + [Parameter()] + [System.Boolean] + $ForceRestart, + + [Parameter()] + [System.Boolean] + $EditionUpgrade, + + [Parameter()] + [System.Boolean] + $VersionUpgrade, + + [Parameter()] + [ValidateSet('Development','Evaluation','ExpressAdvanced')] + [System.String] + $Edition, + + [Parameter()] + [System.String] + $LogPath, + + [Parameter()] + [System.String] + $InstallFolder, + + [Parameter()] + [System.UInt32] + $SetupProcessTimeout = 7200 + ) + + # Must either choose ProductKey or Edition, not both. + if ($Action -eq 'Install' -and $PSBoundParameters.ContainsKey('Edition') -and $PSBoundParameters.ContainsKey('ProductKey')) + { + $errorMessage = $script:localizedData.EditionInvalidParameter + New-InvalidArgumentException -ArgumentName 'Edition, ProductKey' -Message $errorMessage + } + + # Must either choose ProductKey or Edition, not none. + if ($Action -eq 'Install' -and -not $PSBoundParameters.ContainsKey('Edition') -and -not $PSBoundParameters.ContainsKey('ProductKey')) + { + $errorMessage = $script:localizedData.EditionMissingParameter + New-InvalidArgumentException -ArgumentName 'Edition, ProductKey' -Message $errorMessage + } + + if (-not (Test-Path -Path $SourcePath) -or (Get-Item -Path $SourcePath).Extension -ne '.exe') + { + $errorMessage = $script:localizedData.SourcePathNotFound -f $SourcePath + New-InvalidArgumentException -ArgumentName 'SourcePath' -Message $errorMessage + } + + $InstanceName = $InstanceName.ToUpper() + + $SourcePath = [Environment]::ExpandEnvironmentVariables($SourcePath) + + $parametersToEvaluateTrailingSlash = @( + 'SourcePath', + 'InstallFolder' + ) + + # Making sure paths are correct. + foreach ($parameterName in $parametersToEvaluateTrailingSlash) + { + if ($PSBoundParameters.ContainsKey($parameterName)) + { + $parameterValue = Get-Variable -Name $parameterName -ValueOnly + $formattedPath = Format-Path -Path $parameterValue -TrailingSlash + Set-Variable -Name $parameterName -Value $formattedPath + } + } + + if ($SourceCredential) + { + $executableParentFolder = Split-Path -Path $SourcePath -Parent + $executableFileName = Split-Path -Path $SourcePath -Leaf + + $invokeInstallationMediaCopyParameters = @{ + SourcePath = $executableParentFolder + SourceCredential = $SourceCredential + PassThru = $true + } + + $newExecutableParentFolder = Invoke-InstallationMediaCopy @invokeInstallationMediaCopyParameters + + # Switch SourcePath to point to the new local location. + $SourcePath = Join-Path -Path $newExecutableParentFolder -ChildPath $executableFileName + } + + Write-Verbose -Message ($script:localizedData.UsingExecutable -f $SourcePath) + + $setupArguments = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + } + + if ($Action -eq 'Install') + { + $setupArguments += @{ + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + } + } + else + { + $setupArguments += @{ + 'uninstall' = [System.Management.Automation.SwitchParameter] $true + } + } + + <# + This is a list of parameters that are allowed to be translated into + arguments. + #> + $allowedParametersAsArguments = @( + 'ProductKey' + 'SuppressRestart' + 'EditionUpgrade' + 'Edition' + 'LogPath' + 'InstallFolder' + ) + + $argumentParameters = $PSBoundParameters.Keys | Where-Object -FilterScript { + $_ -in $allowedParametersAsArguments + } + + <# + Handle translation between parameter name and argument name. + Also making sure using the correct casing, e.g. 'log' and not 'Log'. + #> + switch ($argumentParameters) + { + 'ProductKey' + { + $setupArguments += @{ + 'PID' = $ProductKey + } + } + + 'SuppressRestart' + { + $setupArguments += @{ + 'norestart' = [System.Management.Automation.SwitchParameter] $true + } + } + + 'EditionUpgrade' + { + $setupArguments += @{ + 'EditionUpgrade' = [System.Management.Automation.SwitchParameter] $true + } + } + + 'Edition' + { + $setupArguments += @{ + 'Edition' = Convert-EditionName -Name $Edition + } + } + + 'LogPath' + { + $setupArguments += @{ + 'log' = $LogPath + } + } + + default + { + $setupArguments += @{ + $_ = Get-Variable -Name $_ -ValueOnly + } + } + } + + # Build the argument string to be passed to setup + $argumentString = '' + foreach ($currentSetupArgument in $setupArguments.GetEnumerator()) + { + # Arrays are handled specially + if ($currentSetupArgument.Value -is [System.Management.Automation.SwitchParameter]) + { + $argumentString += '/{0}' -f $currentSetupArgument.Key + } + else + { + $argumentString += '/{0}={1}' -f $currentSetupArgument.Key, $currentSetupArgument.Value + } + + # Add a space between arguments. + $argumentString += ' ' + } + + # Trim whitespace at start and end of string. + $argumentString = $argumentString.Trim() + + # Save the arguments for the log output + $logOutput = $argumentString + + # Replace sensitive values for verbose output + if ($PSBoundParameters.ContainsKey('ProductKey')) + { + $logOutput = $logOutput -replace $ProductKey, '*****-*****-*****-*****-*****' + } + + Write-Verbose -Message ($script:localizedData.SetupArguments -f $logOutput) + + <# + This handles when PsDscRunAsCredential is set, or running + as the SYSTEM account. + #> + + $startProcessParameters = @{ + FilePath = $SourcePath + ArgumentList = $argumentString + Timeout = $SetupProcessTimeout + } + + $processExitCode = Start-SqlSetupProcess @startProcessParameters + + Write-Verbose -Message ($script:localizedData.SetupExitMessage -f $processExitCode) + + if ($processExitCode -eq 0) + { + Write-Verbose -Message ($script:localizedData.SetupSuccessful -f $script:localizedData.$Action) + } + elseif ($processExitCode -eq 3010) + { + Write-Warning -Message ($script:localizedData.SetupSuccessfulRestartRequired -f $script:localizedData.$Action) + + $global:DSCMachineStatus = 1 + } + else + { + if ($PSBoundParameters.ContainsKey('LogPath')) + { + $errorMessage = $script:localizedData.SetupFailedWithLog -f $LogPath + } + else + { + $errorMessage = $script:localizedData.SetupFailed + } + + New-InvalidResultException -Message $errorMessage + } + + <# + If ForceRestart is set it will always restart, and override SuppressRestart. + If SuppressRestart is set it will always override any pending restart. + #> + if ($ForceRestart) + { + $global:DSCMachineStatus = 1 + } + elseif ($global:DSCMachineStatus -eq 1 -and $SuppressRestart) + { + # Suppressing restart to make sure the node is not restarted. + $global:DSCMachineStatus = 0 + + Write-Verbose -Message $script:localizedData.SuppressRestart + } + elseif (-not $SuppressRestart -and (Test-PendingRestart)) + { + $global:DSCMachineStatus = 1 + } + + if ($global:DSCMachineStatus -eq 1) + { + Write-Verbose -Message $script:localizedData.Restart + } +} + +<# + .SYNOPSIS + Tests if the Microsoft SQL Server Reporting Service instance is installed. + + .PARAMETER InstanceName + Name of the Microsoft SQL Server Reporting Service instance to installed. + This can only be set to 'SSRS'. { 'SSRS' } + + .PARAMETER IAcceptLicenseTerms + Accept licens terms. This must be set to 'Yes'. { 'Yes' } + + .PARAMETER SourcePath + The path to the installation media file to be used for installation, + e.g an UNC path to a shared resource. Environment variables can be used + in the path. + + .PARAMETER Action + The action to be performed. Default value is 'Install' which performs + either install or upgrade. + { *Install* | Uninstall } + + .PARAMETER SourceCredential + Credentials used to access the path set in the parameter 'SourcePath'. + + .PARAMETER SuppressRestart + Suppresses any attempts to restart. + + .PARAMETER ProductKey + Sets the custom license key, e.g. '12345-12345-12345-12345-12345'. + + .PARAMETER ForceRestart + Forces a restart after installation is finished. + + .PARAMETER EditionUpgrade + Upgrades the edition of the installed product. Requires that either the + ProductKey or the Edition parameter is also assigned. Default is $false. + + .PARAMETER VersionUpgrade + Upgrades installed product version, if the major product version of the + source executable is higher than the major current version. Requires that + either the ProductKey or the Edition parameter is also assigned. Default + is $false. + + .PARAMETER Edition + Sets the custom free edition. + { 'Development' | 'Evaluation' | 'ExpressAdvanced' } + + .PARAMETER LogPath + Specifies the setup log file location, e.g. 'log.txt'. By default, log + files are created under %TEMP%. + + .PARAMETER InstallFolder + Sets the install folder, e.g. 'C:\Program Files\SSRS'. Default value is + 'C:\Program Files\Microsoft SQL Server Reporting Services'. + + .PARAMETER SetupProcessTimeout + The timeout, in seconds, to wait for the setup process to finish. + Default value is 7200 seconds (2 hours). If the setup process does not + finish before this time, and error will be thrown. +#> +function Test-TargetResource +{ + [CmdletBinding()] + [OutputType([System.Boolean])] + param + ( + [Parameter(Mandatory = $true)] + [ValidateSet('SSRS')] + [System.String] + $InstanceName, + + [Parameter(Mandatory = $true)] + [ValidateSet('Yes')] + [System.String] + $IAcceptLicenseTerms, + + [Parameter(Mandatory = $true)] + [System.String] + $SourcePath, + + [Parameter()] + [ValidateSet('Install','Uninstall')] + [System.String] + $Action = 'Install', + + [Parameter()] + [System.Management.Automation.PSCredential] + $SourceCredential, + + [Parameter()] + [System.Boolean] + $SuppressRestart, + + [Parameter()] + [System.String] + $ProductKey, + + [Parameter()] + [System.Boolean] + $ForceRestart, + + [Parameter()] + [System.Boolean] + $EditionUpgrade, + + [Parameter()] + [System.Boolean] + $VersionUpgrade, + + [Parameter()] + [ValidateSet('Development','Evaluation','ExpressAdvanced')] + [System.String] + $Edition, + + [Parameter()] + [System.String] + $LogPath, + + [Parameter()] + [System.String] + $InstallFolder, + + [Parameter()] + [System.UInt32] + $SetupProcessTimeout = 7200 + ) + + Write-Verbose -Message ( + $script:localizedData.TestingConfiguration + ) + + $getTargetResourceParameters = @{ + InstanceName = $InstanceName + IAcceptLicenseTerms = $IAcceptLicenseTerms + SourcePath = $SourcePath + } + + $getTargetResourceResult = Get-TargetResource @getTargetResourceParameters + + $returnValue = $false + + <# + We determine if the Microsoft SQL Server Reporting Service instance is + installed if the instance name is found in the registry. + #> + if ($Action -eq 'Install') + { + $fileVersion = Get-FileProductVersion -Path $SourcePath + + if ($getTargetResourceResult.InstanceName) + { + $installedVersion = [System.Version] $getTargetResourceResult.CurrentVersion + + # The major version is evaluated if VersionUpgrade is set to $true + if (-not $VersionUpgrade -or ($VersionUpgrade -and $installedVersion -ge $fileVersion)) + { + $returnValue = $true + } + else + { + Write-Verbose -Message ( + $script:localizedData.WrongVersionFound ` + -f $fileVersion.ToString(), $installedVersion.ToString() + ) + } + } + else + { + Write-Verbose -Message ( + $script:localizedData.MissingVersion ` + -f $fileVersion.ToString() + ) + } + } + + if ($Action -eq 'Uninstall' -and $null -eq $getTargetResourceResult.InstanceName) + { + $returnValue = $true + } + + return $returnValue +} + +<# + .SYNOPSIS + Converts between the edition names used by the resource and the + installation media. + + .PARAMETER Name + The edition name to convert. + + .OUTPUTS + Returns the equivalent name of what was provided in the parameter Name. + For example, if Name is set to 'Dev', the cmdlet returns 'Development'. + If Name is set to 'Development', the cmdlet returns 'Dev'. +#> +function Convert-EditionName +{ + [CmdletBinding()] + [OutputType([System.String])] + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $Name + ) + + switch ($Name) + { + # Resource edition names + 'Development' + { + $convertEditionNameResult = 'Dev' + } + + 'Evaluation' + { + $convertEditionNameResult = 'Eval' + } + + 'ExpressAdvanced' + { + $convertEditionNameResult = 'ExprAdv' + } + + # Installation media edition names + 'Dev' + { + $convertEditionNameResult = 'Development' + } + + 'Eval' + { + $convertEditionNameResult = 'Evaluation' + } + + 'ExprAdv' + { + $convertEditionNameResult = 'ExpressAdvanced' + } + } + + return $convertEditionNameResult +} + +<# + .SYNOPSIS + Gets the product version of a executable. + + .PARAMETER Path + The path to the executable to return product version for. + + .OUTPUTS + Returns the product version as [System.Version] type. +#> +function Get-FileProductVersion +{ + [CmdletBinding()] + [OutputType([System.Version])] + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $Path + ) + + return [System.Version] (Get-Item -Path $Path).VersionInfo.ProductVersion +} diff --git a/DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.schema.mof b/DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.schema.mof new file mode 100644 index 000000000..95378034d --- /dev/null +++ b/DSCResources/MSFT_SqlRSSetup/MSFT_SqlRSSetup.schema.mof @@ -0,0 +1,21 @@ +[ClassVersion("1.0.0.0"), FriendlyName("SqlRSSetup")] +class MSFT_SqlRSSetup : OMI_BaseResource +{ + [Key, Description("Name of the Microsoft SQL Server Reporting Service instance to installed. This can only be set to 'SSRS'. { 'SSRS' }"), ValueMap{"SSRS"}, Values{"SSRS"}] String InstanceName; + [Required, Description("Accept licens terms. This must be set to 'Yes'. { 'Yes' }"), ValueMap{"Yes"}, Values{"Yes"}] String IAcceptLicenseTerms; + [Required, Description("The path to the installation media file to be used for installation, e.g an UNC path to a shared resource. Environment variables can be used in the path.")] String SourcePath; + [Write, Description("The action to be performed. Default value is 'Install' which performs either install or upgrade. { *Install* | Uninstall }"), ValueMap{"Install","Uninstall"}, Values{"Install","Uninstall"}] String Action; + [Write, EmbeddedInstance("MSFT_Credential"), Description("Credentials used to access the path set in the parameter 'SourcePath'.")] String SourceCredential; + [Write, Description("Suppresses any attempts to restart.")] Boolean SuppressRestart; + [Write, Description("Sets the custom license key, e.g. '12345-12345-12345-12345-12345'.")] String ProductKey; + [Write, Description("Forces a restart after installation is finished.")] Boolean ForceRestart; + [Write, Description("Upgrades the edition of the installed product. Requires that either the ProductKey or the Edition parameter is also assigned. Default is $false.")] Boolean EditionUpgrade; + [Write, Description("Upgrades installed product version, if the major product version of the source executable is higher than the major current version. Requires that either the ProductKey or the Edition parameter is also assigned. Default is $false.")] Boolean VersionUpgrade; + [Write, Description("Sets the custom free edition. { 'Development' | 'Evaluation' | 'ExpressAdvanced' }"), ValueMap{"Development","Evaluation","ExpressAdvanced"}, Values{"Development","Evaluation","ExpressAdvanced"}] String Edition; + [Write, Description("Specifies the setup log file location, e.g. 'log.txt'. By default, log files are created under %TEMP%.")] String LogPath; + [Write, Description("Sets the install folder, e.g. 'C:\\Program Files\\SSRS'. Default value is 'C:\\Program Files\\Microsoft SQL Server Reporting Services'.")] String InstallFolder; + [Write, Description("The timeout, in seconds, to wait for the setup process to finish. Default value is 7200 seconds (2 hours). If the setup process does not finish before this time, and error will be thrown.")] UInt32 SetupProcessTimeout; + [Read, Description("Returns the path to error dump log files.")] String ErrorDumpDirectory; + [Read, Description("Returns the current version of the installed Microsoft SQL Server Reporting Service instance.")] String CurrentVersion; + [Read, Description("Returns the current name of the Microsoft SQL Server Reporting Service instance Windows service.")] String ServiceName; +}; diff --git a/DSCResources/MSFT_SqlRSSetup/en-US/MSFT_SqlRSSetup.strings.psd1 b/DSCResources/MSFT_SqlRSSetup/en-US/MSFT_SqlRSSetup.strings.psd1 new file mode 100644 index 000000000..f29d8ea1b --- /dev/null +++ b/DSCResources/MSFT_SqlRSSetup/en-US/MSFT_SqlRSSetup.strings.psd1 @@ -0,0 +1,25 @@ +# Localized resources for SqlSetup + +ConvertFrom-StringData @' + TestingConfiguration = Determines if the Microsoft SQL Server Reporting Service instance is installed. + FoundInstance = Found Microsoft SQL Server Reporting Service instance named '{0}'. + InstanceNotFound = Could not find a Microsoft SQL Server Reporting Service instance. + Install = Install + Uninstall = Uninstall + UsingExecutable = Using executable at '{0}'. + SetupArguments = Starting executable using the arguments: {0} + SetupExitMessage = Executable exited with code '{0}'. + SetupSuccessfulRestartRequired = {0} finished successfully, but a restart is required. + SetupFailed = Please see the log file in the %TEMP% folder. + SetupFailedWithLog = Please see the log file '{0}'. If not path was provided, the default path for log files is %TEMP%. + SetupSuccessful = {0} finished successfully. + Restart = Restarting the target node. + SuppressRestart = Suppressing restart of target node. + EditionInvalidParameter = Both the parameters Edition and ProductKey was specified. Only either parameter Edition or ProductKey is allowed. + EditionMissingParameter = Neither the parameters Edition and ProductKey was specified. + SourcePathNotFound = The source path '{0}' does not exist, or the path does not specify an executable file. + VersionFound = The Microsoft SQL Server Reporting Service instance is version '{0}'. + PackageNotFound = Could not determine the version of the Microsoft SQL Server Reporting Service instance. + WrongVersionFound = Expected version '{0}', but version '{1}' is installed. + MissingVersion = Expected version '{0}' to be installed. +'@ diff --git a/DSCResources/MSFT_SqlScript/MSFT_SqlScript.psm1 b/DSCResources/MSFT_SqlScript/MSFT_SqlScript.psm1 index e437b13c2..f2abc4936 100644 --- a/DSCResources/MSFT_SqlScript/MSFT_SqlScript.psm1 +++ b/DSCResources/MSFT_SqlScript/MSFT_SqlScript.psm1 @@ -1,5 +1,11 @@ -$script:currentPath = Split-Path -Path $MyInvocation.MyCommand.Path -Parent -Import-Module -Name (Join-Path -Path (Split-Path -Path (Split-Path -Path $script:currentPath -Parent) -Parent) -ChildPath 'SqlServerDscHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlScriptQuery/MSFT_SqlScriptQuery.psm1 b/DSCResources/MSFT_SqlScriptQuery/MSFT_SqlScriptQuery.psm1 index f788e2e02..6a88c1b54 100644 --- a/DSCResources/MSFT_SqlScriptQuery/MSFT_SqlScriptQuery.psm1 +++ b/DSCResources/MSFT_SqlScriptQuery/MSFT_SqlScriptQuery.psm1 @@ -1,5 +1,11 @@ -$script:currentPath = Split-Path -Path $MyInvocation.MyCommand.Path -Parent -Import-Module -Name (Join-Path -Path (Split-Path -Path (Split-Path -Path $script:currentPath -Parent) -Parent) -ChildPath 'SqlServerDscHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlServerConfiguration/MSFT_SqlServerConfiguration.psm1 b/DSCResources/MSFT_SqlServerConfiguration/MSFT_SqlServerConfiguration.psm1 index ce273a68b..789025a60 100644 --- a/DSCResources/MSFT_SqlServerConfiguration/MSFT_SqlServerConfiguration.psm1 +++ b/DSCResources/MSFT_SqlServerConfiguration/MSFT_SqlServerConfiguration.psm1 @@ -1,9 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlServerConfiguration' diff --git a/DSCResources/MSFT_SqlServerDatabaseMail/MSFT_SqlServerDatabaseMail.psm1 b/DSCResources/MSFT_SqlServerDatabaseMail/MSFT_SqlServerDatabaseMail.psm1 index efc7b7ac9..9c6bbacae 100644 --- a/DSCResources/MSFT_SqlServerDatabaseMail/MSFT_SqlServerDatabaseMail.psm1 +++ b/DSCResources/MSFT_SqlServerDatabaseMail/MSFT_SqlServerDatabaseMail.psm1 @@ -1,9 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlServerDatabaseMail' @@ -702,7 +704,7 @@ function Test-TargetResource if ($Ensure -eq 'Present') { - $returnValue = Test-SQLDscParameterState ` + $returnValue = Test-DscParameterState ` -CurrentValues $getTargetResourceResult ` -DesiredValues $PSBoundParameters ` -ValuesToCheck @( diff --git a/DSCResources/MSFT_SqlServerEndpoint/MSFT_SqlServerEndpoint.psm1 b/DSCResources/MSFT_SqlServerEndpoint/MSFT_SqlServerEndpoint.psm1 index c55f2d9f3..927d250f2 100644 --- a/DSCResources/MSFT_SqlServerEndpoint/MSFT_SqlServerEndpoint.psm1 +++ b/DSCResources/MSFT_SqlServerEndpoint/MSFT_SqlServerEndpoint.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS Returns the current state of the endpoint. diff --git a/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 b/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 index 92f3b5af2..5c8b5a060 100644 --- a/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 +++ b/DSCResources/MSFT_SqlServerEndpointPermission/MSFT_SqlServerEndpointPermission.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS Returns the current state of the permissions for the principal (login). diff --git a/DSCResources/MSFT_SqlServerEndpointState/MSFT_SqlServerEndpointState.psm1 b/DSCResources/MSFT_SqlServerEndpointState/MSFT_SqlServerEndpointState.psm1 index 59b6bf88b..35df74894 100644 --- a/DSCResources/MSFT_SqlServerEndpointState/MSFT_SqlServerEndpointState.psm1 +++ b/DSCResources/MSFT_SqlServerEndpointState/MSFT_SqlServerEndpointState.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS Returns the current state of an endpoint. diff --git a/DSCResources/MSFT_SqlServerLogin/MSFT_SqlServerLogin.psm1 b/DSCResources/MSFT_SqlServerLogin/MSFT_SqlServerLogin.psm1 index 5d041fa33..1c7269d0f 100644 --- a/DSCResources/MSFT_SqlServerLogin/MSFT_SqlServerLogin.psm1 +++ b/DSCResources/MSFT_SqlServerLogin/MSFT_SqlServerLogin.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS @@ -434,24 +439,24 @@ function Test-TargetResource #> if ((Find-ExceptionByNumber -ExceptionToSearch $_.Exception -ErrorNumber 18470)) { - New-VerboseMessage -Message "Password valid, but '$Name' is disabled." + New-VerboseMessage -Message "Password valid, but '$Name' is disabled." } elseif ((Find-ExceptionByNumber -ExceptionToSearch $_.Exception -ErrorNumber 18456)) { New-VerboseMessage -Message $_.Exception.message - + # The password was not correct, password validation failed $testPassed = $false } - else + else { New-VerboseMessage -Message "Unknown error: $($_.Exception.message)" - + # Something else went wrong, rethrow error throw } } - else + else { New-VerboseMessage -Message "Password validation failed for the login '$Name'." $testPassed = $false diff --git a/DSCResources/MSFT_SqlServerMaxDop/MSFT_SqlServerMaxDop.psm1 b/DSCResources/MSFT_SqlServerMaxDop/MSFT_SqlServerMaxDop.psm1 index be82a50cb..e7f84fcc9 100644 --- a/DSCResources/MSFT_SqlServerMaxDop/MSFT_SqlServerMaxDop.psm1 +++ b/DSCResources/MSFT_SqlServerMaxDop/MSFT_SqlServerMaxDop.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS This function gets the max degree of parallelism server configuration option. diff --git a/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 b/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 index 6b15959e6..c792f7791 100644 --- a/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 +++ b/DSCResources/MSFT_SqlServerMemory/MSFT_SqlServerMemory.psm1 @@ -1,4 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) -ChildPath 'SqlServerDscHelper.psm1') -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlServerNetwork/MSFT_SqlServerNetwork.psm1 b/DSCResources/MSFT_SqlServerNetwork/MSFT_SqlServerNetwork.psm1 index 98da5b0e6..44d99e3a1 100644 --- a/DSCResources/MSFT_SqlServerNetwork/MSFT_SqlServerNetwork.psm1 +++ b/DSCResources/MSFT_SqlServerNetwork/MSFT_SqlServerNetwork.psm1 @@ -1,7 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') -Force -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') # Load localized string data $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlServerNetwork' diff --git a/DSCResources/MSFT_SqlServerPermission/MSFT_SqlServerPermission.psm1 b/DSCResources/MSFT_SqlServerPermission/MSFT_SqlServerPermission.psm1 index fbe9f3a99..e458479a7 100644 --- a/DSCResources/MSFT_SqlServerPermission/MSFT_SqlServerPermission.psm1 +++ b/DSCResources/MSFT_SqlServerPermission/MSFT_SqlServerPermission.psm1 @@ -1,6 +1,12 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') + <# .SYNOPSIS Returns the current state of the permissions for the principal (login). diff --git a/DSCResources/MSFT_SqlServerRole/MSFT_SqlServerRole.psm1 b/DSCResources/MSFT_SqlServerRole/MSFT_SqlServerRole.psm1 index 59ae0e17d..5cf77dd17 100644 --- a/DSCResources/MSFT_SqlServerRole/MSFT_SqlServerRole.psm1 +++ b/DSCResources/MSFT_SqlServerRole/MSFT_SqlServerRole.psm1 @@ -1,8 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlServerRole' diff --git a/DSCResources/MSFT_SqlServerSecureConnection/MSFT_SqlServerSecureConnection.psm1 b/DSCResources/MSFT_SqlServerSecureConnection/MSFT_SqlServerSecureConnection.psm1 index e14fd931c..6eb02e3a8 100644 --- a/DSCResources/MSFT_SqlServerSecureConnection/MSFT_SqlServerSecureConnection.psm1 +++ b/DSCResources/MSFT_SqlServerSecureConnection/MSFT_SqlServerSecureConnection.psm1 @@ -1,5 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlServerSecureConnection' diff --git a/DSCResources/MSFT_SqlServiceAccount/MSFT_SqlServiceAccount.psm1 b/DSCResources/MSFT_SqlServiceAccount/MSFT_SqlServiceAccount.psm1 index 463654577..65398dad8 100644 --- a/DSCResources/MSFT_SqlServiceAccount/MSFT_SqlServiceAccount.psm1 +++ b/DSCResources/MSFT_SqlServiceAccount/MSFT_SqlServiceAccount.psm1 @@ -1,8 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlServiceAccount' diff --git a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 index e26c91724..25c123829 100644 --- a/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 +++ b/DSCResources/MSFT_SqlSetup/MSFT_SqlSetup.psm1 @@ -1,8 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent) ` - -ChildPath 'CommonResourceHelper.psm1') +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') $script:localizedData = Get-LocalizedData -ResourceName 'MSFT_SqlSetup' @@ -89,13 +92,7 @@ function Get-TargetResource if ($SourceCredential) { - $newSmbMappingParameters = @{ - RemotePath = $SourcePath - UserName = "$($SourceCredential.GetNetworkCredential().Domain)\$($SourceCredential.GetNetworkCredential().UserName)" - Password = $($SourceCredential.GetNetworkCredential().Password) - } - - $null = New-SmbMapping @newSmbMappingParameters + Connect-UncPath -RemotePath $SourcePath -SourceCredential $SourceCredential } $pathToSetupExecutable = Join-Path -Path $SourcePath -ChildPath 'setup.exe' @@ -106,7 +103,7 @@ function Get-TargetResource if ($SourceCredential) { - Remove-SmbMapping -RemotePath $SourcePath -Force + Disconnect-UncPath -RemotePath $SourcePath } if ($InstanceName -eq 'MSSQLSERVER') @@ -1043,24 +1040,14 @@ function Set-TargetResource 'UpdateSource' ) - # Remove trailing slash ('\') from paths + # Making sure paths are correct. foreach ($parameterName in $parametersToEvaluateTrailingSlash) { if ($PSBoundParameters.ContainsKey($parameterName)) { $parameterValue = Get-Variable -Name $parameterName -ValueOnly - - # Trim backslash, but only if the path contains a full path and not just a qualifier. - if ($parameterValue -and $parameterValue -notmatch '^[a-zA-Z]:\\$') - { - Set-Variable -Name $parameterName -Value $parameterValue.TrimEnd('\') - } - - # If the path only contains a qualifier but no backslash ('M:'), then a backslash is added ('M:\'). - if ($parameterValue -match '^[a-zA-Z]:$') - { - Set-Variable -Name $parameterName -Value "$parameterValue\" - } + $formattedPath = Format-Path -Path $parameterValue -TrailingSlash + Set-Variable -Name $parameterName -Value $formattedPath } } @@ -1068,29 +1055,13 @@ function Set-TargetResource if ($SourceCredential) { - $newSmbMappingParameters = @{ - RemotePath = $SourcePath - UserName = "$($SourceCredential.GetNetworkCredential().Domain)\$($SourceCredential.GetNetworkCredential().UserName)" - Password = $($SourceCredential.GetNetworkCredential().Password) + $invokeInstallationMediaCopyParameters = @{ + SourcePath = $SourcePath + SourceCredential = $SourceCredential + PassThru = $true } - $null = New-SmbMapping @newSmbMappingParameters - - # Create a destination folder so the media files aren't written to the root of the Temp folder. - $mediaDestinationFolder = Split-Path -Path $SourcePath -Leaf - if (-not $mediaDestinationFolder ) - { - $mediaDestinationFolder = New-Guid | Select-Object -ExpandProperty Guid - } - - $mediaDestinationPath = Join-Path -Path (Get-TemporaryFolder) -ChildPath $mediaDestinationFolder - - Write-Verbose -Message ($script:localizedData.RobocopyIsCopying -f $SourcePath, $mediaDestinationPath) - Copy-ItemWithRobocopy -Path $SourcePath -DestinationPath $mediaDestinationPath - - Remove-SmbMapping -RemotePath $SourcePath -Force - - $SourcePath = $mediaDestinationPath + $SourcePath = Invoke-InstallationMediaCopy @invokeInstallationMediaCopyParameters } $pathToSetupExecutable = Join-Path -Path $SourcePath -ChildPath 'setup.exe' @@ -1101,20 +1072,23 @@ function Set-TargetResource # Determine features to install $featuresToInstall = '' - foreach ($feature in $Features.Split(',')) - { - # Given that all the returned features are uppercase, make sure that the feature to search for is also uppercase - $feature = $feature.ToUpper(); + $featuresArray = $Features -split ',' + + foreach ($feature in $featuresArray) + { if (($sqlVersion -in ('13','14')) -and ($feature -in ('ADV_SSMS','SSMS'))) { $errorMessage = $script:localizedData.FeatureNotSupported -f $feature New-InvalidOperationException -Message $errorMessage } - if (-not ($getTargetResourceResult.Features.Contains($feature))) + $foundFeaturesArray = $getTargetResourceResult.Features -split ',' + + if ($feature -notin $foundFeaturesArray) { - $featuresToInstall += "$feature," + # Must make sure the feature names are provided in upper-case. + $featuresToInstall += '{0},' -f $feature.ToUpper() } else { @@ -1577,13 +1551,13 @@ function Set-TargetResource # Logic added as a fix for Issue#1254 SqlSetup:Fails when a root directory is specified if($currentSetupArgument.Value -match '^[a-zA-Z]:\\$') { - $setupArgumentValue = $currentSetupArgument.Value + $setupArgumentValue = $currentSetupArgument.Value } - else + else { - $setupArgumentValue = '"{0}"' -f $currentSetupArgument.Value + $setupArgumentValue = '"{0}"' -f $currentSetupArgument.Value } - + } } @@ -1654,7 +1628,7 @@ function Set-TargetResource Write-Verbose -Message $setupExitMessageSuccessful } - if ($ForceReboot -or ($null -ne (Get-ItemProperty -Path 'HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager' -Name 'PendingFileRenameOperations' -ErrorAction SilentlyContinue))) + if ($ForceReboot -or (Test-PendingRestart)) { if (-not ($SuppressReboot)) { @@ -2142,18 +2116,25 @@ function Test-TargetResource $boundParameters = $PSBoundParameters $getTargetResourceResult = Get-TargetResource @getTargetResourceParameters - Write-Verbose -Message ($script:localizedData.FeaturesFound -f $($getTargetResourceResult.Features)) + if ($null -eq $getTargetResourceResult.Features -or $getTargetResourceResult.Features -eq '') + { + Write-Verbose -Message $script:localizedData.NoFeaturesFound + } + else + { + Write-Verbose -Message ($script:localizedData.FeaturesFound -f $getTargetResourceResult.Features) + } $result = $true - if ($getTargetResourceResult.Features ) + if ($getTargetResourceResult.Features) { - foreach ($feature in $Features.Split(",")) - { - # Given that all the returned features are uppercase, make sure that the feature to search for is also uppercase - $feature = $feature.ToUpper(); + $featuresArray = $Features -split ',' + $foundFeaturesArray = $getTargetResourceResult.Features -split ',' - if(!($getTargetResourceResult.Features.Contains($feature))) + foreach ($feature in $featuresArray) + { + if ($feature -notin $foundFeaturesArray) { Write-Verbose -Message ($script:localizedData.UnableToFindFeature -f $feature, $($getTargetResourceResult.Features)) $result = $false @@ -2233,114 +2214,6 @@ function Get-FirstItemPropertyValue return $registryPropertyValue } -<# - .SYNOPSIS - Copy folder structure using Robocopy. Every file and folder, including empty ones are copied. - - .PARAMETER Path - Source path to be copied. - - .PARAMETER DestinationPath - The path to the destination. -#> -function Copy-ItemWithRobocopy -{ - [CmdletBinding()] - param - ( - [Parameter(Mandatory = $true)] - [ValidateNotNullOrEmpty()] - [System.String] - $Path, - - [Parameter(Mandatory = $true)] - [ValidateNotNullOrEmpty()] - [System.String] - $DestinationPath - ) - $quotedPath = '"{0}"' -f $Path - $quotedDestinationPath = '"{0}"' -f $DestinationPath - $robocopyExecutable = Get-Command -Name "Robocopy.exe" -ErrorAction Stop - - $robocopyArgumentSilent = '/njh /njs /ndl /nc /ns /nfl' - $robocopyArgumentCopySubDirectoriesIncludingEmpty = '/e' - $robocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource = '/purge' - - if ([System.Version]$robocopyExecutable.FileVersionInfo.ProductVersion -ge [System.Version]'6.3.9600.16384') - { - Write-Verbose -Message $script:localizedData.RobocopyUsingUnbufferedIo - - $robocopyArgumentUseUnbufferedIO = '/J' - } - else - { - Write-Verbose -Message $script:localizedData.RobocopyNotUsingUnbufferedIo - } - - $robocopyArgumentList = '{0} {1} {2} {3} {4} {5}' -f $quotedPath, - $quotedDestinationPath, - $robocopyArgumentCopySubDirectoriesIncludingEmpty, - $robocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource, - $robocopyArgumentUseUnbufferedIO, - $robocopyArgumentSilent - - $robocopyStartProcessParameters = @{ - FilePath = $robocopyExecutable.Name - ArgumentList = $robocopyArgumentList - } - - Write-Verbose -Message ($script:localizedData.RobocopyArguments -f $robocopyArgumentList ) - $robocopyProcess = Start-Process @robocopyStartProcessParameters -Wait -NoNewWindow -PassThru - - switch ($($robocopyProcess.ExitCode)) - { - {$_ -in 8, 16} - { - $errorMessage = $script:localizedData.RobocopyErrorCopying -f $_ - New-InvalidOperationException -Message $errorMessage - } - - {$_ -gt 7 } - { - $errorMessage = $script:localizedData.RobocopyFailuresCopying -f $_ - New-InvalidResultException -Message $errorMessage - } - - 1 - { - Write-Verbose -Message $script:localizedData.RobocopySuccessful - } - - 2 - { - Write-Verbose -Message $script:localizedData.RobocopyRemovedExtraFilesAtDestination - } - - 3 - { - Write-Verbose -Message $script:localizedData.RobocopySuccessfulAndRemovedExtraFilesAtDestination - } - - {$_ -eq 0 -or $null -eq $_ } - { - Write-Verbose -Message $script:localizedData.RobocopyAllFilesPresent - } - } -} - -<# - .SYNOPSIS - Returns the path of the current user's temporary folder. -#> -function Get-TemporaryFolder -{ - [CmdletBinding()] - [OutputType([System.String])] - param() - - return [IO.Path]::GetTempPath() -} - <# .SYNOPSIS Returns the decimal representation of an IP Addresses. @@ -2455,50 +2328,6 @@ function Get-ServiceAccountParameters return $parameters } -<# - .SYNOPSIS - Starts the SQL setup process. - - .PARAMETER FilePath - String containing the path to setup.exe. - - .PARAMETER ArgumentList - The arguments that should be passed to setup.exe. - - .PARAMETER Timeout - The timeout in seconds to wait for the process to finish. -#> -function Start-SqlSetupProcess -{ - param - ( - [Parameter(Mandatory = $true)] - [System.String] - $FilePath, - - [Parameter()] - [System.String] - $ArgumentList, - - [Parameter(Mandatory = $true)] - [System.UInt32] - $Timeout - ) - - $startProcessParameters = @{ - FilePath = $FilePath - ArgumentList = $ArgumentList - } - - $sqlSetupProcess = Start-Process @startProcessParameters -PassThru -NoNewWindow -ErrorAction Stop - - Write-Verbose -Message ($script:localizedData.StartSetupProcess -f $sqlSetupProcess.Id, $startProcessParameters.FilePath, $Timeout) - - Wait-Process -InputObject $sqlSetupProcess -Timeout $Timeout -ErrorAction Stop - - return $sqlSetupProcess.ExitCode -} - <# .SYNOPSIS Converts the start mode property returned by a Win32_Service CIM object to the resource properties *StartupType equivalent diff --git a/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 b/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 index ca1cd0878..fbce73d37 100644 --- a/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 +++ b/DSCResources/MSFT_SqlSetup/en-US/MSFT_SqlSetup.strings.psd1 @@ -57,20 +57,11 @@ ConvertFrom-StringData @' Reboot = Rebooting target node. SuppressReboot = Suppressing reboot of target node. TestFailedAfterSet = Test-TargetResource returned false after calling Set-TargetResource. - FeaturesFound = Features found: {0} + FeaturesFound = Found features already installed: {0} + NoFeaturesFound = No features are installed. UnableToFindFeature = Unable to find feature '{0}' among the installed features: '{1}'. EvaluatingClusterParameters = Clustered install, checking parameters. ClusterParameterIsNotInDesiredState = {0} '{1}' is not in the desired state for this cluster. - RobocopyUsingUnbufferedIo = Robocopy is using unbuffered I/O. - RobocopyNotUsingUnbufferedIo = Unbuffered I/O cannot be used due to incompatible version of Robocopy. - RobocopyArguments = Robocopy is started with the following arguments: {0} - RobocopyErrorCopying = Robocopy reported errors when copying files. Error code: {0}. - RobocopyFailuresCopying = Robocopy reported that failures occurred when copying files. Error code: {0}. - RobocopySuccessful = Robocopy copied files successfully - RobocopyRemovedExtraFilesAtDestination = Robocopy found files at the destination path that is not present at the source path, these extra files was remove at the destination path. - RobocopySuccessfulAndRemovedExtraFilesAtDestination = Robocopy copied files to destination successfully. Robocopy also found files at the destination path that is not present at the source path, these extra files was remove at the destination path. - RobocopyAllFilesPresent = Robocopy reported that all files already present. - StartSetupProcess = Started the process with id {0} using the path '{1}', and with a timeout value of {2} seconds. EvaluateMasterDataServicesFeature = Detecting Master Data Services (MDS) feature ({0}). MasterDataServicesFeatureFound = Master Data Services (MDS) feature detected. MasterDataServicesFeatureNotFound = Master Data Services (MDS) feature not detected. diff --git a/DSCResources/MSFT_SqlWaitForAG/MSFT_SqlWaitForAG.psm1 b/DSCResources/MSFT_SqlWaitForAG/MSFT_SqlWaitForAG.psm1 index 253dd405e..24b52a835 100644 --- a/DSCResources/MSFT_SqlWaitForAG/MSFT_SqlWaitForAG.psm1 +++ b/DSCResources/MSFT_SqlWaitForAG/MSFT_SqlWaitForAG.psm1 @@ -1,6 +1,11 @@ -Import-Module -Name (Join-Path -Path (Split-Path (Split-Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'SqlServerDscHelper.psm1') ` - -Force +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/DSCResources/MSFT_SqlWindowsFirewall/MSFT_SqlWindowsFirewall.psm1 b/DSCResources/MSFT_SqlWindowsFirewall/MSFT_SqlWindowsFirewall.psm1 index 40ac00119..513fe8e8d 100644 --- a/DSCResources/MSFT_SqlWindowsFirewall/MSFT_SqlWindowsFirewall.psm1 +++ b/DSCResources/MSFT_SqlWindowsFirewall/MSFT_SqlWindowsFirewall.psm1 @@ -1,5 +1,11 @@ -$script:currentPath = Split-Path -Path $MyInvocation.MyCommand.Path -Parent -Import-Module -Name (Join-Path -Path (Split-Path -Path (Split-Path -Path $script:currentPath -Parent) -Parent) -ChildPath 'SqlServerDscHelper.psm1') +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules' + +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + +$script:resourceHelperModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common' +Import-Module -Name (Join-Path -Path $script:resourceHelperModulePath -ChildPath 'DscResource.Common.psm1') <# .SYNOPSIS diff --git a/Examples/README.md b/Examples/README.md index 7bd74297a..9917443eb 100644 --- a/Examples/README.md +++ b/Examples/README.md @@ -21,6 +21,7 @@ These are the links to the examples for each individual resource. - [SqlDatabaseRecoveryModel](Resources/SqlDatabaseRecoveryModel) - [SqlDatabaseRole](Resources/SqlDatabaseRole) - [SqlRS](Resources/SqlRS) +- [SqlRSSetup](Resources/SqlRSSetup) - [SqlScript](Resources/SqlScript) - [SqlScriptQuery](Resources/SqlScriptQuery) - [SqlServerConfiguration](Resources/SqlServerConfiguration) diff --git a/Examples/Resources/SqlRSSetup/1-InstallReportingServices.ps1 b/Examples/Resources/SqlRSSetup/1-InstallReportingServices.ps1 new file mode 100644 index 000000000..da99e9f55 --- /dev/null +++ b/Examples/Resources/SqlRSSetup/1-InstallReportingServices.ps1 @@ -0,0 +1,31 @@ +<# + .EXAMPLE + This example shows how to install a Microsoft SQL Server Reporting Service + instance (2017 or newer). +#> +Configuration Example +{ + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.Management.Automation.PSCredential] + $SqlInstallCredential + ) + + Import-DscResource -ModuleName SqlServerDsc + + node localhost + { + SqlRSSetup 'InstallDefaultInstance' + { + InstanceName = 'SSRS' + IAcceptLicenseTerms = 'Yes' + SourcePath = 'C:\InstallMedia\SQLServerReportingServices.exe' + Edition = 'Development' + + PsDscRunAsCredential = $SqlInstallCredential + } + } +} diff --git a/Examples/Resources/SqlRSSetup/2-UninstallReportingServices.ps1 b/Examples/Resources/SqlRSSetup/2-UninstallReportingServices.ps1 new file mode 100644 index 000000000..4c7a4f61a --- /dev/null +++ b/Examples/Resources/SqlRSSetup/2-UninstallReportingServices.ps1 @@ -0,0 +1,33 @@ +<# + .EXAMPLE + This example shows how to install a Microsoft SQL Server Reporting Service + instance (2017 or newer). +#> +Configuration Example +{ + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.Management.Automation.PSCredential] + $SqlInstallCredential + ) + + Import-DscResource -ModuleName SqlServerDsc + + node localhost + { + SqlRSSetup 'InstallDefaultInstance' + { + InstanceName = 'SSRS' + SourcePath = 'C:\InstallMedia\SQLServerReportingServices.exe' + Action = 'Uninstall' + + # This needs to be set to although it is not used during uninstall. + IAcceptLicenseTerms = 'Yes' + + PsDscRunAsCredential = $SqlInstallCredential + } + } +} diff --git a/SqlServerDscHelper.psm1 b/Modules/DscResource.Common/DscResource.Common.psm1 similarity index 79% rename from SqlServerDscHelper.psm1 rename to Modules/DscResource.Common/DscResource.Common.psm1 index 8f3f682fc..203a37586 100644 --- a/SqlServerDscHelper.psm1 +++ b/Modules/DscResource.Common/DscResource.Common.psm1 @@ -1,9 +1,607 @@ -# Load Localization Data -Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot ` - -ChildPath 'DscResources') ` - -ChildPath 'CommonResourceHelper.psm1') +$script:modulesFolderPath = Split-Path -Path $PSScriptRoot -Parent -$script:localizedData = Get-LocalizedData -ResourceName 'SqlServerDscHelper' -ScriptRoot $PSScriptRoot +$script:localizationModulePath = Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper' +Import-Module -Name (Join-Path -Path $script:localizationModulePath -ChildPath 'DscResource.LocalizationHelper.psm1') + + +$script:localizedData = Get-LocalizedData -ResourceName 'DscResource.Common' -ScriptRoot $PSScriptRoot + +<# + .SYNOPSIS + This method is used to compare current and desired values for any DSC resource. + + .PARAMETER CurrentValues + This is hash table of the current values that are applied to the resource. + + .PARAMETER DesiredValues + This is a PSBoundParametersDictionary of the desired values for the resource. + + .PARAMETER ValuesToCheck + This is a list of which properties in the desired values list should be checked. + If this is empty then all values in DesiredValues are checked. +#> +function Test-DscParameterState +{ + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [System.Collections.Hashtable] + $CurrentValues, + + [Parameter(Mandatory = $true)] + [System.Object] + $DesiredValues, + + [Parameter()] + [System.Array] + $ValuesToCheck + ) + + $returnValue = $true + + if (($DesiredValues.GetType().Name -ne 'HashTable') ` + -and ($DesiredValues.GetType().Name -ne 'CimInstance') ` + -and ($DesiredValues.GetType().Name -ne 'PSBoundParametersDictionary')) + { + $errorMessage = $script:localizedData.PropertyTypeInvalidForDesiredValues -f $($DesiredValues.GetType().Name) + New-InvalidArgumentException -ArgumentName 'DesiredValues' -Message $errorMessage + } + + if (($DesiredValues.GetType().Name -eq 'CimInstance') -and ($null -eq $ValuesToCheck)) + { + $errorMessage = $script:localizedData.PropertyTypeInvalidForValuesToCheck + New-InvalidArgumentException -ArgumentName 'ValuesToCheck' -Message $errorMessage + } + + if (($null -eq $ValuesToCheck) -or ($ValuesToCheck.Count -lt 1)) + { + $keyList = $DesiredValues.Keys + } + else + { + $keyList = $ValuesToCheck + } + + $keyList | ForEach-Object -Process { + if (($_ -ne 'Verbose')) + { + if (($CurrentValues.ContainsKey($_) -eq $false) ` + -or ($CurrentValues.$_ -ne $DesiredValues.$_) ` + -or (($DesiredValues.GetType().Name -ne 'CimInstance' -and $DesiredValues.ContainsKey($_) -eq $true) -and ($null -ne $DesiredValues.$_ -and $DesiredValues.$_.GetType().IsArray))) + { + if ($DesiredValues.GetType().Name -eq 'HashTable' -or ` + $DesiredValues.GetType().Name -eq 'PSBoundParametersDictionary') + { + $checkDesiredValue = $DesiredValues.ContainsKey($_) + } + else + { + # If DesiredValue is a CimInstance. + $checkDesiredValue = $false + if (([System.Boolean]($DesiredValues.PSObject.Properties.Name -contains $_)) -eq $true) + { + if ($null -ne $DesiredValues.$_) + { + $checkDesiredValue = $true + } + } + } + + if ($checkDesiredValue) + { + $desiredType = $DesiredValues.$_.GetType() + $fieldName = $_ + if ($desiredType.IsArray -eq $true) + { + if (($CurrentValues.ContainsKey($fieldName) -eq $false) ` + -or ($null -eq $CurrentValues.$fieldName)) + { + Write-Verbose -Message ($script:localizedData.PropertyValidationError -f $fieldName) -Verbose + + $returnValue = $false + } + else + { + $arrayCompare = Compare-Object -ReferenceObject $CurrentValues.$fieldName ` + -DifferenceObject $DesiredValues.$fieldName + if ($null -ne $arrayCompare) + { + Write-Verbose -Message ($script:localizedData.PropertiesDoesNotMatch -f $fieldName) -Verbose + + $arrayCompare | ForEach-Object -Process { + Write-Verbose -Message ($script:localizedData.PropertyThatDoesNotMatch -f $_.InputObject, $_.SideIndicator) -Verbose + } + + $returnValue = $false + } + } + } + else + { + switch ($desiredType.Name) + { + 'String' + { + if (-not [System.String]::IsNullOrEmpty($CurrentValues.$fieldName) -or ` + -not [System.String]::IsNullOrEmpty($DesiredValues.$fieldName)) + { + Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` + -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose + + $returnValue = $false + } + } + + 'Int32' + { + if (-not ($DesiredValues.$fieldName -eq 0) -or ` + -not ($null -eq $CurrentValues.$fieldName)) + { + Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` + -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose + + $returnValue = $false + } + } + + { $_ -eq 'Int16' -or $_ -eq 'UInt16' -or $_ -eq 'Single' } + { + if (-not ($DesiredValues.$fieldName -eq 0) -or ` + -not ($null -eq $CurrentValues.$fieldName)) + { + Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` + -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose + + $returnValue = $false + } + } + + 'Boolean' + { + if ($CurrentValues.$fieldName -ne $DesiredValues.$fieldName) + { + Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` + -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose + + $returnValue = $false + } + } + + default + { + Write-Warning -Message ($script:localizedData.UnableToCompareProperty ` + -f $fieldName, $desiredType.Name) + + $returnValue = $false + } + } + } + } + } + } + } + + return $returnValue +} + +<# + .SYNOPSIS + Returns the value of the provided in the Name parameter, at the registry + location provided in the Path parameter. + + .PARAMETER Path + String containing the path in the registry to the property name. + + .PARAMETER PropertyName + String containing the name of the property for which the value is returned. +#> +function Get-RegistryPropertyValue +{ + [CmdletBinding()] + [OutputType([System.String])] + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $Path, + + [Parameter(Mandatory = $true)] + [System.String] + $Name + ) + + $getItemPropertyParameters = @{ + Path = $Path + Name = $Name + } + + <# + Using a try/catch block instead of 'SilentlyContinue' to be + able to unit test a failing registry path. + #> + try + { + $getItemPropertyResult = (Get-ItemProperty @getItemPropertyParameters -ErrorAction Stop).$Name + } + catch + { + $getItemPropertyResult = $null + } + + return $getItemPropertyResult +} + +<# + .SYNOPSIS + Returns the value of the provided in the Name parameter, at the registry + location provided in the Path parameter. + + .PARAMETER Path + String containing the path in the registry to the property name. + + .PARAMETER PropertyName + String containing the name of the property for which the value is returned. +#> +function Format-Path +{ + [CmdletBinding()] + [OutputType([System.String])] + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $Path, + + [Parameter()] + [System.Management.Automation.SwitchParameter] + $TrailingSlash + ) + + # Remove trailing slash ('\') from path. + if ($TrailingSlash.IsPresent) + { + <# + Trim backslash, but only if the path contains a full path and + not just a qualifier. + #> + if ($Path -notmatch '^[a-zA-Z]:\\$') + { + $Path = $Path.TrimEnd('\') + } + + <# + If the path only contains a qualifier but no backslash ('M:'), + then a backslash is added ('M:\'). + #> + if ($Path -match '^[a-zA-Z]:$') + { + $Path = '{0}\' -f $Path + } + } + + return $Path +} + +<# + .SYNOPSIS + Copy folder structure using Robocopy. Every file and folder, including empty ones are copied. + + .PARAMETER Path + Source path to be copied. + + .PARAMETER DestinationPath + The path to the destination. +#> +function Copy-ItemWithRobocopy +{ + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $Path, + + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $DestinationPath + ) + + $quotedPath = '"{0}"' -f $Path + $quotedDestinationPath = '"{0}"' -f $DestinationPath + $robocopyExecutable = Get-Command -Name "Robocopy.exe" -ErrorAction Stop + + $robocopyArgumentSilent = '/njh /njs /ndl /nc /ns /nfl' + $robocopyArgumentCopySubDirectoriesIncludingEmpty = '/e' + $robocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource = '/purge' + + if ([System.Version]$robocopyExecutable.FileVersionInfo.ProductVersion -ge [System.Version]'6.3.9600.16384') + { + Write-Verbose -Message $script:localizedData.RobocopyUsingUnbufferedIo -Verbose + + $robocopyArgumentUseUnbufferedIO = '/J' + } + else + { + Write-Verbose -Message $script:localizedData.RobocopyNotUsingUnbufferedIo -Verbose + } + + $robocopyArgumentList = '{0} {1} {2} {3} {4} {5}' -f $quotedPath, + $quotedDestinationPath, + $robocopyArgumentCopySubDirectoriesIncludingEmpty, + $robocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource, + $robocopyArgumentUseUnbufferedIO, + $robocopyArgumentSilent + + $robocopyStartProcessParameters = @{ + FilePath = $robocopyExecutable.Name + ArgumentList = $robocopyArgumentList + } + + Write-Verbose -Message ($script:localizedData.RobocopyArguments -f $robocopyArgumentList) -Verbose + $robocopyProcess = Start-Process @robocopyStartProcessParameters -Wait -NoNewWindow -PassThru + + switch ($($robocopyProcess.ExitCode)) + { + {$_ -in 8, 16} + { + $errorMessage = $script:localizedData.RobocopyErrorCopying -f $_ + New-InvalidOperationException -Message $errorMessage + } + + {$_ -gt 7 } + { + $errorMessage = $script:localizedData.RobocopyFailuresCopying -f $_ + New-InvalidResultException -Message $errorMessage + } + + 1 + { + Write-Verbose -Message $script:localizedData.RobocopySuccessful -Verbose + } + + 2 + { + Write-Verbose -Message $script:localizedData.RobocopyRemovedExtraFilesAtDestination -Verbose + } + + 3 + { + Write-Verbose -Message $script:localizedData.RobocopySuccessfulAndRemovedExtraFilesAtDestination -Verbose + } + + {$_ -eq 0 -or $null -eq $_ } + { + Write-Verbose -Message $script:localizedData.RobocopyAllFilesPresent -Verbose + } + } +} + +<# + .SYNOPSIS + Returns the path of the current user's temporary folder. +#> +function Get-TemporaryFolder +{ + [CmdletBinding()] + [OutputType([System.String])] + param() + + return [IO.Path]::GetTempPath() +} + +<# + .SYNOPSIS + Connects to the source using the provided credentials and then uses + robocopy to download the installation media to a local temporary folder. + + .PARAMETER SourcePath + Source path to be copied. + + .PARAMETER SourceCredential + The credentials to access the SourcePath. + + .PARAMETER PassThru + If used, returns the destination path as string. + + .OUTPUTS + Returns the destination path (when used with the parameter PassThru). +#> +function Invoke-InstallationMediaCopy +{ + [CmdletBinding()] + [OutputType([System.String])] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $SourcePath, + + [Parameter(Mandatory = $true)] + [System.Management.Automation.PSCredential] + $SourceCredential, + + [Parameter()] + [System.Management.Automation.SwitchParameter] + $PassThru + ) + + Connect-UncPath -RemotePath $SourcePath -SourceCredential $SourceCredential + + <# + Create a destination folder so the media files aren't written + to the root of the Temp folder. + #> + $mediaDestinationFolder = Split-Path -Path $SourcePath -Leaf + if (-not $mediaDestinationFolder ) + { + $mediaDestinationFolder = New-Guid | Select-Object -ExpandProperty Guid + } + + $mediaDestinationPath = Join-Path -Path (Get-TemporaryFolder) -ChildPath $mediaDestinationFolder + + Write-Verbose -Message ($script:localizedData.RobocopyIsCopying -f $SourcePath, $mediaDestinationPath) + Copy-ItemWithRobocopy -Path $SourcePath -DestinationPath $mediaDestinationPath + + Disconnect-UncPath -RemotePath $SourcePath + + if ($PassThru.IsPresent) + { + return $mediaDestinationPath + } +} + +<# + .SYNOPSIS + Connects to the UNC path provided in the parameter SourcePath. + Optionally connects using the provided credentials. + + .PARAMETER SourcePath + Source path to connect to. + + .PARAMETER SourceCredential + The credentials to access the path provided in SourcePath. + + .PARAMETER PassThru + If used, returns a MSFT_SmbMapping object that represents the newly + created SMB mapping. + + .OUTPUTS + Returns a MSFT_SmbMapping object that represents the newly created + SMB mapping (ony when used with parameter PassThru). +#> +function Connect-UncPath +{ + [CmdletBinding()] + [OutputType([Microsoft.Management.Infrastructure.CimInstance])] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $RemotePath, + + [Parameter()] + [System.Management.Automation.PSCredential] + $SourceCredential, + + [Parameter()] + [System.Management.Automation.SwitchParameter] + $PassThru + ) + + $newSmbMappingParameters = @{ + RemotePath = $RemotePath + } + + if ($PSBoundParameters.ContainsKey('SourceCredential')) + { + $newSmbMappingParameters['UserName'] = "$($SourceCredential.GetNetworkCredential().Domain)\$($SourceCredential.GetNetworkCredential().UserName)" + $newSmbMappingParameters['Password'] = $SourceCredential.GetNetworkCredential().Password + } + + $newSmbMappingResult = New-SmbMapping @newSmbMappingParameters + + if ($PassThru.IsPresent) + { + return $newSmbMappingResult + } +} + +<# + .SYNOPSIS + Disconnects from the UNC path provided in the parameter SourcePath. + + .PARAMETER SourcePath + Source path to disconnect from. +#> +function Disconnect-UncPath +{ + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $RemotePath + ) + + Remove-SmbMapping -RemotePath $RemotePath -Force +} + +<# + .SYNOPSIS + Queries the registry and returns $true if there is a pending reboot. + + .OUTPUTS + Returns $true if there is a pending reboot, otherwise it returns $false. +#> +function Test-PendingRestart +{ + [CmdletBinding()] + [OutputType([System.Boolean])] + param + ( + ) + + $getRegistryPropertyValueParameters = @{ + Path = 'HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager' + Name = 'PendingFileRenameOperations' + } + + <# + If the key 'PendingFileRenameOperations' does not exist then if should + return $false, otherwise it should return $true. + #> + return $null -ne (Get-RegistryPropertyValue @getRegistryPropertyValueParameters) +} + +<# + .SYNOPSIS + Starts the SQL setup process. + + .PARAMETER FilePath + String containing the path to setup.exe. + + .PARAMETER ArgumentList + The arguments that should be passed to setup.exe. + + .PARAMETER Timeout + The timeout in seconds to wait for the process to finish. +#> +function Start-SqlSetupProcess +{ + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $FilePath, + + [Parameter()] + [System.String] + $ArgumentList, + + [Parameter(Mandatory = $true)] + [System.UInt32] + $Timeout + ) + + $startProcessParameters = @{ + FilePath = $FilePath + ArgumentList = $ArgumentList + } + + $sqlSetupProcess = Start-Process @startProcessParameters -PassThru -NoNewWindow -ErrorAction Stop + + Write-Verbose -Message ($script:localizedData.StartSetupProcess -f $sqlSetupProcess.Id, $startProcessParameters.FilePath, $Timeout) -Verbose + + Wait-Process -InputObject $sqlSetupProcess -Timeout $Timeout -ErrorAction Stop + + return $sqlSetupProcess.ExitCode +} <# .SYNOPSIS @@ -408,174 +1006,6 @@ function New-VerboseMessage Write-Verbose -Message ((Get-Date -format yyyy-MM-dd_HH-mm-ss) + ": $Message") -Verbose } -<# - .SYNOPSIS - This method is used to compare current and desired values for any DSC resource. - - .PARAMETER CurrentValues - This is hash table of the current values that are applied to the resource. - - .PARAMETER DesiredValues - This is a PSBoundParametersDictionary of the desired values for the resource. - - .PARAMETER ValuesToCheck - This is a list of which properties in the desired values list should be checked. - If this is empty then all values in DesiredValues are checked. -#> -function Test-SQLDscParameterState -{ - [CmdletBinding()] - param - ( - [Parameter(Mandatory = $true)] - [System.Collections.Hashtable] - $CurrentValues, - - [Parameter(Mandatory = $true)] - [System.Object] - $DesiredValues, - - [Parameter()] - [System.Array] - $ValuesToCheck - ) - - $returnValue = $true - - if (($DesiredValues.GetType().Name -ne 'HashTable') ` - -and ($DesiredValues.GetType().Name -ne 'CimInstance') ` - -and ($DesiredValues.GetType().Name -ne 'PSBoundParametersDictionary')) - { - $errorMessage = $script:localizedData.PropertyTypeInvalidForDesiredValues -f $($DesiredValues.GetType().Name) - New-InvalidArgumentException -ArgumentName 'DesiredValues' -Message $errorMessage - } - - if (($DesiredValues.GetType().Name -eq 'CimInstance') -and ($null -eq $ValuesToCheck)) - { - $errorMessage = $script:localizedData.PropertyTypeInvalidForValuesToCheck - New-InvalidArgumentException -ArgumentName 'ValuesToCheck' -Message $errorMessage - } - - if (($null -eq $ValuesToCheck) -or ($ValuesToCheck.Count -lt 1)) - { - $keyList = $DesiredValues.Keys - } - else - { - $keyList = $ValuesToCheck - } - - $keyList | ForEach-Object -Process { - if (($_ -ne 'Verbose')) - { - if (($CurrentValues.ContainsKey($_) -eq $false) ` - -or ($CurrentValues.$_ -ne $DesiredValues.$_) ` - -or (($DesiredValues.GetType().Name -ne 'CimInstance' -and $DesiredValues.ContainsKey($_) -eq $true) -and ($null -ne $DesiredValues.$_ -and $DesiredValues.$_.GetType().IsArray))) - { - if ($DesiredValues.GetType().Name -eq 'HashTable' -or ` - $DesiredValues.GetType().Name -eq 'PSBoundParametersDictionary') - { - $checkDesiredValue = $DesiredValues.ContainsKey($_) - } - else - { - # If DesiredValue is a CimInstance. - $checkDesiredValue = $false - if (([System.Boolean]($DesiredValues.PSObject.Properties.Name -contains $_)) -eq $true) - { - if ($null -ne $DesiredValues.$_) - { - $checkDesiredValue = $true - } - } - } - - if ($checkDesiredValue) - { - $desiredType = $DesiredValues.$_.GetType() - $fieldName = $_ - if ($desiredType.IsArray -eq $true) - { - if (($CurrentValues.ContainsKey($fieldName) -eq $false) ` - -or ($null -eq $CurrentValues.$fieldName)) - { - Write-Verbose -Message ($script:localizedData.PropertyValidationError -f $fieldName) -Verbose - - $returnValue = $false - } - else - { - $arrayCompare = Compare-Object -ReferenceObject $CurrentValues.$fieldName ` - -DifferenceObject $DesiredValues.$fieldName - if ($null -ne $arrayCompare) - { - Write-Verbose -Message ($script:localizedData.PropertiesDoesNotMatch -f $fieldName) -Verbose - - $arrayCompare | ForEach-Object -Process { - Write-Verbose -Message ($script:localizedData.PropertyThatDoesNotMatch -f $_.InputObject, $_.SideIndicator) -Verbose - } - - $returnValue = $false - } - } - } - else - { - switch ($desiredType.Name) - { - 'String' - { - if (-not [System.String]::IsNullOrEmpty($CurrentValues.$fieldName) -or ` - -not [System.String]::IsNullOrEmpty($DesiredValues.$fieldName)) - { - Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` - -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose - - $returnValue = $false - } - } - - 'Int32' - { - if (-not ($DesiredValues.$fieldName -eq 0) -or ` - -not ($null -eq $CurrentValues.$fieldName)) - { - Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` - -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose - - $returnValue = $false - } - } - - { $_ -eq 'Int16' -or $_ -eq 'UInt16'} - { - if (-not ($DesiredValues.$fieldName -eq 0) -or ` - -not ($null -eq $CurrentValues.$fieldName)) - { - Write-Verbose -Message ($script:localizedData.ValueOfTypeDoesNotMatch ` - -f $desiredType.Name, $fieldName, $($CurrentValues.$fieldName), $($DesiredValues.$fieldName)) -Verbose - - $returnValue = $false - } - } - - default - { - Write-Warning -Message ($script:localizedData.UnableToCompareProperty ` - -f $fieldName, $desiredType.Name) - - $returnValue = $false - } - } - } - } - } - } - } - - return $returnValue -} - <# .SYNOPSIS Imports the module SQLPS in a standardized way. @@ -1118,8 +1548,8 @@ function Test-LoginEffectivePermissions [System.String[]] $Permissions, - [ValidateSet('SERVER', 'LOGIN')] [Parameter()] + [ValidateSet('SERVER', 'LOGIN')] [System.String] $SecurableClass = 'SERVER', @@ -1346,7 +1776,8 @@ function Test-ImpersonatePermissions New-VerboseMessage -Message ( 'The login "{0}" does not have control server permissions on the instance "{1}\{2}".' -f $testLoginEffectivePermissionsParams.LoginName, $testLoginEffectivePermissionsParams.SQLServer, $testLoginEffectivePermissionsParams.SQLInstanceName ) } - if ( -not [System.String]::IsNullOrEmpty($SecurableName) ) { + if (-not [System.String]::IsNullOrEmpty($SecurableName)) + { # Check for login-specific impersonation permissions $testLoginEffectivePermissionsParams = @{ SQLServer = $ServerObject.ComputerNamePhysicalNetBIOS @@ -1356,6 +1787,7 @@ function Test-ImpersonatePermissions SecurableClass = 'LOGIN' SecurableName = $SecurableName } + $impersonatePermissionsPresent = Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams if ($impersonatePermissionsPresent) { @@ -1376,6 +1808,7 @@ function Test-ImpersonatePermissions SecurableClass = 'LOGIN' SecurableName = $SecurableName } + $impersonatePermissionsPresent = Test-LoginEffectivePermissions @testLoginEffectivePermissionsParams if ($impersonatePermissionsPresent) { @@ -1402,7 +1835,7 @@ function Test-ImpersonatePermissions .OUTPUTS Hash table with the properties SQLServer and SQLInstanceName. #> -function Split-FullSQLInstanceName +function Split-FullSqlInstanceName { param ( @@ -1736,3 +2169,36 @@ function Find-ExceptionByNumber return $errorFound } +Export-ModuleMember -Function @( + 'Test-DscParameterState' + 'Get-RegistryPropertyValue' + 'Format-Path' + 'Copy-ItemWithRobocopy' + 'Get-TemporaryFolder' + 'Invoke-InstallationMediaCopy' + 'Connect-UncPath' + 'Disconnect-UncPath' + 'Test-PendingRestart' + 'Start-SqlSetupProcess' + 'Connect-SQL' + 'Connect-SQLAnalysis' + 'Get-SqlInstanceMajorVersion' + 'New-TerminatingError' + 'New-WarningMessage' + 'New-VerboseMessage' + 'Import-SQLPSModule' + 'Restart-SqlService' + 'Restart-ReportingServicesService' + 'Invoke-Query' + 'Update-AvailabilityGroupReplica' + 'Test-LoginEffectivePermissions' + 'Test-AvailabilityReplicaSeedingModeAutomatic' + 'Get-PrimaryReplicaServerObject' + 'Test-ImpersonatePermissions' + 'Split-FullSqlInstanceName' + 'Test-ClusterPermissions' + 'Test-ActiveNode' + 'Invoke-SqlScript' + 'Get-ServiceAccount' + 'Find-ExceptionByNumber' +) diff --git a/en-US/SqlServerDscHelper.strings.psd1 b/Modules/DscResource.Common/en-US/DscResource.Common.strings.psd1 similarity index 89% rename from en-US/SqlServerDscHelper.strings.psd1 rename to Modules/DscResource.Common/en-US/DscResource.Common.strings.psd1 index 71acf9033..c0b589dc1 100644 --- a/en-US/SqlServerDscHelper.strings.psd1 +++ b/Modules/DscResource.Common/en-US/DscResource.Common.strings.psd1 @@ -1,19 +1,29 @@ -# Localized resources for helper module SqlServerDscHelper. +# Localized resources for helper module DscResource.Common. ConvertFrom-StringData @' - ConnectedToDatabaseEngineInstance = Connected to SQL instance '{0}'. - FailedToConnectToDatabaseEngineInstance = Failed to connect to SQL instance '{0}'. - ConnectedToAnalysisServicesInstance = Connected to Analysis Services instance '{0}'. - FailedToConnectToAnalysisServicesInstance = Failed to connected to Analysis Services instance '{0}'. - SqlMajorVersion = SQL major version is {0}. - SqlServerVersionIsInvalid = Could not get the SQL version for the instance '{0}'. PropertyTypeInvalidForDesiredValues = Property 'DesiredValues' must be either a [System.Collections.Hashtable], [CimInstance] or [PSBoundParametersDictionary]. The type detected was {0}. PropertyTypeInvalidForValuesToCheck = If 'DesiredValues' is a CimInstance, then property 'ValuesToCheck' must contain a value. PropertyValidationError = Expected to find an array value for property {0} in the current values, but it was either not present or was null. This has caused the test method to return false. PropertiesDoesNotMatch = Found an array for property {0} in the current values, but this array does not match the desired state. Details of the changes are below. PropertyThatDoesNotMatch = {0} - {1} ValueOfTypeDoesNotMatch = {0} value for property {1} does not match. Current state is '{2}' and desired state is '{3}'. - UnableToCompareProperty = Unable to compare property {0} as the type {1} is not handled by the Test-SQLDSCParameterState cmdlet. + UnableToCompareProperty = Unable to compare property {0} as the type {1} is not handled by the Test-DscParameterState cmdlet. + RobocopyUsingUnbufferedIo = Robocopy is using unbuffered I/O. + RobocopyNotUsingUnbufferedIo = Unbuffered I/O cannot be used due to incompatible version of Robocopy. + RobocopyArguments = Robocopy is started with the following arguments: {0} + RobocopyErrorCopying = Robocopy reported errors when copying files. Error code: {0}. + RobocopyFailuresCopying = Robocopy reported that failures occurred when copying files. Error code: {0}. + RobocopySuccessful = Robocopy copied files successfully + RobocopyRemovedExtraFilesAtDestination = Robocopy found files at the destination path that is not present at the source path, these extra files was remove at the destination path. + RobocopySuccessfulAndRemovedExtraFilesAtDestination = Robocopy copied files to destination successfully. Robocopy also found files at the destination path that is not present at the source path, these extra files was remove at the destination path. + RobocopyAllFilesPresent = Robocopy reported that all files already present. + StartSetupProcess = Started the process with id {0} using the path '{1}', and with a timeout value of {2} seconds. + ConnectedToDatabaseEngineInstance = Connected to SQL instance '{0}'. + FailedToConnectToDatabaseEngineInstance = Failed to connect to SQL instance '{0}'. + ConnectedToAnalysisServicesInstance = Connected to Analysis Services instance '{0}'. + FailedToConnectToAnalysisServicesInstance = Failed to connected to Analysis Services instance '{0}'. + SqlMajorVersion = SQL major version is {0}. + SqlServerVersionIsInvalid = Could not get the SQL version for the instance '{0}'. PreferredModuleFound = Preferred module SqlServer found. PreferredModuleNotFound = Information: PowerShell module SqlServer not found, trying to use older SQLPS module. ImportedPowerShellModule = Importing PowerShell module '{0}' with version '{1}' from path '{2}'. @@ -144,5 +154,4 @@ ConvertFrom-StringData @' # SQLServerNetwork UnableToUseBothDynamicAndStaticPort = Unable to set both TCP dynamic port and TCP static port. Only one can be set. - '@ diff --git a/sv-SE/SqlServerDscHelper.strings.psd1 b/Modules/DscResource.Common/sv-SE/DscResource.Common.strings.psd1 similarity index 99% rename from sv-SE/SqlServerDscHelper.strings.psd1 rename to Modules/DscResource.Common/sv-SE/DscResource.Common.strings.psd1 index 905d49201..0d2ec4462 100644 --- a/sv-SE/SqlServerDscHelper.strings.psd1 +++ b/Modules/DscResource.Common/sv-SE/DscResource.Common.strings.psd1 @@ -13,7 +13,7 @@ ConvertFrom-StringData @' PropertiesDoesNotMatch = Hittade en matris för egenskapen {0} för nuvarande värden, men denna matris matchar inte önskat läge. Detaljer för ändringarna finns nedan. PropertyThatDoesNotMatch = {0} - {1} ValueOfTypeDoesNotMatch = {0} värde för egenskapen {1} matchar inte. Nuvarande läge är '{2}' och önskat läge är '{3}'. - UnableToCompareProperty = Inte möjligt att jämföra egenskapen {0} som typen {1}. {1} hanteras inte av Test-SQLDscParameterState cmdlet. + UnableToCompareProperty = Inte möjligt att jämföra egenskapen {0} som typen {1}. {1} hanteras inte av Test-DscParameterState cmdlet. PreferredModuleFound = Föredragen modul SqlServer funnen. PreferredModuleNotFound = Information: PowerShell modul SqlServer ej funnen, försöker att använda äldre SQLPS modul. ImportedPowerShellModule = Importerade PowerShell modul '{0}' med version '{1}' från mapp '{2}'. diff --git a/DSCResources/CommonResourceHelper.psm1 b/Modules/DscResource.LocalizationHelper/DscResource.LocalizationHelper.psm1 similarity index 89% rename from DSCResources/CommonResourceHelper.psm1 rename to Modules/DscResource.LocalizationHelper/DscResource.LocalizationHelper.psm1 index bf18d83b8..32a3b847e 100644 --- a/DSCResources/CommonResourceHelper.psm1 +++ b/Modules/DscResource.LocalizationHelper/DscResource.LocalizationHelper.psm1 @@ -1,3 +1,67 @@ +<# + .SYNOPSIS + Retrieves the localized string data based on the machine's culture. + Falls back to en-US strings if the machine's culture is not supported. + + .PARAMETER ResourceName + The name of the resource as it appears before '.strings.psd1' of the localized string file. + For example: + For WindowsOptionalFeature: MSFT_WindowsOptionalFeature + For Service: MSFT_ServiceResource + For Registry: MSFT_RegistryResource + For Helper: SqlServerDscHelper + + .PARAMETER ScriptRoot + Optional. The root path where to expect to find the culture folder. This is only needed + for localization in helper modules. This should not normally be used for resources. + + .NOTES + To be able to use localization in the helper function, this function must + be first in the file, before Get-LocalizedData is used by itself to load + localized data for this helper module (see directly after this function). +#> +function Get-LocalizedData +{ + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $ResourceName, + + [Parameter()] + [ValidateNotNullOrEmpty()] + [System.String] + $ScriptRoot + ) + + if (-not $ScriptRoot) + { + $dscResourcesFolder = Join-Path -Path (Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent) -ChildPath 'DSCResources' + $resourceDirectory = Join-Path -Path $dscResourcesFolder -ChildPath $ResourceName + } + else + { + $resourceDirectory = $ScriptRoot + } + + $localizedStringFileLocation = Join-Path -Path $resourceDirectory -ChildPath $PSUICulture + + if (-not (Test-Path -Path $localizedStringFileLocation)) + { + # Fallback to en-US + $localizedStringFileLocation = Join-Path -Path $resourceDirectory -ChildPath 'en-US' + } + + Import-LocalizedData ` + -BindingVariable 'localizedData' ` + -FileName "$ResourceName.strings.psd1" ` + -BaseDirectory $localizedStringFileLocation + + return $localizedData +} + <# .SYNOPSIS Creates and throws an invalid argument exception. @@ -193,73 +257,10 @@ function New-InvalidResultException throw $errorRecordToThrow } -<# - .SYNOPSIS - Retrieves the localized string data based on the machine's culture. - Falls back to en-US strings if the machine's culture is not supported. - - .PARAMETER ResourceName - The name of the resource as it appears before '.strings.psd1' of the localized string file. - For example: - For WindowsOptionalFeature: MSFT_WindowsOptionalFeature - For Service: MSFT_ServiceResource - For Registry: MSFT_RegistryResource - For Helper: SqlServerDscHelper - - .PARAMETER ScriptRoot - Optional. The root path where to expect to find the culture folder. This is only needed - for localization in helper modules. This should not normally be used for resources. -#> -function Get-LocalizedData -{ - [CmdletBinding()] - param - ( - [Parameter(Mandatory = $true)] - [ValidateNotNullOrEmpty()] - [System.String] - $ResourceName, - - [Parameter()] - [ValidateNotNullOrEmpty()] - [System.String] - $ScriptRoot - ) - - if ( -not $ScriptRoot ) - { - $resourceDirectory = Join-Path -Path $PSScriptRoot -ChildPath $ResourceName - $localizedStringFileLocation = Join-Path -Path $resourceDirectory -ChildPath $PSUICulture - } - else - { - $localizedStringFileLocation = Join-Path -Path $ScriptRoot -ChildPath $PSUICulture - } - - if (-not (Test-Path -Path $localizedStringFileLocation)) - { - # Fallback to en-US - if ( -not $ScriptRoot ) - { - $localizedStringFileLocation = Join-Path -Path $resourceDirectory -ChildPath 'en-US' - } - else - { - $localizedStringFileLocation = Join-Path -Path $ScriptRoot -ChildPath 'en-US' - } - } - - Import-LocalizedData ` - -BindingVariable 'localizedData' ` - -FileName "$ResourceName.strings.psd1" ` - -BaseDirectory $localizedStringFileLocation - - return $localizedData -} - Export-ModuleMember -Function @( 'New-InvalidArgumentException', 'New-InvalidOperationException', 'New-ObjectNotFoundException', - 'New-InvalidResultException', - 'Get-LocalizedData' ) + 'New-InvalidResultException' + 'Get-LocalizedData' +) diff --git a/README.md b/README.md index 54f363864..73c7a208e 100644 --- a/README.md +++ b/README.md @@ -130,6 +130,8 @@ A full list of changes in each version can be found in the [change log](CHANGELO database roles. * [**SqlRS**](#sqlrs) configures SQL Server Reporting. Services to use a database engine in another instance. +* [**SqlRSSetup**](#sqlrssetup) Installs the standalone + [Microsoft SQL Server Reporting Services](https://docs.microsoft.com/en-us/sql/reporting-services/create-deploy-and-manage-mobile-and-paginated-reports). * [**SqlScript**](#sqlscript) resource to extend DSC Get/Set/Test functionality to T-SQL. * [**SqlScriptQuery**](#sqlscriptquery) resource to extend DSC Get/Set/Test @@ -811,6 +813,106 @@ already exist. This is caused when trying to add another URL using the same protocol. For example when trying to add 'http://+:443' when 'http://+:80' already exist. +### SqlRSSetup + +Installs the standalone [Microsoft SQL Server Reporting Services](https://docs.microsoft.com/en-us/sql/reporting-services/create-deploy-and-manage-mobile-and-paginated-reports). + +If both `SourceCredential` and `PsDscRunAsCredential` is used then the +credentials in `SourceCredential` will only be used to copy the +installation media locally, and then the credentials in `PsDscRunAsCredential` +will be used during installation. If `PsDscRunAsCredential` is not +used, then the installation will run as SYSTEM. + +>To install Microsoft SQL Server Reporting Services 2016 (or older), +>please use the resource SqlSetup. + +#### Requirements + +* Target machine must be running Windows Server 2012 or later. +* If `PsDscRunAsCredential` common parameter is used to run the resource, + the specified credential must have permissions to connect to the location + where the Microsoft SQL Server Reporting Services media is placed. +* The parameter IAcceptLicenseTerms must be set to 'Yes'. +* The parameter InstanceName can only be set to 'SSRS' since there is + no way to change the instance name. +* When using action 'Uninstall', the same version of the executable as the version + of the installed product must be used. If not, sometimes the uninstall + is successful (because the executable returns exit code 0) but the + Microsoft SQL Server Reporting Services instance was not actually removed. + +>NOTE: When using the action 'Uninstall' and the target node to begin with +>requires a restart, on the first run the Microsoft SQL Server Reporting +>Services instance will not be uninstalled, but instead exit with code +>3010 and the node will be, by default, restarted. On the second run after +>restart, the Microsoft SQL Server Reporting Services instance will be +>uninstalled. If the parameter SuppressRestart is used, then the node must +>be restarted manually before the Microsoft SQL Server Reporting Services +>instance will be successfully uninstalled. +> +>The Microsoft SQL Server Reporting Services log will indicate that a +>restart is required by outputting; "*No action was taken as a system +>reboot is required (0x8007015E)*". The log is default located in the +>SSRS folder in `%TEMP%`, e.g. `C:\Users\\AppData\Local\Temp\SSRS`. + +#### Parameters + +* **`[String]` InstanceName** _(Key)_: Name of the Microsoft SQL Server + Reporting Service instance to installed. This can only be set to 'SSRS'. + { 'SSRS' } +* **`[String]` IAcceptLicenseTerms** _(Required)_: Accept licens terms. + This must be set to 'Yes'. { 'Yes' } +* **`[String]` SourcePath** _(Required)_: The path to the installation media + file to be used for installation, e.g an UNC path to a shared resource. + Environment variables can be used in the path. +* **`[String]` Action** _(Write)_: The action to be performed. Default + value is 'Install' which performs either install or upgrade. + { *Install* | Uninstall } +* **`[PSCredential]` SourceCredential** _(Write)_: Credentials used to + access the path set in the parameter 'SourcePath'. +* **`[Boolean]` SuppressRestart** _(Write)_: Suppresses any attempts to + restart. +* **`[String]` ProductKey** _(Write)_: Sets the custom license key, e.g. + '12345-12345-12345-12345-12345'. +* **`[Boolean]` ForceRestart** _(Write)_: Forces a restart after installation + is finished. +* **`[Boolean]` EditionUpgrade** _(Write)_: Upgrades the edition of the + installed product. Requires that either the ProductKey or the Edition + parameter is also assigned. By default no edition upgrade is performed. +* **`[Boolean]` VersionUpgrade** _(Write)_: Upgrades installed product + version, if the major product version of the source executable is higher + than the major current version. Requires that either the ProductKey or + the Edition parameter is also assigned. Default is $false. +* **`[String]` Edition** _(Write)_: Sets the custom free edition. + { 'Development' | 'Evaluation' | 'ExpressAdvanced' } +* **`[String]` LogPath** _(Write)_: Specifies the setup log file location, + e.g. 'log.txt'. By default, log files are created under `%TEMP%`. +* **`[String]` InstallFolder** _(Write)_: Sets the install folder, e.g. + 'C:\Program Files\SSRS'. Default value is 'C:\Program Files\Microsoft + SQL Server Reporting Services'. +* **`[UInt32]` SetupProcessTimeout** _(Write)_: The timeout, in seconds, to wait + for the setup process to finish. Default value is 7200 seconds (2 hours). If + the setup process does not finish before this time, and error will be thrown. + +#### Read-Only Properties from Get-TargetResource + +* **`[String]` ErrorDumpDirectory** _(Read)_: Returns the path to error + dump log files. +* **`[String]` CurrentVersion** _(Read)_: Returns the current version + of the installed Microsoft SQL Server Reporting Service instance. +* **`[String]` ServiceName** _(Read)_: Returns the current name + of the Microsoft SQL Server Reporting Service instance Windows service. + +#### Examples + +* [Install Reporting Services](Examples/Resources/SqlRSSetup/1-InstallReportingServices.ps1) +* [Uninstall Reporting Services](Examples/Resources/SqlRSSetup/2-UninstallReportingServices.ps1) + +#### Known issues + +* [SqlRSSetup: Will always make an edition upgrade](https://github.com/PowerShell/SqlServerDsc/issues/1311) + +All issues are not listed here, see [here for all open issues](https://github.com/PowerShell/SqlServerDsc/issues?q=is%3Aissue+is%3Aopen+in%3Atitle+SqlRSSetup). + ### SqlScript Provides the means to run a user generated T-SQL script on the SQL Server instance. diff --git a/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 index a742fdee2..d417f72d5 100644 --- a/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlAgentOperator.Integration.Tests.ps1 @@ -84,7 +84,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -132,7 +132,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 index c100fffbe..53a727398 100644 --- a/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlAlwaysOnService.Integration.Tests.ps1 @@ -121,7 +121,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -167,7 +167,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } diff --git a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 index a49b01f75..26a8d617f 100644 --- a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.Integration.Tests.ps1 @@ -79,11 +79,11 @@ try } $resourceCurrentState.Type | Should -Be 'Data' - ( Join-Path -Path $resourceCurrentState.Path -ChildPath '' ) | Should -Be ( Join-Path -Path $ConfigurationData.AllNodes.DataFilePath -ChildPath '' ) + $resourceCurrentState.Path | Should -Be $ConfigurationData.AllNodes.DataFilePath } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -126,11 +126,11 @@ try } $resourceCurrentState.Type | Should -Be 'Log' - ( Join-Path -Path $resourceCurrentState.Path -ChildPath '' ) | Should -Be ( Join-Path -Path $ConfigurationData.AllNodes.LogFilePath -ChildPath '' ) + $resourceCurrentState.Path | Should -Be $ConfigurationData.AllNodes.LogFilePath } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -173,11 +173,12 @@ try } $resourceCurrentState.Type | Should -Be 'Backup' - ( Join-Path -Path $resourceCurrentState.Path -ChildPath '' ) | Should -Be ( Join-Path -Path $ConfigurationData.AllNodes.BackupFilePath -ChildPath '' ) + # Ending backslash is removed because of regression test for issue #1307. + $resourceCurrentState.Path | Should -Be $ConfigurationData.AllNodes.BackupFilePath.TrimEnd('\') } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 index 92c84c78e..acb624eec 100644 --- a/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 +++ b/Tests/Integration/MSFT_SqlDatabaseDefaultLocation.config.ps1 @@ -27,6 +27,8 @@ else DataFilePath = 'C:\SQLData\' LogFilePath = 'C:\SQLLog\' + + # Ending backslash is regression test for issue #1307. BackupFilePath = 'C:\Backups\' CertificateFile = $env:DscPublicCertificatePath diff --git a/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 index c072f9b99..492fdb216 100644 --- a/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlRS.Integration.Tests.ps1 @@ -1,5 +1,5 @@ # This is used to make sure the integration test run in the correct order. -[Microsoft.DscResourceKit.IntegrationTest(OrderNumber = 2)] +[Microsoft.DscResourceKit.IntegrationTest(OrderNumber = 3)] param() Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') @@ -113,7 +113,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } It 'Should be able to access the ReportServer site without any error' { @@ -203,7 +203,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } <# @@ -261,7 +261,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } It 'Should be able to access the ReportServer site without any error' { diff --git a/Tests/Integration/MSFT_SqlRSSetup.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlRSSetup.Integration.Tests.ps1 new file mode 100644 index 000000000..9e02134a0 --- /dev/null +++ b/Tests/Integration/MSFT_SqlRSSetup.Integration.Tests.ps1 @@ -0,0 +1,161 @@ +# This is used to make sure the integration test run in the correct order. +[Microsoft.DscResourceKit.IntegrationTest(OrderNumber = 2)] +param() + +Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') + +# Run only for SQL 2017 integration testing. +if (Test-SkipContinuousIntegrationTask -Type 'Integration' -Category @('Integration_SQL2017')) +{ + return +} + +$script:dscModuleName = 'SqlServerDsc' +$script:dscResourceFriendlyName = 'SqlRSSetup' +$script:dscResourceName = "MSFT_$($script:dscResourceFriendlyName)" + +#region HEADER +# Integration Test Template Version: 1.3.2 +[String] $script:moduleRoot = Split-Path -Parent (Split-Path -Parent $PSScriptRoot) +if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests'))) -or ` + (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1'))) ) +{ + & git @('clone', 'https://github.com/PowerShell/DscResource.Tests.git', (Join-Path -Path $script:moduleRoot -ChildPath 'DscResource.Tests')) +} + +Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'DSCResource.Tests' -ChildPath 'TestHelper.psm1')) -Force +$TestEnvironment = Initialize-TestEnvironment ` + -DSCModuleName $script:dscModuleName ` + -DSCResourceName $script:dscResourceName ` + -TestType Integration +#endregion + +# Using try/finally to always cleanup. +try +{ + $configFile = Join-Path -Path $PSScriptRoot -ChildPath "$($script:dscResourceName).config.ps1" + . $configFile + + # Download Microsoft SQL Server Reporting Services (October 2017) executable + if (-not (Test-Path -Path $ConfigurationData.AllNodes.SourcePath)) + { + # By switching to 'SilentlyContinue' should theoretically increase the download speed. + $previousProgressPreference = $ProgressPreference + $ProgressPreference = 'SilentlyContinue' + + $script:mockSourceMediaDisplayName = 'Microsoft SQL Server Reporting Services (October 2017)' + $script:mockSourceMediaUrl = 'https://download.microsoft.com/download/E/6/4/E6477A2A-9B58-40F7-8AD6-62BB8491EA78/SQLServerReportingServices.exe' + + Write-Info -Message ('Start downloading the {1} executable at {0}.' -f (Get-Date -Format 'yyyy-MM-dd hh:mm:ss'), $script:mockSourceMediaDisplayName) -Verbose + + Invoke-WebRequest -Uri $script:mockSourceMediaUrl -OutFile $ConfigurationData.AllNodes.SourcePath + + Write-Info -Message ('{1} executable file has SHA1 hash ''{0}''.' -f (Get-FileHash -Path $ConfigurationData.AllNodes.SourcePath -Algorithm 'SHA1').Hash, $script:mockSourceMediaDisplayName) -Verbose + + $ProgressPreference = $previousProgressPreference + + # Double check that the Microsoft SQL Server Reporting Services (October 2017) was downloaded. + if (-not (Test-Path -Path $ConfigurationData.AllNodes.SourcePath)) + { + Write-Warning -Message ('{0} executable could not be downloaded, can not run the integration test.' -f $script:mockSourceMediaDisplayName) + return + } + else + { + Write-Info -Message ('Finished downloading the {1} executable at {0}.' -f (Get-Date -Format 'yyyy-MM-dd hh:mm:ss'), $script:mockSourceMediaDisplayName) -Verbose + } + } + else + { + Write-Info -Message ('{0} executable is already downloaded' -f $script:mockSourceMediaDisplayName) -Verbose + } + + Describe "$($script:dscResourceName)_Integration" { + BeforeAll { + $resourceId = "[$($script:dscResourceFriendlyName)]Integration_Test" + } + + $configurationName = "$($script:dscResourceName)_InstallReportingServicesAsUser_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + + It 'Should be able to call Get-DscConfiguration without throwing' { + { + $script:currentConfiguration = Get-DscConfiguration -Verbose -ErrorAction Stop + } | Should -Not -Throw + } + + It 'Should have set the resource and all the parameters should match' { + $resourceCurrentState = $script:currentConfiguration | Where-Object -FilterScript { + $_.ConfigurationName -eq $configurationName ` + -and $_.ResourceId -eq $resourceId + } + + $resourceCurrentState.InstanceName | Should -Be $ConfigurationData.AllNodes.InstanceName + $resourceCurrentState.InstallFolder | Should -Be 'C:\Program Files\Microsoft SQL Server Reporting Services' + $resourceCurrentState.ServiceName | Should -Be 'SQLServerReportingServices' + $resourceCurrentState.ErrorDumpDirectory | Should -Be 'C:\Program Files\Microsoft SQL Server Reporting Services\SSRS\LogFiles' + $resourceCurrentState.CurrentVersion | Should -BeGreaterThan ([System.Version] '14.0.0.0') + } + + It 'Should return $true when Test-DscConfiguration is run' { + Test-DscConfiguration -Verbose | Should -Be 'True' + } + } + + $configurationName = "$($script:dscResourceName)_StopReportingServicesInstance_Config" + + Context ('When using configuration {0}' -f $configurationName) { + It 'Should compile and apply the MOF without throwing' { + { + $configurationParameters = @{ + OutputPath = $TestDrive + # The variable $ConfigurationData was dot-sourced above. + ConfigurationData = $ConfigurationData + } + + & $configurationName @configurationParameters + + $startDscConfigurationParameters = @{ + Path = $TestDrive + ComputerName = 'localhost' + Wait = $true + Verbose = $true + Force = $true + ErrorAction = 'Stop' + } + + Start-DscConfiguration @startDscConfigurationParameters + } | Should -Not -Throw + } + } + } +} +finally +{ + #region FOOTER + Restore-TestEnvironment -TestEnvironment $TestEnvironment + #endregion +} diff --git a/Tests/Integration/MSFT_SqlRSSetup.config.ps1 b/Tests/Integration/MSFT_SqlRSSetup.config.ps1 new file mode 100644 index 000000000..c9db6c923 --- /dev/null +++ b/Tests/Integration/MSFT_SqlRSSetup.config.ps1 @@ -0,0 +1,90 @@ +#region HEADER +# Integration Test Config Template Version: 1.2.0 +#endregion + +$configFile = [System.IO.Path]::ChangeExtension($MyInvocation.MyCommand.Path, 'json') +if (Test-Path -Path $configFile) +{ + <# + Allows reading the configuration data from a JSON file, + for real testing scenarios outside of the CI. + #> + $ConfigurationData = Get-Content -Path $configFile | ConvertFrom-Json +} +else +{ + $ConfigurationData = @{ + AllNodes = @( + @{ + NodeName = 'localhost' + InstanceName = 'SSRS' + IAcceptLicenseTerms = 'Yes' + SourcePath = Join-Path -Path $env:TEMP -ChildPath 'SQLServerReportingServices.exe' + Edition = 'Development' + + UserName = "$env:COMPUTERNAME\SqlInstall" + Password = 'P@ssw0rd1' + + CertificateFile = $env:DscPublicCertificatePath + } + ) + } +} + +<# + .SYNOPSIS + Installs a Microsoft SQL Server Reporting Services instance. + + .NOTES + When this test was written the build worker already contained a + Microsoft SQL Server Reporting Services instance. + If it exist, it will be upgraded. + + Uninstall is not tested, because when upgrading the existing Microsoft + SQL Server 2017 Reporting Services instance it requires a restart which + prevents uninstall until the node is rebooted. +#> +Configuration MSFT_SqlRSSetup_InstallReportingServicesAsUser_Config +{ + Import-DscResource -ModuleName 'SqlServerDsc' + + node $AllNodes.NodeName + { + SqlRSSetup 'Integration_Test' + { + InstanceName = $Node.InstanceName + IAcceptLicenseTerms = $Node.IAcceptLicenseTerms + SourcePath = $Node.SourcePath + Edition = $Node.Edition + + # The build worker contains already an instance, make sure to upgrade it. + VersionUpgrade = $true + + # Suppressing restart because the build worker are not allowed to be restarted. + SuppressRestart = $true + + PsDscRunAsCredential = New-Object ` + -TypeName System.Management.Automation.PSCredential ` + -ArgumentList @($Node.Username, (ConvertTo-SecureString -String $Node.Password -AsPlainText -Force)) + } + } +} + +<# + .SYNOPSIS + Stopping the Microsoft SQL Server Reporting Services instance to + save memory on the build worker. +#> +Configuration MSFT_SqlRSSetup_StopReportingServicesInstance_Config +{ + Import-DscResource -ModuleName 'PSDscResources' + + node $AllNodes.NodeName + { + Service 'StopReportingServicesInstance' + { + Name = 'SQLServerReportingServices' + State = 'Stopped' + } + } +} diff --git a/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 index ddaec72df..253986735 100644 --- a/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlScript.Integration.Tests.ps1 @@ -159,7 +159,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -208,7 +208,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 index 8c4376419..9e188b375 100644 --- a/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlScriptQuery.Integration.Tests.ps1 @@ -127,7 +127,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -176,7 +176,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 index a10bfafc0..781aa30a5 100644 --- a/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerDatabaseMail.Integration.Tests.ps1 @@ -91,7 +91,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -146,7 +146,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 index c15753605..f6bfdc140 100644 --- a/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerEndPoint.Integration.Tests.ps1 @@ -86,7 +86,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -132,7 +132,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 index d423666b6..d77714b44 100644 --- a/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerLogin.Integration.Tests.ps1 @@ -117,7 +117,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -166,7 +166,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -215,7 +215,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -264,7 +264,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -313,7 +313,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -361,7 +361,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 index 54da4494c..5422de148 100644 --- a/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerNetwork.Integration.Tests.ps1 @@ -89,7 +89,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -138,7 +138,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 index a94d83f05..724d3ae85 100644 --- a/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerRole.Integration.Tests.ps1 @@ -87,7 +87,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -138,7 +138,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -192,7 +192,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -246,7 +246,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -305,7 +305,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -359,7 +359,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -410,7 +410,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 index 760d3369e..b2524c9be 100644 --- a/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServerSecureConnection.Integration.Tests.ps1 @@ -100,7 +100,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -147,7 +147,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 index c3157e203..b04b5b952 100644 --- a/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlServiceAccount.Integration.Tests.ps1 @@ -119,7 +119,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -166,7 +166,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -213,7 +213,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -260,7 +260,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -334,7 +334,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -381,7 +381,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -428,7 +428,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -475,7 +475,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } } diff --git a/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 b/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 index eaf77e99e..155fa718b 100644 --- a/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 +++ b/Tests/Integration/MSFT_SqlSetup.Integration.Tests.ps1 @@ -285,7 +285,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -424,7 +424,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } @@ -556,7 +556,7 @@ try } It 'Should return $true when Test-DscConfiguration is run' { - Test-DscConfiguration -Verbose | Should -Be $true + Test-DscConfiguration -Verbose | Should -Be 'True' } } diff --git a/Tests/Integration/README.md b/Tests/Integration/README.md index 77b3c6e63..eaa53f575 100644 --- a/Tests/Integration/README.md +++ b/Tests/Integration/README.md @@ -46,6 +46,7 @@ be used by other integration tests. > Note: User account names was kept to a maximum of 15 characters. + User | Password | Permission | Description --- | --- | --- | --- .\SqlInstall | P@ssw0rd1 | Local Windows administrator. Administrator of Database Engine instance DSCSQLTEST\*. | Runs Setup for the default instance. @@ -55,11 +56,54 @@ User | Password | Permission | Description .\svc-SqlSecondary | yig-C^Equ3 | Local user. | Used by other tests, but created here. .\svc-SqlAgentSec | yig-C^Equ3 | Local user. | Used by other tests. sa | P@ssw0rd1 | Administrator of the Database Engine instances DSCSQLTEST. | + *\* This is due to that the integration tests runs the resource SqlAlwaysOnService with this user and that means that this user must have permission to access the properties `IsClustered` and `IsHadrEnable`.* +## SqlRSSetup + +**Run order:** 2 + +**Depends on:** SqlSetup (for the local installation account) + +The integration tests will install (or upgrade) a Microsoft SQL Server +2017 Reporting Services instance and leave it on the AppVeyor build worker +for other integration tests to use. + +>**NOTE:** Uninstall is not tested, because when upgrading the existing +>Microsoft SQL Server Reporting Services instance it requires a restart, +>that prevents uninstall until the node is restarted. AppVeyor build +>workers are not allowed to be restarted during testing phase. + +Instance | State +--- | --- +SSRS | Stopped + +>**Note:** The Reporting Services instance is not configured after it is +>installed or upgraded, but if there are already an instance of Reporting +>Services installed on the build worker, it could have been configured. +>Other integration tests need to take that into consideration. + +### Properties for the instance SSRS + +- **InstanceName:** SSRS +- **CurrentVersion:** ^14.0.6981.38291 (depends on the version downloaded) +- **ErrorDumpDirectory:** C:\Program Files\Microsoft SQL Server Reporting Services\SSRS\LogFiles +- **LogPath:** C:\Users\appveyor\AppData\Local\Temp\SSRS +- **InstallFolder:** C:\Program Files\Microsoft SQL Server Reporting Services +- **ServiceName:** SQLServerReportingServices +- **Edition:** Developer + +### Users + + +User | Password | Description +--- | --- | --- | --- +.\SqlInstall | P@ssw0rd1 | The Reporting Services instance is installed using this account. + + ## SqlAlwaysOnService **Run order:** 2 @@ -71,41 +115,19 @@ an IP address of '192.168.40.10'. To be able to activate the AlwaysOn service th tests creates an Active Directory Detached Cluster with an IP address of '192.168.40.11' and the cluster will ignore any other static IP addresses. ->**Note:** During the tests the gateway of the loopback adatper named 'ClusterNetwork' + +>**Note:** During the tests the gateway of the loopback adapter named 'ClusterNetwork' >will be set to '192.168.40.254', because it is a requirement to create the cluster, >but the gateway will be removed in the last clean up test. Gateway is removed so >that there will be no conflict with the default gateway. ->*Note:** The Active Directory Detached Cluster is not fully functioning in the +>**Note:** The Active Directory Detached Cluster is not fully functioning in the >sense that it cannot start the Name resource in the 'Cluster Group', but it >starts enough to be able to run integration tests for AlwaysOn service.s + The tests will leave the AlwaysOn service disabled. -## SqlRS - -**Run order:** 2 - -**Depends on:** SqlSetup - -The integration tests will install the following instances and leave it on the -AppVeyor build worker for other integration tests to use. - -Instance | Feature | Description ---- | --- | --- -DSCRS2016 | RS | The Reporting Services is initialized, and in a working state. - ->**Note:** The Reporting Services service is stopped to save memory on the build ->worker. - -### Properties for the instance - -- **Collation:** Finnish\_Swedish\_CI\_AS -- **InstallSharedDir:** C:\Program Files\Microsoft SQL Server -- **InstallSharedWOWDir:** C:\Program Files (x86)\Microsoft SQL Server -- **DatabaseServerName:** `$env:COMPUTERNAME` -- **DatabaseInstanceName:** DSCSQLTEST - ## SqlDatabaseDefaultLocation **Run order:** 2 @@ -153,6 +175,30 @@ DscUser4 | SQL | P@ssw0rd1 | *None* > **Note:** Login DscUser3 was create disabled and was used to test removal of > a login. +## SqlRS + +**Run order:** 3 + +**Depends on:** SqlSetup, SqlRSSetup + +The integration tests will install the following instances and leave it on the +AppVeyor build worker for other integration tests to use. + +Instance | Feature | Description +--- | --- | --- +DSCRS2016 | RS | The Reporting Services is initialized, and in a working state. + +>**Note:** The Reporting Services service is stopped to save memory on the build +>worker. + +### Properties for the instance + +- **Collation:** Finnish\_Swedish\_CI\_AS +- **InstallSharedDir:** C:\Program Files\Microsoft SQL Server +- **InstallSharedWOWDir:** C:\Program Files (x86)\Microsoft SQL Server +- **DatabaseServerName:** `$env:COMPUTERNAME` +- **DatabaseInstanceName:** DSCSQLTEST + ## SqlServerRole **Run order:** 3 diff --git a/Tests/Unit/CommonResourceHelper.Tests.ps1 b/Tests/Unit/CommonResourceHelper.Tests.ps1 deleted file mode 100644 index 3206ee4cd..000000000 --- a/Tests/Unit/CommonResourceHelper.Tests.ps1 +++ /dev/null @@ -1,210 +0,0 @@ -<# - .SYNOPSIS - Automated unit test for helper functions in module CommonResourceHelper. - - .NOTES - To run this script locally, please make sure to first run the bootstrap - script. Read more at - https://github.com/PowerShell/SqlServerDsc/blob/dev/CONTRIBUTING.md#bootstrap-script-assert-testenvironment -#> - -Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') - -if (Test-SkipContinuousIntegrationTask -Type 'Unit') -{ - return -} - -$script:helperModuleName = 'CommonResourceHelper' - -Describe "$script:helperModuleName Unit Tests" { - BeforeAll { - # Import the CommonResourceHelper module to test - $dscResourcesFolderFilePath = Join-Path -Path (Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent) ` - -ChildPath 'DscResources' - - Import-Module -Name (Join-Path -Path $dscResourcesFolderFilePath ` - -ChildPath "$script:helperModuleName.psm1") -Force - } - - InModuleScope $script:helperModuleName { - Describe 'Get-LocalizedData' { - $mockTestPath = { - return $mockTestPathReturnValue - } - - $mockImportLocalizedData = { - $BaseDirectory | Should -Be $mockExpectedLanguagePath - } - - BeforeEach { - Mock -CommandName Test-Path -MockWith $mockTestPath -Verifiable - Mock -CommandName Import-LocalizedData -MockWith $mockImportLocalizedData -Verifiable - } - - Context 'When loading localized data for Swedish' { - $mockExpectedLanguagePath = 'sv-SE' - $mockTestPathReturnValue = $true - - It 'Should call Import-LocalizedData with sv-SE language' { - Mock -CommandName Join-Path -MockWith { - return 'sv-SE' - } -Verifiable - - { Get-LocalizedData -ResourceName 'DummyResource' } | Should -Not -Throw - - Assert-MockCalled -CommandName Join-Path -Exactly -Times 2 -Scope It - Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It - } - - $mockExpectedLanguagePath = 'en-US' - $mockTestPathReturnValue = $false - - It 'Should call Import-LocalizedData and fallback to en-US if sv-SE language does not exist' { - Mock -CommandName Join-Path -MockWith { - return $ChildPath - } -Verifiable - - { Get-LocalizedData -ResourceName 'DummyResource' } | Should -Not -Throw - - Assert-MockCalled -CommandName Join-Path -Exactly -Times 3 -Scope It - Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It - } - - Context 'When $ScriptRoot is set to a path' { - $mockExpectedLanguagePath = 'sv-SE' - $mockTestPathReturnValue = $true - - It 'Should call Import-LocalizedData with sv-SE language' { - Mock -CommandName Join-Path -MockWith { - return 'sv-SE' - } -Verifiable - - { Get-LocalizedData -ResourceName 'DummyResource' -ScriptRoot '.' } | Should -Not -Throw - - Assert-MockCalled -CommandName Join-Path -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It - } - - $mockExpectedLanguagePath = 'en-US' - $mockTestPathReturnValue = $false - - It 'Should call Import-LocalizedData and fallback to en-US if sv-SE language does not exist' { - Mock -CommandName Join-Path -MockWith { - return $ChildPath - } -Verifiable - - { Get-LocalizedData -ResourceName 'DummyResource' -ScriptRoot '.' } | Should -Not -Throw - - Assert-MockCalled -CommandName Join-Path -Exactly -Times 2 -Scope It - Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It - } - } - } - - Context 'When loading localized data for English' { - Mock -CommandName Join-Path -MockWith { - return 'en-US' - } -Verifiable - - $mockExpectedLanguagePath = 'en-US' - $mockTestPathReturnValue = $true - - It 'Should call Import-LocalizedData with en-US language' { - { Get-LocalizedData -ResourceName 'DummyResource' } | Should -Not -Throw - } - } - - Assert-VerifiableMock - } - - Describe 'New-InvalidResultException' { - Context 'When calling with Message parameter only' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - - { New-InvalidResultException -Message $mockErrorMessage } | Should -Throw $mockErrorMessage - } - } - - Context 'When calling with both the Message and ErrorRecord parameter' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - $mockExceptionErrorMessage = 'Mocked exception error message' - - $mockException = New-Object -TypeName System.Exception -ArgumentList $mockExceptionErrorMessage - $mockErrorRecord = New-Object -TypeName System.Management.Automation.ErrorRecord -ArgumentList $mockException, $null, 'InvalidResult', $null - - { New-InvalidResultException -Message $mockErrorMessage -ErrorRecord $mockErrorRecord } | Should -Throw ('System.Exception: {0} ---> System.Exception: {1}' -f $mockErrorMessage, $mockExceptionErrorMessage) - } - } - - Assert-VerifiableMock - } - - Describe 'New-ObjectNotFoundException' { - Context 'When calling with Message parameter only' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - - { New-ObjectNotFoundException -Message $mockErrorMessage } | Should -Throw $mockErrorMessage - } - } - - Context 'When calling with both the Message and ErrorRecord parameter' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - $mockExceptionErrorMessage = 'Mocked exception error message' - - $mockException = New-Object -TypeName System.Exception -ArgumentList $mockExceptionErrorMessage - $mockErrorRecord = New-Object -TypeName System.Management.Automation.ErrorRecord -ArgumentList $mockException, $null, 'InvalidResult', $null - - { New-ObjectNotFoundException -Message $mockErrorMessage -ErrorRecord $mockErrorRecord } | Should -Throw ('System.Exception: {0} ---> System.Exception: {1}' -f $mockErrorMessage, $mockExceptionErrorMessage) - } - } - - Assert-VerifiableMock - } - - Describe 'New-InvalidOperationException' { - Context 'When calling with Message parameter only' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - - { New-InvalidOperationException -Message $mockErrorMessage } | Should -Throw $mockErrorMessage - } - } - - Context 'When calling with both the Message and ErrorRecord parameter' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - $mockExceptionErrorMessage = 'Mocked exception error message' - - $mockException = New-Object -TypeName System.Exception -ArgumentList $mockExceptionErrorMessage - $mockErrorRecord = New-Object -TypeName System.Management.Automation.ErrorRecord -ArgumentList $mockException, $null, 'InvalidResult', $null - - { New-InvalidOperationException -Message $mockErrorMessage -ErrorRecord $mockErrorRecord } | Should -Throw ('System.InvalidOperationException: {0} ---> System.Exception: {1}' -f $mockErrorMessage, $mockExceptionErrorMessage) - } - } - - Assert-VerifiableMock - } - - Describe 'New-InvalidArgumentException' { - Context 'When calling with both the Message and ArgumentName parameter' { - It 'Should throw the correct error' { - $mockErrorMessage = 'Mocked error' - $mockArgumentName = 'MockArgument' - - { New-InvalidArgumentException -Message $mockErrorMessage -ArgumentName $mockArgumentName } | Should -Throw ('Parameter name: {0}' -f $mockArgumentName) - } - } - - Assert-VerifiableMock - } - } -} diff --git a/Tests/Unit/SqlServerDSCHelper.Tests.ps1 b/Tests/Unit/DscResource.Common.Tests.ps1 similarity index 64% rename from Tests/Unit/SqlServerDSCHelper.Tests.ps1 rename to Tests/Unit/DscResource.Common.Tests.ps1 index 451d7e6ff..6e175df60 100644 --- a/Tests/Unit/SqlServerDSCHelper.Tests.ps1 +++ b/Tests/Unit/DscResource.Common.Tests.ps1 @@ -1,6 +1,6 @@ <# .SYNOPSIS - Automated unit test for helper functions in module SqlServerDscHelper. + Automated unit test for helper functions in module DscResource.Common. .NOTES To run this script locally, please make sure to first run the bootstrap @@ -8,10 +8,6 @@ https://github.com/PowerShell/SqlServerDsc/blob/dev/CONTRIBUTING.md#bootstrap-script-assert-testenvironment #> -# To run these tests, we have to fake login credentials -[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingConvertToSecureStringWithPlainText', '')] -param () - Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') if (Test-SkipContinuousIntegrationTask -Type 'Unit') @@ -19,1760 +15,2392 @@ if (Test-SkipContinuousIntegrationTask -Type 'Unit') return } -# Unit Test Template Version: 1.1.0 - -$script:helperModuleName = 'SqlServerDscHelper' - -[System.String] $script:moduleRoot = Split-Path -Parent (Split-Path -Parent $PSScriptRoot) -if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests'))) -or ` - (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1'))) ) -{ - & git @('clone','https://github.com/PowerShell/DscResource.Tests.git',(Join-Path -Path $script:moduleRoot -ChildPath '\DSCResource.Tests\')) -} +# Import the DscResource.Common module to test +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules\DscResource.Common' -Import-Module (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1') -Force -Import-Module (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent | Split-Path -Parent) -ChildPath "$script:helperModuleName.psm1") -Scope Global -Force +Import-Module -Name (Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.Common.psm1') -Force # Loading mocked classes -Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) - -Add-Type -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SqlPowerShellSqlExecutionException.cs') -Import-Module -Name (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Global -Force - -# Begin Testing -InModuleScope $script:helperModuleName { - $mockNewObject_MicrosoftAnalysisServicesServer = { - return New-Object -TypeName Object | - Add-Member -MemberType ScriptMethod -Name Connect -Value { - param( - [Parameter(Mandatory = $true)] - [ValidateNotNullOrEmpty()] - [System.String] - $dataSource - ) - - if ($dataSource -ne $mockExpectedDataSource) - { - throw ("Datasource was expected to be '{0}', but was '{1}'." -f $mockExpectedDataSource,$dataSource) - } - - if ($mockThrowInvalidOperation) - { - throw 'Unable to connect.' - } - } -PassThru -Force - } +Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') +Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SqlPowerShellSqlExecutionException.cs') - $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter = { - $TypeName -eq 'Microsoft.AnalysisServices.Server' - } +# Importing SQLPS stubs +Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global - $mockNewObject_MicrosoftDatabaseEngine = { - <# - $ArgumentList[0] will contain the ServiceInstance when calling mock New-Object. - But since the mock New-Object will also be called without arguments, we first - have to evaluate if $ArgumentList contains values. - #> - if( $ArgumentList.Count -gt 0) - { - $serverInstance = $ArgumentList[0] - } +InModuleScope 'DscResource.Common' { + Describe 'DscResource.Common\Test-DscParameterState' -Tag 'TestDscParameterState' { + Context -Name 'When passing values' -Fixture { + It 'Should return true for two identical tables' { + $mockDesiredValues = @{ Example = 'test' } - return New-Object -TypeName Object | - Add-Member -MemberType ScriptProperty -Name Status -Value { - if ($mockExpectedDatabaseEngineInstance -eq 'MSSQLSERVER') - { - $mockExpectedServiceInstance = $mockExpectedDatabaseEngineServer - } - else - { - $mockExpectedServiceInstance = "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + $testParameters = @{ + CurrentValues = $mockDesiredValues + DesiredValues = $mockDesiredValues } - if ( $this.ConnectionContext.ServerInstance -eq $mockExpectedServiceInstance ) - { - return 'Online' - } - else - { - return $null - } - } -PassThru | - Add-Member -MemberType NoteProperty -Name ConnectionContext -Value ( - New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name ServerInstance -Value $serverInstance -PassThru | - #Add-Member -MemberType ScriptProperty -Name LoginSecure -Value { [System.Boolean] $mockExpectedDatabaseEngineLoginSecure } -PassThru -Force | - Add-Member -MemberType NoteProperty -Name LoginSecure -Value $true -PassThru | - Add-Member -MemberType NoteProperty -Name Login -Value '' -PassThru | - Add-Member -MemberType NoteProperty -Name SecurePassword -Value $null -PassThru | - Add-Member -MemberType NoteProperty -Name ConnectAsUser -Value $false -PassThru | - Add-Member -MemberType NoteProperty -Name ConnectAsUserPassword -Value '' -PassThru | - Add-Member -MemberType NoteProperty -Name ConnectAsUserName -Value '' -PassThru | - Add-Member -MemberType ScriptMethod -Name Connect -Value { - if ($mockExpectedDatabaseEngineInstance -eq 'MSSQLSERVER') - { - $mockExpectedServiceInstance = $mockExpectedDatabaseEngineServer - } - else - { - $mockExpectedServiceInstance = "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" - } + Test-DscParameterState @testParameters | Should -Be $true + } - if ($this.serverInstance -ne $mockExpectedServiceInstance) - { - throw ("Mock method Connect() was expecting ServerInstance to be '{0}', but was '{1}'." -f $mockExpectedServiceInstance, $this.serverInstance ) - } + It 'Should return false when a value is different for [System.String]' { + $mockCurrentValues = @{ Example = [System.String] 'something' } + $mockDesiredValues = @{ Example = [System.String] 'test' } - if ($mockThrowInvalidOperation) - { - throw 'Unable to connect.' - } - } -PassThru -Force - ) -PassThru -Force - } + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } - $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter = { - $TypeName -eq 'Microsoft.SqlServer.Management.Smo.Server' - } + Test-DscParameterState @testParameters | Should -Be $false + } - $mockThrowLocalizedMessage = { - throw $Message - } + It 'Should return false when a value is different for [System.Int32]' { + $mockCurrentValues = @{ Example = [System.Int32] 1 } + $mockDesiredValues = @{ Example = [System.Int32] 2 } - $mockSqlMajorVersion = 13 - $mockInstanceName = 'TEST' - - $mockSetupCredentialUserName = 'TestUserName12345' - $mockSetupCredentialPassword = 'StrongOne7.' - $mockSetupCredentialSecurePassword = ConvertTo-SecureString -String $mockSetupCredentialPassword -AsPlainText -Force - $mockSetupCredential = New-Object -TypeName PSCredential -ArgumentList ($mockSetupCredentialUserName, $mockSetupCredentialSecurePassword) - - $mockLocalSystemAccountUserName = 'NT AUTHORITY\SYSTEM' - $mockLocalSystemAccountCredential = New-Object System.Management.Automation.PSCredential $mockLocalSystemAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) - $mockManagedServiceAccountUserName = 'CONTOSO\msa$' - $mockManagedServiceAccountCredential = New-Object System.Management.Automation.PSCredential $mockManagedServiceAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) - $mockDomainAccountUserName = 'CONTOSO\User1' - $mockLocalServiceAccountUserName = 'NT SERVICE\MyService' - $mockLocalServiceAccountCredential = New-Object System.Management.Automation.PSCredential $mockLocalServiceAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) - $mockDomainAccountCredential = New-Object System.Management.Automation.PSCredential $mockDomainAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) - $mockInnerException = New-Object System.Exception "This is a mock inner excpetion object" - $mockInnerException | Add-Member -Name 'Number' -Value 2 -MemberType NoteProperty - $mockException = New-Object System.Exception "This is a mock exception object", $mockInnerException - $mockException | Add-Member -Name 'Number' -Value 1 -MemberType NoteProperty - - - Describe 'Testing Restart-SqlService' { - Context 'Restart-SqlService standalone instance' { - BeforeEach { - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'MSSQLSERVER' - InstanceName = '' - ServiceName = 'MSSQLSERVER' - Status = $mockDynamicStatus - IsClustered = $false - } - } -Verifiable -ParameterFilter { $InstanceName -eq 'MSSQLSERVER' } + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'NOCLUSTERCHECK' - InstanceName = 'NOCLUSTERCHECK' - ServiceName = 'NOCLUSTERCHECK' - Status = $mockDynamicStatus - IsClustered = $true - } - } -Verifiable -ParameterFilter { $InstanceName -eq 'NOCLUSTERCHECK' } + Test-DscParameterState @testParameters | Should -Be $false + } - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'NOCONNECT' - InstanceName = 'NOCONNECT' - ServiceName = 'NOCONNECT' - Status = $mockDynamicStatus - IsClustered = $true - } - } -Verifiable -ParameterFilter { $InstanceName -eq 'NOCONNECT' } + It 'Should return false when a value is different for [Int16]' { + $mockCurrentValues = @{ Example = [System.Int16] 1 } + $mockDesiredValues = @{ Example = [System.Int16] 2 } - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'NOAGENT' - InstanceName = 'NOAGENT' - ServiceName = 'NOAGENT' - Status = $mockDynamicStatus - } - } -Verifiable -ParameterFilter { $InstanceName -eq 'NOAGENT' } + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'STOPPEDAGENT' - InstanceName = 'STOPPEDAGENT' - ServiceName = 'STOPPEDAGENT' - Status = $mockDynamicStatus - } - } -Verifiable -ParameterFilter { $InstanceName -eq 'STOPPEDAGENT' } + Test-DscParameterState @testParameters | Should -Be $false } - BeforeAll { - ## SQL instance with running SQL Agent Service - Mock -CommandName Get-Service -MockWith { - return @{ - Name = 'MSSQLSERVER' - DisplayName = 'Microsoft SQL Server (MSSQLSERVER)' - DependentServices = @( - @{ - Name = 'SQLSERVERAGENT' - DisplayName = 'SQL Server Agent (MSSQLSERVER)' - Status = 'Running' - DependentServices = @() - } - ) - } - } -Verifiable -ParameterFilter { $Name -eq 'MSSQLSERVER' } + It 'Should return false when a value is different for [UInt16]' { + $mockCurrentValues = @{ Example = [System.UInt16] 1 } + $mockDesiredValues = @{ Example = [System.UInt16] 2 } - ## SQL instance with no installed SQL Agent Service - Mock -CommandName Get-Service -MockWith { - return @{ - Name = 'MSSQL$NOAGENT' - DisplayName = 'Microsoft SQL Server (NOAGENT)' - DependentServices = @() - } - } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$NOAGENT' } + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } - ## SQL instance with no installed SQL Agent Service - Mock -CommandName Get-Service -MockWith { - return @{ - Name = 'MSSQL$NOCLUSTERCHECK' - DisplayName = 'Microsoft SQL Server (NOCLUSTERCHECK)' - DependentServices = @() - } - } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$NOCLUSTERCHECK' } + Test-DscParameterState @testParameters | Should -Be $false + } - ## SQL instance with no installed SQL Agent Service - Mock -CommandName Get-Service -MockWith { - return @{ - Name = 'MSSQL$NOCONNECT' - DisplayName = 'Microsoft SQL Server (NOCONNECT)' - DependentServices = @() - } - } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$NOCONNECT' } + It 'Should return false when a value is different for [Boolean]' { + $mockCurrentValues = @{ Example = [System.Boolean] $true } + $mockDesiredValues = @{ Example = [System.Boolean] $false } - ## SQL instance with stopped SQL Agent Service - Mock -CommandName Get-Service -MockWith { - return @{ - Name = 'MSSQL$STOPPEDAGENT' - DisplayName = 'Microsoft SQL Server (STOPPEDAGENT)' - DependentServices = @( - @{ - Name = 'SQLAGENT$STOPPEDAGENT' - DisplayName = 'SQL Server Agent (STOPPEDAGENT)' - Status = 'Stopped' - DependentServices = @() - } - ) - } - } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$STOPPEDAGENT' } + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } - Mock -CommandName Restart-Service -Verifiable - Mock -CommandName Start-Service -Verifiable + Test-DscParameterState @testParameters | Should -Be $false } - $mockDynamicStatus = 'Online' + It 'Should return false when a value is missing' { + $mockCurrentValues = @{ } + $mockDesiredValues = @{ Example = 'test' } - It 'Should restart SQL Service and running SQL Agent service' { - { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'MSSQLSERVER' } | Should -Not -Throw + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } - Assert-MockCalled -CommandName Connect-SQL -ParameterFilter { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 1 + Test-DscParameterState @testParameters | Should -Be $false } - It 'Should restart SQL Service, and not do cluster cluster check' { - Mock -CommandName Get-CimInstance + It 'Should return true when only a specified value matches, but other non-listed values do not' { + $mockCurrentValues = @{ Example = 'test'; SecondExample = 'true' } + $mockDesiredValues = @{ Example = 'test'; SecondExample = 'false' } - { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'NOCLUSTERCHECK' -SkipClusterCheck } | Should -Not -Throw + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + ValuesToCheck = @('Example') + } - Assert-MockCalled -CommandName Connect-SQL -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 - Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 0 + Test-DscParameterState @testParameters | Should -Be $true } - It 'Should restart SQL Service, and not do cluster cluster check nor check online status' { - Mock -CommandName Get-CimInstance + It 'Should return false when only specified values do not match, but other non-listed values do ' { + $mockCurrentValues = @{ Example = 'test'; SecondExample = 'true' } + $mockDesiredValues = @{ Example = 'test'; SecondExample = 'false' } - { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'NOCONNECT' -SkipClusterCheck -SkipWaitForOnline } | Should -Not -Throw + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + ValuesToCheck = @('SecondExample') + } - Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Connect-SQL -Scope It -Exactly -Times 0 - Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 - Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 0 + Test-DscParameterState @testParameters | Should -Be $false } - It 'Should restart SQL Service and not try to restart missing SQL Agent service' { - { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'NOAGENT' } | Should -Not -Throw + It 'Should return false when an empty hash table is used in the current values' { + $mockCurrentValues = @{ } + $mockDesiredValues = @{ Example = 'test'; SecondExample = 'false' } - Assert-MockCalled -CommandName Connect-SQL { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + Test-DscParameterState @testParameters | Should -Be $false } - It 'Should restart SQL Service and not try to restart stopped SQL Agent service' { - { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'STOPPEDAGENT' } | Should -Not -Throw + It 'Should return true when evaluating a table against a CimInstance' { + $mockCurrentValues = @{ Handle = '0'; ProcessId = '1000' } - Assert-MockCalled -CommandName Connect-SQL { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 - } + $mockWin32ProcessProperties = @{ + Handle = 0 + ProcessId = 1000 + } - Context 'When it fails to connect to the instance within the timeout period' { - BeforeEach { - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'MSSQLSERVER' - InstanceName = '' - ServiceName = 'MSSQLSERVER' - Status = $mockDynamicStatus - } - } -Verifiable -ParameterFilter { $InstanceName -eq 'MSSQLSERVER' } + $mockNewCimInstanceParameters = @{ + ClassName = 'Win32_Process' + Property = $mockWin32ProcessProperties + Key = 'Handle' + ClientOnly = $true } - $mockDynamicStatus = 'Offline' + $mockDesiredValues = New-CimInstance @mockNewCimInstanceParameters - It 'Should throw the correct error message' { - $errorMessage = $localizedData.FailedToConnectToInstanceTimeout -f $env:ComputerName, 'MSSQLSERVER', 1 + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + ValuesToCheck = @('Handle','ProcessId') + } - { - Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'MSSQLSERVER' -Timeout 1 - } | Should -Throw $errorMessage + Test-DscParameterState @testParameters | Should -Be $true + } - Assert-MockCalled -CommandName Connect-SQL -ParameterFilter { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 + It 'Should return false when evaluating a table against a CimInstance and a value is wrong' { + $mockCurrentValues = @{ Handle = '1'; ProcessId = '1000' } - Assert-MockCalled -CommandName Connect-SQL -ParameterFilter { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $true - } -Scope It -Exactly -Times 1 + $mockWin32ProcessProperties = @{ + Handle = 0 + ProcessId = 1000 } + + $mockNewCimInstanceParameters = @{ + ClassName = 'Win32_Process' + Property = $mockWin32ProcessProperties + Key = 'Handle' + ClientOnly = $true + } + + $mockDesiredValues = New-CimInstance @mockNewCimInstanceParameters + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + ValuesToCheck = @('Handle','ProcessId') + } + + Test-DscParameterState @testParameters | Should -Be $false + } + + It 'Should return true when evaluating a hash table containing an array' { + $mockCurrentValues = @{ Example = 'test'; SecondExample = @('1','2') } + $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + Test-DscParameterState @testParameters | Should -Be $true + } + + It 'Should return false when evaluating a hash table containing an array with wrong values' { + $mockCurrentValues = @{ Example = 'test'; SecondExample = @('A','B') } + $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + Test-DscParameterState @testParameters | Should -Be $false + } + + It 'Should return false when evaluating a hash table containing an array, but the CurrentValues are missing an array' { + $mockCurrentValues = @{ Example = 'test' } + $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + Test-DscParameterState @testParameters | Should -Be $false + } + + It 'Should return false when evaluating a hash table containing an array, but the property i CurrentValues is $null' { + $mockCurrentValues = @{ Example = 'test'; SecondExample = $null } + $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + Test-DscParameterState @testParameters | Should -Be $false } } - Context 'Restart-SqlService clustered instance' { - BeforeEach { - Mock -CommandName Connect-SQL -MockWith { - return @{ - Name = 'MSSQLSERVER' - InstanceName = '' - ServiceName = 'MSSQLSERVER' - IsClustered = $true - Status = $mockDynamicStatus + Context -Name 'When passing invalid types for DesiredValues' -Fixture { + It 'Should throw the correct error when DesiredValues is of wrong type' { + $mockCurrentValues = @{ Example = 'something' } + $mockDesiredValues = 'NotHashTable' + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + $mockCorrectErrorMessage = ($script:localizedData.PropertyTypeInvalidForDesiredValues -f $testParameters.DesiredValues.GetType().Name) + { Test-DscParameterState @testParameters } | Should -Throw $mockCorrectErrorMessage + } + + It 'Should write a warning when DesiredValues contain an unsupported type' { + Mock -CommandName Write-Warning -Verifiable + + # This is a dummy type to test with a type that could never be a correct one. + class MockUnknownType + { + [ValidateNotNullOrEmpty()] + [System.String] + $Property1 + + [ValidateNotNullOrEmpty()] + [System.String] + $Property2 + + MockUnknownType() + { } - } -Verifiable -ParameterFilter { ($ServerName -eq 'CLU01') -and ($InstanceName -eq 'MSSQLSERVER') } + } - Mock -CommandName Connect-SQL -MockWith { + $mockCurrentValues = @{ Example = New-Object -TypeName MockUnknownType } + $mockDesiredValues = @{ Example = New-Object -TypeName MockUnknownType } + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + } + + Test-DscParameterState @testParameters | Should -Be $false + + Assert-MockCalled -CommandName Write-Warning -Exactly -Times 1 + } + } + + Context -Name 'When passing an CimInstance as DesiredValue and ValuesToCheck is $null' -Fixture { + It 'Should throw the correct error' { + $mockCurrentValues = @{ Example = 'something' } + + $mockWin32ProcessProperties = @{ + Handle = 0 + ProcessId = 1000 + } + + $mockNewCimInstanceParameters = @{ + ClassName = 'Win32_Process' + Property = $mockWin32ProcessProperties + Key = 'Handle' + ClientOnly = $true + } + + $mockDesiredValues = New-CimInstance @mockNewCimInstanceParameters + + $testParameters = @{ + CurrentValues = $mockCurrentValues + DesiredValues = $mockDesiredValues + ValuesToCheck = $null + } + + $mockCorrectErrorMessage = $script:localizedData.PropertyTypeInvalidForValuesToCheck + { Test-DscParameterState @testParameters } | Should -Throw $mockCorrectErrorMessage + } + } + + Assert-VerifiableMock + } + + Describe 'DscResource.Common\Get-RegistryPropertyValue' -Tag 'GetRegistryPropertyValue' { + BeforeAll { + $mockWrongRegistryPath = 'HKLM:\SOFTWARE\AnyPath' + $mockCorrectRegistryPath = 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\RS' + $mockPropertyName = 'InstanceName' + $mockPropertyValue = 'AnyValue' + } + + Context 'When there are no property in the registry' { + BeforeAll { + Mock -CommandName Get-ItemProperty -MockWith { return @{ - Name = 'NAMEDINSTANCE' - InstanceName = 'NAMEDINSTANCE' - ServiceName = 'NAMEDINSTANCE' - IsClustered = $true - Status = $mockDynamicStatus + 'UnknownProperty' = $mockPropertyValue } - } -Verifiable -ParameterFilter { ($ServerName -eq 'CLU01') -and ($InstanceName -eq 'NAMEDINSTANCE') } + } + } - Mock -CommandName Connect-SQL -MockWith { + It 'Should return $null' { + $result = Get-RegistryPropertyValue -Path $mockWrongRegistryPath -Name $mockPropertyName + $result | Should -BeNullOrEmpty + + Assert-MockCalled Get-ItemProperty -Exactly -Times 1 -Scope It + } + } + + Context 'When the call to Get-ItemProperty throws an error (i.e. when the path does not exist)' { + BeforeAll { + Mock -CommandName Get-ItemProperty -MockWith { + throw 'mocked error' + } + } + + It 'Should not throw an error, but return $null' { + $result = Get-RegistryPropertyValue -Path $mockWrongRegistryPath -Name $mockPropertyName + $result | Should -BeNullOrEmpty + + Assert-MockCalled Get-ItemProperty -Exactly -Times 1 -Scope It + } + } + + Context 'When there are a property in the registry' { + BeforeAll { + $mockGetItemProperty_InstanceName = { return @{ - Name = 'STOPPEDAGENT' - InstanceName = 'STOPPEDAGENT' - ServiceName = 'STOPPEDAGENT' - IsClustered = $true - Status = $mockDynamicStatus + $mockPropertyName = $mockPropertyValue } - } -Verifiable -ParameterFilter { ($ServerName -eq 'CLU01') -and ($InstanceName -eq 'STOPPEDAGENT') } + } + + $mockGetItemProperty_InstanceName_ParameterFilter = { + $Path -eq $mockCorrectRegistryPath ` + -and $Name -eq $mockPropertyName + } + + Mock -CommandName Get-ItemProperty ` + -MockWith $mockGetItemProperty_InstanceName ` + -ParameterFilter $mockGetItemProperty_InstanceName_ParameterFilter } - BeforeAll { - Mock -CommandName Get-CimInstance -MockWith { - @('MSSQLSERVER','NAMEDINSTANCE','STOPPEDAGENT') | ForEach-Object { - $mock = New-Object -TypeName Microsoft.Management.Infrastructure.CimInstance -ArgumentList 'MSCluster_Resource','root/MSCluster' + It 'Should return the correct value' { + $result = Get-RegistryPropertyValue -Path $mockCorrectRegistryPath -Name $mockPropertyName + $result | Should -Be $mockPropertyValue - $mock | Add-Member -MemberType NoteProperty -Name 'Name' -Value "SQL Server ($($_))" -TypeName 'String' - $mock | Add-Member -MemberType NoteProperty -Name 'Type' -Value 'SQL Server' -TypeName 'String' - $mock | Add-Member -MemberType NoteProperty -Name 'PrivateProperties' -Value @{ InstanceName = $_ } + Assert-MockCalled Get-ItemProperty -Exactly -Times 1 -Scope It + } + } + } + + Describe 'DscResource.Common\Format-Path' -Tag 'FormatPath' { + BeforeAll { + $mockCorrectPath = 'C:\Correct\Path' + $mockPathWithTrailingBackslash = 'C:\Correct\Path\' + $mockPathWithOnlyQualifier = 'M:' + $mockCorrectQualifierPath = 'M:\' + } + + Context 'When there is a path that is wrongly formatted, but now formatting was requested' { + It 'Should return the same wrongly formatted path' { + $result = Format-Path -Path $mockPathWithTrailingBackslash + $result | Should -BeExactly $mockPathWithTrailingBackslash + } + } + + Context 'When there is a path that is formatted correctly, and using TrailingSlash' { + It 'Should return the same path' { + $result = Format-Path -Path $mockCorrectPath -TrailingSlash + $result | Should -BeExactly $mockCorrectPath + } + } + + Context 'When there is a path that has a trailing backslash, and using TrailingSlash' { + It 'Should return the path without trailing backslash' { + $result = Format-Path -Path $mockPathWithTrailingBackslash -TrailingSlash + $result | Should -BeExactly $mockCorrectPath + } + } + + Context 'When there is a path that has only a qualifier, and using TrailingSlash' { + It 'Should return the path with trailing backslash after the qualifier' { + $result = Format-Path -Path $mockPathWithOnlyQualifier -TrailingSlash + $result | Should -BeExactly $mockCorrectQualifierPath + } + } + } + + # Tests only the parts of the code that does not already get tested thru the other tests. + Describe 'DscResource.Common\Copy-ItemWithRobocopy' -Tag 'CopyItemWithRobocopy' { + BeforeAll { + $mockRobocopyExecutableName = 'Robocopy.exe' + $mockRobocopyExecutableVersionWithoutUnbufferedIO = '6.2.9200.00000' + $mockRobocopyExecutableVersionWithUnbufferedIO = '6.3.9600.16384' + $mockRobocopyExecutableVersion = '' # Set dynamically during runtime + $mockRobocopyArgumentSilent = '/njh /njs /ndl /nc /ns /nfl' + $mockRobocopyArgumentCopySubDirectoriesIncludingEmpty = '/e' + $mockRobocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource = '/purge' + $mockRobocopyArgumentUseUnbufferedIO = '/J' + $mockRobocopyArgumentSourcePath = 'C:\Source\SQL2016' + $mockRobocopyArgumentDestinationPath = 'D:\Temp' + $mockRobocopyArgumentSourcePathWithSpaces = 'C:\Source\SQL2016 STD SP1' + $mockRobocopyArgumentDestinationPathWithSpaces = 'D:\Temp\DSC SQL2016' + + $mockGetCommand = { + return @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'Name' -Value $mockRobocopyExecutableName -PassThru | + Add-Member -MemberType ScriptProperty -Name FileVersionInfo -Value { + return @( ( New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'ProductVersion' -Value $mockRobocopyExecutableVersion -PassThru -Force + ) ) + } -PassThru -Force + ) + ) + } + + $mockStartSqlSetupProcessExpectedArgument = '' # Set dynamically during runtime + $mockStartSqlSetupProcessExitCode = 0 # Set dynamically during runtime + + $mockStartSqlSetupProcess_Robocopy = { + if ( $ArgumentList -cne $mockStartSqlSetupProcessExpectedArgument ) + { + throw "Expected arguments was not the same as the arguments in the function call.`nExpected: '$mockStartSqlSetupProcessExpectedArgument' `n But was: '$ArgumentList'" + } + + return New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'ExitCode' -Value 0 -PassThru -Force + } + + $mockStartSqlSetupProcess_Robocopy_WithExitCode = { + return New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'ExitCode' -Value $mockStartSqlSetupProcessExitCode -PassThru -Force + } + } + + Context 'When Copy-ItemWithRobocopy is called it should return the correct arguments' { + BeforeEach { + Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable + Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy -Verifiable + $mockRobocopyArgumentSourcePathQuoted = '"{0}"' -f $mockRobocopyArgumentSourcePath + $mockRobocopyArgumentDestinationPathQuoted = '"{0}"' -f $mockRobocopyArgumentDestinationPath + } + + + It 'Should use Unbuffered IO when copying' { + $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO + + $mockStartSqlSetupProcessExpectedArgument = + $mockRobocopyArgumentSourcePathQuoted, + $mockRobocopyArgumentDestinationPathQuoted, + $mockRobocopyArgumentCopySubDirectoriesIncludingEmpty, + $mockRobocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource, + $mockRobocopyArgumentUseUnbufferedIO, + $mockRobocopyArgumentSilent -join ' ' + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should not use Unbuffered IO when copying' { + $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithoutUnbufferedIO + + $mockStartSqlSetupProcessExpectedArgument = + $mockRobocopyArgumentSourcePathQuoted, + $mockRobocopyArgumentDestinationPathQuoted, + $mockRobocopyArgumentCopySubDirectoriesIncludingEmpty, + $mockRobocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource, + '', + $mockRobocopyArgumentSilent -join ' ' + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + } + + Context 'When Copy-ItemWithRobocopy throws an exception it should return the correct error messages' { + BeforeEach { + $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO + + Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable + Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy_WithExitCode -Verifiable + } + + It 'Should throw the correct error message when error code is 8' { + $mockStartSqlSetupProcessExitCode = 8 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Throw "Robocopy reported errors when copying files. Error code: $mockStartSqlSetupProcessExitCode." + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should throw the correct error message when error code is 16' { + $mockStartSqlSetupProcessExitCode = 16 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Throw "Robocopy reported errors when copying files. Error code: $mockStartSqlSetupProcessExitCode." + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should throw the correct error message when error code is greater than 7 (but not 8 or 16)' { + $mockStartSqlSetupProcessExitCode = 9 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Throw "Robocopy reported that failures occurred when copying files. Error code: $mockStartSqlSetupProcessExitCode." + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + } + + Context 'When Copy-ItemWithRobocopy is called and finishes successfully it should return the correct exit code' { + BeforeEach { + $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO + + Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable + Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy_WithExitCode -Verifiable + } + + AfterEach { + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should finish successfully with exit code 1' { + $mockStartSqlSetupProcessExitCode = 1 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should finish successfully with exit code 2' { + $mockStartSqlSetupProcessExitCode = 2 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should finish successfully with exit code 3' { + $mockStartSqlSetupProcessExitCode = 3 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePath + DestinationPath = $mockRobocopyArgumentDestinationPath + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + } + } + + Context 'When Copy-ItemWithRobocopy is called with spaces in paths and finishes successfully it should return the correct exit code' { + BeforeEach { + $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO + + Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable + Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy_WithExitCode -Verifiable + } + + AfterEach { + Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It + } + + It 'Should finish successfully with exit code 1' { + $mockStartSqlSetupProcessExitCode = 1 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePathWithSpaces + DestinationPath = $mockRobocopyArgumentDestinationPathWithSpaces + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + + } + + It 'Should finish successfully with exit code 2' { + $mockStartSqlSetupProcessExitCode = 2 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePathWithSpaces + DestinationPath = $mockRobocopyArgumentDestinationPathWithSpaces + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + + } + + It 'Should finish successfully with exit code 3' { + $mockStartSqlSetupProcessExitCode = 3 + + $copyItemWithRobocopyParameter = @{ + Path = $mockRobocopyArgumentSourcePathWithSpaces + DestinationPath = $mockRobocopyArgumentDestinationPathWithSpaces + } + + { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw + } + } + } + + Describe 'DscResource.Common\Get-TemporaryFolder' -Tag 'GetTemporaryFolder' { + BeforeAll { + $mockExpectedTempPath = [IO.Path]::GetTempPath() + } + + Context 'When using Get-TemporaryFolder' { + It 'Should return the correct temporary path' { + Get-TemporaryFolder | Should -BeExactly $mockExpectedTempPath + } + } + } + + Describe 'DscResource.Common\Invoke-InstallationMediaCopy' -Tag 'InvokeInstallationMediaCopy' { + BeforeAll { + $mockSourcePathUNC = '\\server\share' + $mockSourcePathUNCWithLeaf = '\\server\share\leaf' + $mockSourcePathGuid = 'cc719562-0f46-4a16-8605-9f8a47c70402' + $mockDestinationPath = 'TestDrive:\' + + $mockShareCredentialUserName = 'COMPANY\SqlAdmin' + $mockShareCredentialPassword = 'dummyPassW0rd' + $mockShareCredential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList @( + $mockShareCredentialUserName, + ($mockShareCredentialPassword | ConvertTo-SecureString -AsPlainText -Force) + ) + + $mockGetTemporaryFolder = { + return $mockDestinationPath + } - return $mock - } - } -Verifiable -ParameterFilter { ($ClassName -eq 'MSCluster_Resource') -and ($Filter -eq "Type = 'SQL Server'") } + $mockNewGuid = { + return New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'Guid' -Value $mockSourcePathGuid -PassThru -Force + } - Mock -CommandName Get-CimAssociatedInstance -MockWith { - $mock = New-Object -TypeName Microsoft.Management.Infrastructure.CimInstance -ArgumentList 'MSCluster_Resource','root/MSCluster' + Mock -CommandName Connect-UncPath + Mock -CommandName Disconnect-UncPath + Mock -CommandName Copy-ItemWithRobocopy + Mock -CommandName Get-TemporaryFolder -MockWith $mockGetTemporaryFolder + Mock -CommandName New-Guid -MockWith $mockNewGuid + } - $mock | Add-Member -MemberType NoteProperty -Name 'Name' -Value "SQL Server Agent ($($InputObject.PrivateProperties.InstanceName))" -TypeName 'String' - $mock | Add-Member -MemberType NoteProperty -Name 'Type' -Value 'SQL Server Agent' -TypeName 'String' - $mock | Add-Member -MemberType NoteProperty -Name 'State' -Value (@{ $true = 3; $false = 2 }[($InputObject.PrivateProperties.InstanceName -eq 'STOPPEDAGENT')]) -TypeName 'Int32' + Context 'When invoking installation media copy, using SourcePath containing leaf' { + It 'Should call the correct mocks' { + { + $invokeInstallationMediaCopyParameters = @{ + SourcePath = $mockSourcePathUNCWithLeaf + SourceCredential = $mockShareCredential + } - return $mock - } -Verifiable -ParameterFilter { $ResultClassName -eq 'MSCluster_Resource' } + Invoke-InstallationMediaCopy @invokeInstallationMediaCopyParameters + } | Should -Not -Throw - Mock -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Verifiable - Mock -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Verifiable + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName New-Guid -Exactly -Times 0 -Scope 'It' + Assert-MockCalled -CommandName Get-TemporaryFolder -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Copy-ItemWithRobocopy -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 1 -Scope 'It' } - $mockDynamicStatus = 'Online' + It 'Should return the correct destination path' { + $invokeInstallationMediaCopyParameters = @{ + SourcePath = $mockSourcePathUNCWithLeaf + SourceCredential = $mockShareCredential + PassThru = $true + } - It 'Should restart SQL Server and SQL Agent resources for a clustered default instance' { - { Restart-SqlService -SQLServer 'CLU01' } | Should -Not -Throw + $invokeInstallationMediaCopyResult = Invoke-InstallationMediaCopy @invokeInstallationMediaCopyParameters - Assert-MockCalled -CommandName Connect-SQL { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-CimAssociatedInstance -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Scope It -Exactly -Times 2 + $invokeInstallationMediaCopyResult | Should -Be ('{0}leaf' -f $mockDestinationPath) } + } - It 'Should restart SQL Server and SQL Agent resources for a clustered named instance' { - { Restart-SqlService -SQLServer 'CLU01' -SQLInstanceName 'NAMEDINSTANCE' } | Should -Not -Throw + Context 'When invoking installation media copy, using SourcePath without a leaf' { + It 'Should call the correct mocks' { + { + $invokeInstallationMediaCopyParameters = @{ + SourcePath = $mockSourcePathUNC + SourceCredential = $mockShareCredential + } - Assert-MockCalled -CommandName Connect-SQL { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-CimAssociatedInstance -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Scope It -Exactly -Times 2 + Invoke-InstallationMediaCopy @invokeInstallationMediaCopyParameters + } | Should -Not -Throw + + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName New-Guid -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Get-TemporaryFolder -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Copy-ItemWithRobocopy -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 1 -Scope 'It' } - It 'Should not try to restart a SQL Agent resource that is not online' { - { Restart-SqlService -SQLServer 'CLU01' -SQLInstanceName 'STOPPEDAGENT' } | Should -Not -Throw + It 'Should return the correct destination path' { + $invokeInstallationMediaCopyParameters = @{ + SourcePath = $mockSourcePathUNC + SourceCredential = $mockShareCredential + PassThru = $true + } - Assert-MockCalled -CommandName Connect-SQL { - $PSBoundParameters.ContainsKey('ErrorAction') -eq $false - } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Get-CimAssociatedInstance -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Scope It -Exactly -Times 1 - Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Scope It -Exactly -Times 1 + $invokeInstallationMediaCopyResult = Invoke-InstallationMediaCopy @invokeInstallationMediaCopyParameters + $invokeInstallationMediaCopyResult | Should -Be ('{0}{1}' -f $mockDestinationPath, $mockSourcePathGuid) } } } - Describe 'Testing Connect-SQLAnalysis' { - BeforeEach { - Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable - Mock -CommandName New-Object ` - -MockWith $mockNewObject_MicrosoftAnalysisServicesServer ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter ` - -Verifiable + Describe 'DscResource.Common\Connect-UncPath' -Tag 'ConnectUncPath' { + BeforeAll { + $mockSourcePathUNC = '\\server\share' + + $mockShareCredentialUserName = 'COMPANY\SqlAdmin' + $mockShareCredentialPassword = 'dummyPassW0rd' + $mockShareCredential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList @( + $mockShareCredentialUserName, + ($mockShareCredentialPassword | ConvertTo-SecureString -AsPlainText -Force) + ) + + Mock -CommandName New-SmbMapping -MockWith { + return @{ + RemotePath = $mockSourcePathUNC + } + } } - Context 'When connecting to the default instance using Windows Authentication' { - It 'Should not throw when connecting' { - $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME" + Context 'When connecting to a UNC path without credentials (using current credentials)' { + It 'Should call the correct mocks' { + { + $connectUncPathParameters = @{ + RemotePath = $mockSourcePathUNC + } - { Connect-SQLAnalysis } | Should -Not -Throw + Connect-UncPath @connectUncPathParameters + } | Should -Not -Throw - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + Assert-MockCalled -CommandName New-SmbMapping -ParameterFilter { + $RemotePath -eq $mockSourcePathUNC ` + -and $PSBoundParameters.ContainsKey('UserName') -eq $false + } -Exactly -Times 1 -Scope 'It' } } - Context 'When connecting to the named instance using Windows Authentication' { - It 'Should not throw when connecting' { - $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME\$mockInstanceName" + Context 'When connecting to a UNC path with specific credentials' { + It 'Should call the correct mocks' { + { + $connectUncPathParameters = @{ + RemotePath = $mockSourcePathUNC + SourceCredential = $mockShareCredential + } - { Connect-SQLAnalysis -SQLInstanceName $mockInstanceName } | Should -Not -Throw + Connect-UncPath @connectUncPathParameters + } | Should -Not -Throw - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + Assert-MockCalled -CommandName New-SmbMapping -ParameterFilter { + $RemotePath -eq $mockSourcePathUNC ` + -and $UserName -eq $mockShareCredentialUserName + } -Exactly -Times 1 -Scope 'It' } } - Context 'When connecting to the named instance using Windows Authentication impersonation' { - It 'Should not throw when connecting' { - $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME\$mockInstanceName;User ID=$mockSetupCredentialUserName;Password=$mockSetupCredentialPassword" - - { Connect-SQLAnalysis -SQLInstanceName $mockInstanceName -SetupCredential $mockSetupCredential } | Should -Not -Throw + Context 'When connecting to a UNC path and using parameter PassThru' { + It 'Should return the correct MSFT_SmbMapping object' { + $connectUncPathParameters = @{ + RemotePath = $mockSourcePathUNC + SourceCredential = $mockShareCredential + PassThru = $true + } - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + $connectUncPathResult = Connect-UncPath @connectUncPathParameters + $connectUncPathResult.RemotePath | Should -Be $mockSourcePathUNC } } + } - Context 'When connecting to the default instance using the correct service instance but does not return a correct Analysis Service object' { - It 'Should throw the correct error' { - $mockExpectedDataSource = '' + Describe 'DscResource.Common\Disconnect-UncPath' -Tag 'DisconnectUncPath' { + BeforeAll { + $mockSourcePathUNC = '\\server\share' - Mock -CommandName New-Object ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter ` - -Verifiable + Mock -CommandName Remove-SmbMapping + } - $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToAnalysisServicesInstance -f $env:COMPUTERNAME) - { Connect-SQLAnalysis } | Should -Throw $mockCorrectErrorMessage + Context 'When disconnecting from an UNC path' { + It 'Should call the correct mocks' { + { + $disconnectUncPathParameters = @{ + RemotePath = $mockSourcePathUNC + } - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + Disconnect-UncPath @disconnectUncPathParameters + } | Should -Not -Throw + + Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 1 -Scope 'It' } } + } - Context 'When connecting to the default instance using a Analysis Service instance that does not exist' { - It 'Should throw the correct error' { - $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME" + Describe 'DscResource.Common\Test-PendingRestart' -Tag 'TestPendingRestart' { + Context 'When there is a pending reboot' { + BeforeAll { + Mock -CommandName Get-RegistryPropertyValue -MockWith { + return 'AnyValue' + } + } - # Force the mock of Connect() method to throw 'Unable to connect.' - $mockThrowInvalidOperation = $true + It 'Should return $true' { + $testPendingRestartResult = Test-PendingRestart + $testPendingRestartResult | Should -BeTrue - $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToAnalysisServicesInstance -f $env:COMPUTERNAME) - { Connect-SQLAnalysis } | Should -Throw $mockCorrectErrorMessage + Assert-MockCalled -CommandName Get-RegistryPropertyValue -Exactly -Times 1 -Scope 'It' + } + } - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + Context 'When there are no pending reboot' { + BeforeAll { + Mock -CommandName Get-RegistryPropertyValue + } - # Setting it back to the default so it does not disturb other tests. - $mockThrowInvalidOperation = $false + It 'Should return $true' { + $testPendingRestartResult = Test-PendingRestart + $testPendingRestartResult | Should -BeFalse + + Assert-MockCalled -CommandName Get-RegistryPropertyValue -Exactly -Times 1 -Scope 'It' } } + } - # This test is to test the mock so that it throws correct when data source is not the expected data source - Context 'When connecting to the named instance using another data source then expected' { - It 'Should throw the correct error' { - $mockExpectedDataSource = "Force wrong data source" - - $testParameters = @{ - SQLServer = 'DummyHost' - SQLInstanceName = $mockInstanceName + Describe 'DscResource.Common\Start-SqlSetupProcess' -Tag 'StartSqlSetupProcess' { + Context 'When starting a process successfully' { + It 'Should return exit code 0' { + $startSqlSetupProcessParameters = @{ + FilePath = 'powershell.exe' + ArgumentList = '-Command &{Start-Sleep -Seconds 2}' + Timeout = 30 } - $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToAnalysisServicesInstance -f "$($testParameters.SQLServer)\$($testParameters.SQLInstanceName)") - { Connect-SQLAnalysis @testParameters } | Should -Throw $mockCorrectErrorMessage - - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + $processExitCode = Start-SqlSetupProcess @startSqlSetupProcessParameters + $processExitCode | Should -BeExactly 0 } } - Assert-VerifiableMock + Context 'When starting a process and the process does not finish before the timeout period' { + It 'Should throw an error message' { + $startSqlSetupProcessParameters = @{ + FilePath = 'powershell.exe' + ArgumentList = '-Command &{Start-Sleep -Seconds 4}' + Timeout = 2 + } + + { Start-SqlSetupProcess @startSqlSetupProcessParameters } | Should -Throw -ErrorId 'ProcessNotTerminated,Microsoft.PowerShell.Commands.WaitProcessCommand' + } + } } - Describe 'Testing Invoke-Query' { - $mockExpectedQuery = '' + Describe 'DscResource.Common\Restart-SqlService' -Tag 'RestartSqlService' { + Context 'Restart-SqlService standalone instance' { + BeforeEach { + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'MSSQLSERVER' + InstanceName = '' + ServiceName = 'MSSQLSERVER' + Status = $mockDynamicStatus + IsClustered = $false + } + } -Verifiable -ParameterFilter { $InstanceName -eq 'MSSQLSERVER' } - $mockConnectSql = { - return @( - ( - New-Object -TypeName PSObject -Property @{ - Databases = @{ - 'master' = ( - New-Object -TypeName PSObject -Property @{ Name = 'master' } | - Add-Member -MemberType ScriptMethod -Name ExecuteNonQuery -Value { - param - ( - [Parameter()] - [System.String] - $sqlCommand - ) - - if ( $sqlCommand -ne $mockExpectedQuery ) - { - throw - } - } -PassThru | - Add-Member -MemberType ScriptMethod -Name ExecuteWithResults -Value { - param - ( - [Parameter()] - [System.String] - $sqlCommand - ) - - if ( $sqlCommand -ne $mockExpectedQuery ) - { - throw - } - - return New-Object -TypeName System.Data.DataSet - } -PassThru - ) - } + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'NOCLUSTERCHECK' + InstanceName = 'NOCLUSTERCHECK' + ServiceName = 'NOCLUSTERCHECK' + Status = $mockDynamicStatus + IsClustered = $true } - ) - ) - } + } -Verifiable -ParameterFilter { $InstanceName -eq 'NOCLUSTERCHECK' } - BeforeEach { - Mock -CommandName Connect-SQL -MockWith $mockConnectSql -ModuleName $script:dscResourceName -Verifiable - Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable - } + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'NOCONNECT' + InstanceName = 'NOCONNECT' + ServiceName = 'NOCONNECT' + Status = $mockDynamicStatus + IsClustered = $true + } + } -Verifiable -ParameterFilter { $InstanceName -eq 'NOCONNECT' } - $queryParams = @{ - SQLServer = 'Server1' - SQLInstanceName = 'MSSQLSERVER' - Database = 'master' - Query = '' - } + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'NOAGENT' + InstanceName = 'NOAGENT' + ServiceName = 'NOAGENT' + Status = $mockDynamicStatus + } + } -Verifiable -ParameterFilter { $InstanceName -eq 'NOAGENT' } - Context 'Execute a query with no results' { - It 'Should execute the query silently' { - $queryParams.Query = "EXEC sp_configure 'show advanced option', '1'" - $mockExpectedQuery = $queryParams.Query.Clone() + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'STOPPEDAGENT' + InstanceName = 'STOPPEDAGENT' + ServiceName = 'STOPPEDAGENT' + Status = $mockDynamicStatus + } + } -Verifiable -ParameterFilter { $InstanceName -eq 'STOPPEDAGENT' } + } - { Invoke-Query @queryParams } | Should -Not -Throw + BeforeAll { + ## SQL instance with running SQL Agent Service + Mock -CommandName Get-Service -MockWith { + return @{ + Name = 'MSSQLSERVER' + DisplayName = 'Microsoft SQL Server (MSSQLSERVER)' + DependentServices = @( + @{ + Name = 'SQLSERVERAGENT' + DisplayName = 'SQL Server Agent (MSSQLSERVER)' + Status = 'Running' + DependentServices = @() + } + ) + } + } -Verifiable -ParameterFilter { $Name -eq 'MSSQLSERVER' } + + ## SQL instance with no installed SQL Agent Service + Mock -CommandName Get-Service -MockWith { + return @{ + Name = 'MSSQL$NOAGENT' + DisplayName = 'Microsoft SQL Server (NOAGENT)' + DependentServices = @() + } + } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$NOAGENT' } + + ## SQL instance with no installed SQL Agent Service + Mock -CommandName Get-Service -MockWith { + return @{ + Name = 'MSSQL$NOCLUSTERCHECK' + DisplayName = 'Microsoft SQL Server (NOCLUSTERCHECK)' + DependentServices = @() + } + } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$NOCLUSTERCHECK' } + + ## SQL instance with no installed SQL Agent Service + Mock -CommandName Get-Service -MockWith { + return @{ + Name = 'MSSQL$NOCONNECT' + DisplayName = 'Microsoft SQL Server (NOCONNECT)' + DependentServices = @() + } + } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$NOCONNECT' } + + ## SQL instance with stopped SQL Agent Service + Mock -CommandName Get-Service -MockWith { + return @{ + Name = 'MSSQL$STOPPEDAGENT' + DisplayName = 'Microsoft SQL Server (STOPPEDAGENT)' + DependentServices = @( + @{ + Name = 'SQLAGENT$STOPPEDAGENT' + DisplayName = 'SQL Server Agent (STOPPEDAGENT)' + Status = 'Stopped' + DependentServices = @() + } + ) + } + } -Verifiable -ParameterFilter { $Name -eq 'MSSQL$STOPPEDAGENT' } - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Mock -CommandName Restart-Service -Verifiable + Mock -CommandName Start-Service -Verifiable } - It 'Should throw the correct error, ExecuteNonQueryFailed, when executing the query fails' { - $queryParams.Query = 'BadQuery' + $mockDynamicStatus = 'Online' - { Invoke-Query @queryParams } | Should -Throw ($script:localizedData.ExecuteNonQueryFailed -f $queryParams.Database) + It 'Should restart SQL Service and running SQL Agent service' { + { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'MSSQLSERVER' } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Connect-SQL -ParameterFilter { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 1 } - } - Context 'Execute a query with results' { - It 'Should execute the query and return a result set' { - $queryParams.Query = 'SELECT name FROM sys.databases' - $mockExpectedQuery = $queryParams.Query.Clone() + It 'Should restart SQL Service, and not do cluster cluster check' { + Mock -CommandName Get-CimInstance - Invoke-Query @queryParams -WithResults | Should -Not -BeNullOrEmpty + { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'NOCLUSTERCHECK' -SkipClusterCheck } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Connect-SQL -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 + Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 0 } - It 'Should throw the correct error, ExecuteQueryWithResultsFailed, when executing the query fails' { - $queryParams.Query = 'BadQuery' + It 'Should restart SQL Service, and not do cluster cluster check nor check online status' { + Mock -CommandName Get-CimInstance - { Invoke-Query @queryParams -WithResults } | Should -Throw ($script:localizedData.ExecuteQueryWithResultsFailed -f $queryParams.Database) + { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'NOCONNECT' -SkipClusterCheck -SkipWaitForOnline } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Connect-SQL -Scope It -Exactly -Times 0 + Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 + Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 0 } - } - } - - Describe "Testing Update-AvailabilityGroupReplica" { - Context 'When the Availability Group Replica is altered' { - It 'Should silently alter the Availability Group Replica' { - $availabilityReplica = New-Object -TypeName Microsoft.SqlServer.Management.Smo.AvailabilityReplica - { Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityReplica } | Should -Not -Throw + It 'Should restart SQL Service and not try to restart missing SQL Agent service' { + { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'NOAGENT' } | Should -Not -Throw + Assert-MockCalled -CommandName Connect-SQL { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 } - It 'Should throw the correct error, AlterAvailabilityGroupReplicaFailed, when altering the Availability Group Replica fails' { - $availabilityReplica = New-Object -TypeName Microsoft.SqlServer.Management.Smo.AvailabilityReplica - $availabilityReplica.Name = 'AlterFailed' + It 'Should restart SQL Service and not try to restart stopped SQL Agent service' { + { Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'STOPPEDAGENT' } | Should -Not -Throw - { Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityReplica } | Should -Throw ($script:localizedData.AlterAvailabilityGroupReplicaFailed -f $availabilityReplica.Name) + Assert-MockCalled -CommandName Connect-SQL { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Restart-Service -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 0 } - } - } - - Describe "Testing Test-LoginEffectivePermissions" { - $mockAllServerPermissionsPresent = @( - 'Connect SQL', - 'Alter Any Availability Group', - 'View Server State' - ) + Context 'When it fails to connect to the instance within the timeout period' { + BeforeEach { + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'MSSQLSERVER' + InstanceName = '' + ServiceName = 'MSSQLSERVER' + Status = $mockDynamicStatus + } + } -Verifiable -ParameterFilter { $InstanceName -eq 'MSSQLSERVER' } + } - $mockServerPermissionsMissing = @( - 'Connect SQL', - 'View Server State' - ) + $mockDynamicStatus = 'Offline' - $mockAllLoginPermissionsPresent = @( - 'View Definition', - 'Impersonate' - ) + It 'Should throw the correct error message' { + $errorMessage = $localizedData.FailedToConnectToInstanceTimeout -f $env:ComputerName, 'MSSQLSERVER', 1 - $mockLoginPermissionsMissing = @( - 'View Definition' - ) + { + Restart-SqlService -SQLServer $env:ComputerName -SQLInstanceName 'MSSQLSERVER' -Timeout 1 + } | Should -Throw $errorMessage - $mockInvokeQueryPermissionsSet = @() # Will be set dynamically in the check + Assert-MockCalled -CommandName Connect-SQL -ParameterFilter { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 - $mockInvokeQueryPermissionsResult = { - return New-Object -TypeName PSObject -Property @{ - Tables = @{ - Rows = @{ - permission_name = $mockInvokeQueryPermissionsSet - } + Assert-MockCalled -CommandName Connect-SQL -ParameterFilter { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $true + } -Scope It -Exactly -Times 1 } } } - $testLoginEffectiveServerPermissionsParams = @{ - SQLServer = 'Server1' - SQLInstanceName = 'MSSQLSERVER' - Login = 'NT SERVICE\ClusSvc' - Permissions = @() - } + Context 'Restart-SqlService clustered instance' { + BeforeEach { + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'MSSQLSERVER' + InstanceName = '' + ServiceName = 'MSSQLSERVER' + IsClustered = $true + Status = $mockDynamicStatus + } + } -Verifiable -ParameterFilter { ($ServerName -eq 'CLU01') -and ($InstanceName -eq 'MSSQLSERVER') } - $testLoginEffectiveLoginPermissionsParams = @{ - SQLServer = 'Server1' - SQLInstanceName = 'MSSQLSERVER' - Login = 'NT SERVICE\ClusSvc' - Permissions = @() - SecurableClass = 'LOGIN' - SecurableName = 'Login1' - } + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'NAMEDINSTANCE' + InstanceName = 'NAMEDINSTANCE' + ServiceName = 'NAMEDINSTANCE' + IsClustered = $true + Status = $mockDynamicStatus + } + } -Verifiable -ParameterFilter { ($ServerName -eq 'CLU01') -and ($InstanceName -eq 'NAMEDINSTANCE') } - BeforeEach { - Mock -CommandName Invoke-Query -MockWith $mockInvokeQueryPermissionsResult -Verifiable - } + Mock -CommandName Connect-SQL -MockWith { + return @{ + Name = 'STOPPEDAGENT' + InstanceName = 'STOPPEDAGENT' + ServiceName = 'STOPPEDAGENT' + IsClustered = $true + Status = $mockDynamicStatus + } + } -Verifiable -ParameterFilter { ($ServerName -eq 'CLU01') -and ($InstanceName -eq 'STOPPEDAGENT') } + } - Context 'When all of the permissions are present' { - It 'Should return $true when the desired server permissions are present' { - $mockInvokeQueryPermissionsSet = $mockAllServerPermissionsPresent.Clone() - $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() + BeforeAll { + Mock -CommandName Get-CimInstance -MockWith { + @('MSSQLSERVER','NAMEDINSTANCE','STOPPEDAGENT') | ForEach-Object { + $mock = New-Object -TypeName Microsoft.Management.Infrastructure.CimInstance -ArgumentList 'MSCluster_Resource','root/MSCluster' - Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $true + $mock | Add-Member -MemberType NoteProperty -Name 'Name' -Value "SQL Server ($($_))" -TypeName 'String' + $mock | Add-Member -MemberType NoteProperty -Name 'Type' -Value 'SQL Server' -TypeName 'String' + $mock | Add-Member -MemberType NoteProperty -Name 'PrivateProperties' -Value @{ InstanceName = $_ } - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly - } + return $mock + } + } -Verifiable -ParameterFilter { ($ClassName -eq 'MSCluster_Resource') -and ($Filter -eq "Type = 'SQL Server'") } - It 'Should return $true when the desired login permissions are present' { - $mockInvokeQueryPermissionsSet = $mockAllLoginPermissionsPresent.Clone() - $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() + Mock -CommandName Get-CimAssociatedInstance -MockWith { + $mock = New-Object -TypeName Microsoft.Management.Infrastructure.CimInstance -ArgumentList 'MSCluster_Resource','root/MSCluster' - Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $true + $mock | Add-Member -MemberType NoteProperty -Name 'Name' -Value "SQL Server Agent ($($InputObject.PrivateProperties.InstanceName))" -TypeName 'String' + $mock | Add-Member -MemberType NoteProperty -Name 'Type' -Value 'SQL Server Agent' -TypeName 'String' + $mock | Add-Member -MemberType NoteProperty -Name 'State' -Value (@{ $true = 3; $false = 2 }[($InputObject.PrivateProperties.InstanceName -eq 'STOPPEDAGENT')]) -TypeName 'Int32' - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + return $mock + } -Verifiable -ParameterFilter { $ResultClassName -eq 'MSCluster_Resource' } + + Mock -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Verifiable + Mock -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Verifiable } - } - Context 'When a permission is missing' { - It 'Should return $false when the desired server permissions are not present' { - $mockInvokeQueryPermissionsSet = $mockServerPermissionsMissing.Clone() - $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() + $mockDynamicStatus = 'Online' - Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $false + It 'Should restart SQL Server and SQL Agent resources for a clustered default instance' { + { Restart-SqlService -SQLServer 'CLU01' } | Should -Not -Throw - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Connect-SQL { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-CimAssociatedInstance -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Scope It -Exactly -Times 2 } - It 'Should return $false when the specified login has no server permissions assigned' { - $mockInvokeQueryPermissionsSet = @() - $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() + It 'Should restart SQL Server and SQL Agent resources for a clustered named instance' { + { Restart-SqlService -SQLServer 'CLU01' -SQLInstanceName 'NAMEDINSTANCE' } | Should -Not -Throw - Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $false + Assert-MockCalled -CommandName Connect-SQL { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-CimAssociatedInstance -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Scope It -Exactly -Times 2 + } - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + It 'Should not try to restart a SQL Agent resource that is not online' { + { Restart-SqlService -SQLServer 'CLU01' -SQLInstanceName 'STOPPEDAGENT' } | Should -Not -Throw + + Assert-MockCalled -CommandName Connect-SQL { + $PSBoundParameters.ContainsKey('ErrorAction') -eq $false + } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-CimInstance -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Get-CimAssociatedInstance -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'TakeOffline' } -Scope It -Exactly -Times 1 + Assert-MockCalled -CommandName Invoke-CimMethod -ParameterFilter { $MethodName -eq 'BringOnline' } -Scope It -Exactly -Times 1 } + } + } + + Describe 'DscResource.Common\Connect-SQLAnalysis' -Tag 'ConnectSQLAnalysis' { + BeforeAll { + $mockInstanceName = 'TEST' - It 'Should return $false when the desired login permissions are not present' { - $mockInvokeQueryPermissionsSet = $mockLoginPermissionsMissing.Clone() - $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() + $mockNewObject_MicrosoftAnalysisServicesServer = { + return New-Object -TypeName Object | + Add-Member -MemberType ScriptMethod -Name Connect -Value { + param( + [Parameter(Mandatory = $true)] + [ValidateNotNullOrEmpty()] + [System.String] + $dataSource + ) - Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $false + if ($dataSource -ne $mockExpectedDataSource) + { + throw ("Datasource was expected to be '{0}', but was '{1}'." -f $mockExpectedDataSource,$dataSource) + } - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + if ($mockThrowInvalidOperation) + { + throw 'Unable to connect.' + } + } -PassThru -Force } - It 'Should return $false when the specified login has no login permissions assigned' { - $mockInvokeQueryPermissionsSet = @() - $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() - - Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $false + $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter = { + $TypeName -eq 'Microsoft.AnalysisServices.Server' + } - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + $mockThrowLocalizedMessage = { + throw $Message } + + $mockSetupCredentialUserName = 'TestUserName12345' + $mockSetupCredentialPassword = 'StrongOne7.' + $mockSetupCredentialSecurePassword = ConvertTo-SecureString -String $mockSetupCredentialPassword -AsPlainText -Force + $mockSetupCredential = New-Object -TypeName PSCredential -ArgumentList ($mockSetupCredentialUserName, $mockSetupCredentialSecurePassword) } - } - <# - This is the path to the latest version of SQLPS, to test that only the - newest SQLPS module is returned. - #> - $sqlPsLatestModulePath = 'C:\Program Files (x86)\Microsoft SQL Server\130\Tools\PowerShell\Modules\SQLPS\Sqlps.ps1' + BeforeEach { + Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable + Mock -CommandName New-Object ` + -MockWith $mockNewObject_MicrosoftAnalysisServicesServer ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter ` + -Verifiable + } - <# - For SQLPS module this should be the root of the module. - The .psd1 file is parsed from the module full path in the code. - #> - $sqlPsExpectedModulePath = Split-Path -Path $sqlPsLatestModulePath -Parent + Context 'When connecting to the default instance using Windows Authentication' { + It 'Should not throw when connecting' { + $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME" + { Connect-SQLAnalysis } | Should -Not -Throw - $mockImportModule = { - if ($Name -ne $mockExpectedModuleNameToImport) - { - throw ('Wrong module was loaded. Expected {0}, but was {1}.' -f $mockExpectedModuleNameToImport, $Name[0]) + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + } } - switch ($Name) - { - 'SqlServer' - { - $importModuleResult = @{ - ModuleType = 'Script' - Version = '21.0.17279' - Name = $Name - } - } + Context 'When connecting to the named instance using Windows Authentication' { + It 'Should not throw when connecting' { + $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME\$mockInstanceName" - $sqlPsExpectedModulePath - { - # Can not use $Name because that contain the path to the module manifest. - $importModuleResult = @( - @{ - ModuleType = 'Script' - Version = '0.0' - # Intentionally formatted to correctly mimic a real run. - Name = 'Sqlps' - Path = $sqlPsLatestModulePath - } - @{ - ModuleType = 'Manifest' - Version = '1.0' - # Intentionally formatted to correctly mimic a real run. - Name = 'sqlps' - Path = $sqlPsLatestModulePath - } - ) + { Connect-SQLAnalysis -SQLInstanceName $mockInstanceName } | Should -Not -Throw + + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter } } - return $importModuleResult - } - - $mockGetModuleSqlServer = { - # Return an array to test so that the latest version is only imported. - return @( - New-Object -TypeName PSObject -Property @{ - Name = 'SqlServer' - Version = [Version] '1.0' - } + Context 'When connecting to the named instance using Windows Authentication impersonation' { + It 'Should not throw when connecting' { + $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME\$mockInstanceName;User ID=$mockSetupCredentialUserName;Password=$mockSetupCredentialPassword" - New-Object -TypeName PSObject -Property @{ - Name = 'SqlServer' - Version = [Version] '2.0' - } - ) - } + { Connect-SQLAnalysis -SQLInstanceName $mockInstanceName -SetupCredential $mockSetupCredential } | Should -Not -Throw - $mockGetModuleSqlPs = { - # Return an array to test so that the latest version is only imported. - return @( - New-Object -TypeName PSObject -Property @{ - Name = 'SQLPS' - # This is a path to an older version of SQL PS than $sqlPsLatestModulePath. - Path = 'C:\Program Files (x86)\Microsoft SQL Server\120\Tools\PowerShell\Modules\SQLPS\Sqlps.ps1' + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter } + } - New-Object -TypeName PSObject -Property @{ - Name = 'SQLPS' - Path = $sqlPsLatestModulePath - } - ) - } + Context 'When connecting to the default instance using the correct service instance but does not return a correct Analysis Service object' { + It 'Should throw the correct error' { + $mockExpectedDataSource = '' - $mockGetModule_SqlServer_ParameterFilter = { - $FullyQualifiedName.Name -eq 'SqlServer' -and $ListAvailable -eq $true - } + Mock -CommandName New-Object ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter ` + -Verifiable - $mockGetModule_SQLPS_ParameterFilter = { - $FullyQualifiedName.Name -eq 'SQLPS' -and $ListAvailable -eq $true - } + $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToAnalysisServicesInstance -f $env:COMPUTERNAME) + { Connect-SQLAnalysis } | Should -Throw $mockCorrectErrorMessage - Describe 'Testing Import-SQLPSModule' -Tag 'ImportSQLPSModule' { - BeforeEach { - Mock -CommandName Push-Location -Verifiable - Mock -CommandName Pop-Location -Verifiable - Mock -CommandName Import-Module -MockWith $mockImportModule -Verifiable - Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + } } - Context 'When module SqlServer is already loaded into the session' { - BeforeAll { - Mock -CommandName Get-Module -MockWith { - return @{ - Name = 'SqlServer' - } - } - } + Context 'When connecting to the default instance using a Analysis Service instance that does not exist' { + It 'Should throw the correct error' { + $mockExpectedDataSource = "Data Source=$env:COMPUTERNAME" - It 'Should use the already loaded module and not call Import-Module' { - { Import-SQLPSModule } | Should -Not -Throw + # Force the mock of Connect() method to throw 'Unable to connect.' + $mockThrowInvalidOperation = $true - Assert-MockCalled -CommandName Import-Module -Exactly -Times 0 -Scope It + $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToAnalysisServicesInstance -f $env:COMPUTERNAME) + { Connect-SQLAnalysis } | Should -Throw $mockCorrectErrorMessage + + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter + + # Setting it back to the default so it does not disturb other tests. + $mockThrowInvalidOperation = $false } } - Context 'When module SQLPS is already loaded into the session' { - BeforeAll { - Mock -CommandName Get-Module -MockWith { - return @{ - Name = 'SQLPS' - } + # This test is to test the mock so that it throws correct when data source is not the expected data source + Context 'When connecting to the named instance using another data source then expected' { + It 'Should throw the correct error' { + $mockExpectedDataSource = "Force wrong data source" + + $testParameters = @{ + SQLServer = 'DummyHost' + SQLInstanceName = $mockInstanceName } - } - It 'Should use the already loaded module and not call Import-Module' { - { Import-SQLPSModule } | Should -Not -Throw + $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToAnalysisServicesInstance -f "$($testParameters.SQLServer)\$($testParameters.SQLInstanceName)") + { Connect-SQLAnalysis @testParameters } | Should -Throw $mockCorrectErrorMessage - Assert-MockCalled -CommandName Import-Module -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftAnalysisServicesServer_ParameterFilter } } - Context 'When module SqlServer exists, but not loaded into the session' { - BeforeAll { - Mock -CommandName Get-Module -ParameterFilter { - $PSBoundParameters.ContainsKey('Name') -eq $true - } + Assert-VerifiableMock + } - $mockExpectedModuleNameToImport = 'SqlServer' + Describe 'DscResource.Common\Invoke-Query' -Tag 'InvokeQuery' { + BeforeAll { + $mockExpectedQuery = '' + + $mockConnectSql = { + return @( + ( + New-Object -TypeName PSObject -Property @{ + Databases = @{ + 'master' = ( + New-Object -TypeName PSObject -Property @{ Name = 'master' } | + Add-Member -MemberType ScriptMethod -Name ExecuteNonQuery -Value { + param + ( + [Parameter()] + [System.String] + $sqlCommand + ) + + if ( $sqlCommand -ne $mockExpectedQuery ) + { + throw + } + } -PassThru | + Add-Member -MemberType ScriptMethod -Name ExecuteWithResults -Value { + param + ( + [Parameter()] + [System.String] + $sqlCommand + ) + + if ( $sqlCommand -ne $mockExpectedQuery ) + { + throw + } + + return New-Object -TypeName System.Data.DataSet + } -PassThru + ) + } + } + ) + ) } - It 'Should import the SqlServer module without throwing' { - Mock -CommandName Get-Module -MockWith $mockGetModuleSqlServer -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable + $mockThrowLocalizedMessage = { + throw $Message + } + } - { Import-SQLPSModule } | Should -Not -Throw + BeforeEach { + Mock -CommandName Connect-SQL -MockWith $mockConnectSql -ModuleName $script:dscResourceName -Verifiable + Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable + } - Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It - } + $queryParams = @{ + SQLServer = 'Server1' + SQLInstanceName = 'MSSQLSERVER' + Database = 'master' + Query = '' } - Context 'When only module SQLPS exists, but not loaded into the session, and using -Force' { - BeforeAll { - Mock -CommandName Remove-Module - Mock -CommandName Get-Module -ParameterFilter { - $PSBoundParameters.ContainsKey('Name') -eq $true - } + Context 'Execute a query with no results' { + It 'Should execute the query silently' { + $queryParams.Query = "EXEC sp_configure 'show advanced option', '1'" + $mockExpectedQuery = $queryParams.Query.Clone() - $mockExpectedModuleNameToImport = $sqlPsExpectedModulePath + { Invoke-Query @queryParams } | Should -Not -Throw + + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly } - It 'Should import the SqlServer module without throwing' { - Mock -CommandName Get-Module -MockWith $mockGetModuleSqlPs -ParameterFilter $mockGetModule_SQLPS_ParameterFilter -Verifiable - Mock -CommandName Get-Module -MockWith { - return $null - } -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable + It 'Should throw the correct error, ExecuteNonQueryFailed, when executing the query fails' { + $queryParams.Query = 'BadQuery' - { Import-SQLPSModule -Force } | Should -Not -Throw + { Invoke-Query @queryParams } | Should -Throw ($script:localizedData.ExecuteNonQueryFailed -f $queryParams.Database) - Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SQLPS_ParameterFilter -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Remove-Module -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly } } - Context 'When neither SqlServer or SQLPS exists' { - $mockExpectedModuleNameToImport = $sqlPsExpectedModulePath - - It 'Should throw the correct error message' { - Mock -CommandName Get-Module + Context 'Execute a query with results' { + It 'Should execute the query and return a result set' { + $queryParams.Query = 'SELECT name FROM sys.databases' + $mockExpectedQuery = $queryParams.Query.Clone() - { Import-SQLPSModule } | Should -Throw $script:localizedData.PowerShellSqlModuleNotFound + Invoke-Query @queryParams -WithResults | Should -Not -BeNullOrEmpty - Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SQLPS_ParameterFilter -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Push-Location -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Pop-Location -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Import-Module -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly } - } - Context 'When Import-Module fails to load the module' { - $mockExpectedModuleNameToImport = 'SqlServer' - - It 'Should throw the correct error message' { - $errorMessage = 'Mock Import-Module throwing a mocked error.' - Mock -CommandName Get-Module -MockWith $mockGetModuleSqlServer -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable - Mock -CommandName Import-Module -MockWith { - throw $errorMessage - } + It 'Should throw the correct error, ExecuteQueryWithResultsFailed, when executing the query fails' { + $queryParams.Query = 'BadQuery' - { Import-SQLPSModule } | Should -Throw ($script:localizedData.FailedToImportPowerShellSqlModule -f $mockExpectedModuleNameToImport) + { Invoke-Query @queryParams -WithResults } | Should -Throw ($script:localizedData.ExecuteQueryWithResultsFailed -f $queryParams.Database) - Assert-MockCalled -CommandName Get-Module -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly } } + } - # This is to test the tests (so the mock throws correctly) - Context 'When mock Import-Module is called with wrong module name' { - $mockExpectedModuleNameToImport = 'UnknownModule' + Describe 'DscResource.Common\Update-AvailabilityGroupReplica' -Tag 'UpdateAvailabilityGroupReplica' { + Context 'When the Availability Group Replica is altered' { + It 'Should silently alter the Availability Group Replica' { + $availabilityReplica = New-Object -TypeName Microsoft.SqlServer.Management.Smo.AvailabilityReplica - It 'Should throw the correct error message' { - Mock -CommandName Get-Module -MockWith $mockGetModuleSqlServer -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable + { Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityReplica } | Should -Not -Throw - { Import-SQLPSModule } | Should -Throw ($script:localizedData.FailedToImportPowerShellSqlModule -f 'SqlServer') + } - Assert-MockCalled -CommandName Get-Module -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It + It 'Should throw the correct error, AlterAvailabilityGroupReplicaFailed, when altering the Availability Group Replica fails' { + $availabilityReplica = New-Object -TypeName Microsoft.SqlServer.Management.Smo.AvailabilityReplica + $availabilityReplica.Name = 'AlterFailed' + + { Update-AvailabilityGroupReplica -AvailabilityGroupReplica $availabilityReplica } | Should -Throw ($script:localizedData.AlterAvailabilityGroupReplicaFailed -f $availabilityReplica.Name) } } - - Assert-VerifiableMock } - $mockGetItemProperty_MicrosoftSQLServer_InstanceNames_SQL = { - return @( - ( - New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name $mockInstanceName -Value $mockInstance_InstanceId -PassThru -Force - ) + Describe 'DscResource.Common\Test-LoginEffectivePermissions' -Tag 'TestLoginEffectivePermissions' { + + $mockAllServerPermissionsPresent = @( + 'Connect SQL', + 'Alter Any Availability Group', + 'View Server State' ) - } - $mockGetItemProperty_MicrosoftSQLServer_FullInstanceId_Setup = { - return @( - ( - New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'Version' -Value "$($mockSqlMajorVersion).0.4001.0" -PassThru -Force - ) + $mockServerPermissionsMissing = @( + 'Connect SQL', + 'View Server State' ) - } - $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL = { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' - } + $mockAllLoginPermissionsPresent = @( + 'View Definition', + 'Impersonate' + ) - $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup = { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockInstance_InstanceId\Setup" - } + $mockLoginPermissionsMissing = @( + 'View Definition' + ) - Describe 'Testing Get-SqlInstanceMajorVersion' -Tag GetSqlInstanceMajorVersion { - BeforeEach { - Mock -CommandName Get-ItemProperty ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL ` - -MockWith $mockGetItemProperty_MicrosoftSQLServer_InstanceNames_SQL ` - -Verifiable + $mockInvokeQueryPermissionsSet = @() # Will be set dynamically in the check - Mock -CommandName Get-ItemProperty ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup ` - -MockWith $mockGetItemProperty_MicrosoftSQLServer_FullInstanceId_Setup ` - -Verifiable + $mockInvokeQueryPermissionsResult = { + return New-Object -TypeName PSObject -Property @{ + Tables = @{ + Rows = @{ + permission_name = $mockInvokeQueryPermissionsSet + } + } + } } - $mockInstance_InstanceId = "MSSQL$($mockSqlMajorVersion).$($mockInstanceName)" - - Context 'When calling Get-SqlInstanceMajorVersion' { - It 'Should return the correct major SQL version number' { - $result = Get-SqlInstanceMajorVersion -SQLInstanceName $mockInstanceName - $result | Should -Be $mockSqlMajorVersion - - Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL + $testLoginEffectiveServerPermissionsParams = @{ + SQLServer = 'Server1' + SQLInstanceName = 'MSSQLSERVER' + Login = 'NT SERVICE\ClusSvc' + Permissions = @() + } - Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup - } + $testLoginEffectiveLoginPermissionsParams = @{ + SQLServer = 'Server1' + SQLInstanceName = 'MSSQLSERVER' + Login = 'NT SERVICE\ClusSvc' + Permissions = @() + SecurableClass = 'LOGIN' + SecurableName = 'Login1' } - Context 'When calling Get-SqlInstanceMajorVersion and nothing is returned' { - It 'Should throw the correct error' { - Mock -CommandName Get-ItemProperty ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup ` - -MockWith { - return New-Object -TypeName Object - } -Verifiable + BeforeEach { + Mock -CommandName Invoke-Query -MockWith $mockInvokeQueryPermissionsResult -Verifiable + } - $mockCorrectErrorMessage = ($script:localizedData.SqlServerVersionIsInvalid -f $mockInstanceName) - { Get-SqlInstanceMajorVersion -SQLInstanceName $mockInstanceName } | Should -Throw $mockCorrectErrorMessage + Context 'When all of the permissions are present' { + It 'Should return $true when the desired server permissions are present' { + $mockInvokeQueryPermissionsSet = $mockAllServerPermissionsPresent.Clone() + $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() - Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL + Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $true - Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } - } - Assert-VerifiableMock - } + It 'Should return $true when the desired login permissions are present' { + $mockInvokeQueryPermissionsSet = $mockAllLoginPermissionsPresent.Clone() + $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() - Describe 'Testing Get-PrimaryReplicaServerObject' { - BeforeEach { - $mockServerObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.Server - $mockServerObject.DomainInstanceName = 'Server1' + Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $true - $mockAvailabilityGroup = New-Object -TypeName Microsoft.SqlServer.Management.Smo.AvailabilityGroup - $mockAvailabilityGroup.PrimaryReplicaServerName = 'Server1' + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } } - $mockConnectSql = { - Param - ( - [Parameter()] - [System.String] - $ServerName, + Context 'When a permission is missing' { + It 'Should return $false when the desired server permissions are not present' { + $mockInvokeQueryPermissionsSet = $mockServerPermissionsMissing.Clone() + $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() - [Parameter()] - [System.String] - $InstanceName - ) + Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $false - $mock = @( - ( - New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'DomainInstanceName' -Value $ServerName -PassThru - ) - ) + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } - # Type the mock as a server object - $mock.PSObject.TypeNames.Insert(0,'Microsoft.SqlServer.Management.Smo.Server') + It 'Should return $false when the specified login has no server permissions assigned' { + $mockInvokeQueryPermissionsSet = @() + $testLoginEffectiveServerPermissionsParams.Permissions = $mockAllServerPermissionsPresent.Clone() - return $mock - } + Test-LoginEffectivePermissions @testLoginEffectiveServerPermissionsParams | Should -Be $false - Mock -CommandName Connect-SQL -MockWith $mockConnectSql -Verifiable + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } - Context 'When the supplied server object is the primary replica' { - It 'Should return the same server object that was supplied' { - $result = Get-PrimaryReplicaServerObject -ServerObject $mockServerObject -AvailabilityGroup $mockAvailabilityGroup + It 'Should return $false when the desired login permissions are not present' { + $mockInvokeQueryPermissionsSet = $mockLoginPermissionsMissing.Clone() + $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() - $result.DomainInstanceName | Should -Be $mockServerObject.DomainInstanceName - $result.DomainInstanceName | Should -Be $mockAvailabilityGroup.PrimaryReplicaServerName + Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $false - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 0 -Exactly + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } - It 'Should return the same server object that was supplied when the PrimaryReplicaServerNameProperty is empty' { - $mockAvailabilityGroup.PrimaryReplicaServerName = '' - - $result = Get-PrimaryReplicaServerObject -ServerObject $mockServerObject -AvailabilityGroup $mockAvailabilityGroup + It 'Should return $false when the specified login has no login permissions assigned' { + $mockInvokeQueryPermissionsSet = @() + $testLoginEffectiveLoginPermissionsParams.Permissions = $mockAllLoginPermissionsPresent.Clone() - $result.DomainInstanceName | Should -Be $mockServerObject.DomainInstanceName - $result.DomainInstanceName | Should -Not -Be $mockAvailabilityGroup.PrimaryReplicaServerName + Test-LoginEffectivePermissions @testLoginEffectiveLoginPermissionsParams | Should -Be $false - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 0 -Exactly + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } } + } - Context 'When the supplied server object is not the primary replica' { - It 'Should the server object of the primary replica' { - $mockAvailabilityGroup.PrimaryReplicaServerName = 'Server2' + Describe 'DscResource.Common\Import-SQLPSModule' -Tag 'ImportSQLPSModule' { + BeforeAll { + <# + This is the path to the latest version of SQLPS, to test that only the + newest SQLPS module is returned. + #> + $sqlPsLatestModulePath = 'C:\Program Files (x86)\Microsoft SQL Server\130\Tools\PowerShell\Modules\SQLPS\Sqlps.ps1' - $result = Get-PrimaryReplicaServerObject -ServerObject $mockServerObject -AvailabilityGroup $mockAvailabilityGroup + <# + For SQLPS module this should be the root of the module. + The .psd1 file is parsed from the module full path in the code. + #> + $sqlPsExpectedModulePath = Split-Path -Path $sqlPsLatestModulePath -Parent - $result.DomainInstanceName | Should -Not -Be $mockServerObject.DomainInstanceName - $result.DomainInstanceName | Should -Be $mockAvailabilityGroup.PrimaryReplicaServerName - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly - } - } - } + $mockImportModule = { + if ($Name -ne $mockExpectedModuleNameToImport) + { + throw ('Wrong module was loaded. Expected {0}, but was {1}.' -f $mockExpectedModuleNameToImport, $Name[0]) + } - Describe 'Testing Test-AvailabilityReplicaSeedingModeAutomatic' { + switch ($Name) + { + 'SqlServer' + { + $importModuleResult = @{ + ModuleType = 'Script' + Version = '21.0.17279' + Name = $Name + } + } - BeforeEach { - $mockSqlVersion = 13 - $mockConnectSql = { - Param - ( - [Parameter()] - [System.String] - $ServerName, + $sqlPsExpectedModulePath + { + # Can not use $Name because that contain the path to the module manifest. + $importModuleResult = @( + @{ + ModuleType = 'Script' + Version = '0.0' + # Intentionally formatted to correctly mimic a real run. + Name = 'Sqlps' + Path = $sqlPsLatestModulePath + } + @{ + ModuleType = 'Manifest' + Version = '1.0' + # Intentionally formatted to correctly mimic a real run. + Name = 'sqlps' + Path = $sqlPsLatestModulePath + } + ) + } + } - [Parameter()] - [System.String] - $InstanceName - ) + return $importModuleResult + } - $mock = @( - ( - New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'Version' -Value $mockSqlVersion -PassThru - ) + $mockGetModuleSqlServer = { + # Return an array to test so that the latest version is only imported. + return @( + New-Object -TypeName PSObject -Property @{ + Name = 'SqlServer' + Version = [Version] '1.0' + } + + New-Object -TypeName PSObject -Property @{ + Name = 'SqlServer' + Version = [Version] '2.0' + } ) + } - # Type the mock as a server object - $mock.PSObject.TypeNames.Insert(0,'Microsoft.SqlServer.Management.Smo.Server') + $mockGetModuleSqlPs = { + # Return an array to test so that the latest version is only imported. + return @( + New-Object -TypeName PSObject -Property @{ + Name = 'SQLPS' + # This is a path to an older version of SQL PS than $sqlPsLatestModulePath. + Path = 'C:\Program Files (x86)\Microsoft SQL Server\120\Tools\PowerShell\Modules\SQLPS\Sqlps.ps1' + } - return $mock + New-Object -TypeName PSObject -Property @{ + Name = 'SQLPS' + Path = $sqlPsLatestModulePath + } + ) } - $mockSeedingMode = 'Manual' - $mockInvokeQuery = { - return @{ - Tables = @{ - Rows = @{ - seeding_mode_desc = $mockSeedingMode - } - } - } + $mockGetModule_SqlServer_ParameterFilter = { + $FullyQualifiedName.Name -eq 'SqlServer' -and $ListAvailable -eq $true } - Mock -CommandName Connect-SQL -MockWith $mockConnectSql -Verifiable - Mock -CommandName Invoke-Query -MockWith $mockInvokeQuery -Verifiable + $mockGetModule_SQLPS_ParameterFilter = { + $FullyQualifiedName.Name -eq 'SQLPS' -and $ListAvailable -eq $true + } + + $mockThrowLocalizedMessage = { + throw $Message + } } - $testAvailabilityReplicaSeedingModeAutomaticParams = @{ - SQLServer = 'Server1' - SQLInstanceName = 'MSSQLSERVER' - AvailabilityGroupName = 'Group1' - AvailabilityReplicaName = 'Replica2' + BeforeEach { + Mock -CommandName Push-Location -Verifiable + Mock -CommandName Pop-Location -Verifiable + Mock -CommandName Import-Module -MockWith $mockImportModule -Verifiable + Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable } - Context 'When the replica seeding mode is manual' { - # Test SQL 2012 and 2014. Not testing earlier versions because Availability Groups were introduced in SQL 2012. - foreach ( $instanceVersion in @(11,12) ) - { - It ( 'Should return $false when the instance version is {0}' -f $instanceVersion ) { - $mockSqlVersion = $instanceVersion + Context 'When module SqlServer is already loaded into the session' { + BeforeAll { + Mock -CommandName Get-Module -MockWith { + return @{ + Name = 'SqlServer' + } + } + } + + It 'Should use the already loaded module and not call Import-Module' { + { Import-SQLPSModule } | Should -Not -Throw - Test-AvailabilityReplicaSeedingModeAutomatic @testAvailabilityReplicaSeedingModeAutomaticParams | Should -Be $false + Assert-MockCalled -CommandName Import-Module -Exactly -Times 0 -Scope It + } + } - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly + Context 'When module SQLPS is already loaded into the session' { + BeforeAll { + Mock -CommandName Get-Module -MockWith { + return @{ + Name = 'SQLPS' + } } } - # Test SQL 2016 and later - foreach ( $instanceVersion in @(13,14) ) - { - It ( 'Should return $false when the instance version is {0} and the replica seeding mode is manual' -f $instanceVersion ) { - $mockSqlVersion = $instanceVersion + It 'Should use the already loaded module and not call Import-Module' { + { Import-SQLPSModule } | Should -Not -Throw - Test-AvailabilityReplicaSeedingModeAutomatic @testAvailabilityReplicaSeedingModeAutomaticParams | Should -Be $false + Assert-MockCalled -CommandName Import-Module -Exactly -Times 0 -Scope It + } + } - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + Context 'When module SqlServer exists, but not loaded into the session' { + BeforeAll { + Mock -CommandName Get-Module -ParameterFilter { + $PSBoundParameters.ContainsKey('Name') -eq $true } + + $mockExpectedModuleNameToImport = 'SqlServer' } - } - Context 'When the replica seeding mode is automatic' { - # Test SQL 2016 and later - foreach ( $instanceVersion in @(13,14) ) - { - It ( 'Should return $true when the instance version is {0} and the replica seeding mode is automatic' -f $instanceVersion ) { - $mockSqlVersion = $instanceVersion - $mockSeedingMode = 'Automatic' + It 'Should import the SqlServer module without throwing' { + Mock -CommandName Get-Module -MockWith $mockGetModuleSqlServer -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable - Test-AvailabilityReplicaSeedingModeAutomatic @testAvailabilityReplicaSeedingModeAutomaticParams | Should -Be $true + { Import-SQLPSModule } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly - } + Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It } } - } - $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter = { - $Permissions -eq @('IMPERSONATE ANY LOGIN') - } + Context 'When only module SQLPS exists, but not loaded into the session, and using -Force' { + BeforeAll { + Mock -CommandName Remove-Module + Mock -CommandName Get-Module -ParameterFilter { + $PSBoundParameters.ContainsKey('Name') -eq $true + } - $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter = { - $Permissions -eq @('CONTROL SERVER') - } + $mockExpectedModuleNameToImport = $sqlPsExpectedModulePath + } - $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter = { - $Permissions -eq @('IMPERSONATE') - } + It 'Should import the SqlServer module without throwing' { + Mock -CommandName Get-Module -MockWith $mockGetModuleSqlPs -ParameterFilter $mockGetModule_SQLPS_ParameterFilter -Verifiable + Mock -CommandName Get-Module -MockWith { + return $null + } -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable - $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter = { - $Permissions -eq @('CONTROL') - } + { Import-SQLPSModule -Force } | Should -Not -Throw - Describe 'Testing Test-ImpersonatePermissions' { - $mockConnectionContextObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.ServerConnection - $mockConnectionContextObject.TrueLogin = 'Login1' + Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SQLPS_ParameterFilter -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Remove-Module -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It + } + } - $mockServerObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.Server - $mockServerObject.ComputerNamePhysicalNetBIOS = 'Server1' - $mockServerObject.ServiceName = 'MSSQLSERVER' - $mockServerObject.ConnectionContext = $mockConnectionContextObject + Context 'When neither SqlServer or SQLPS exists' { + $mockExpectedModuleNameToImport = $sqlPsExpectedModulePath - BeforeEach { - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -MockWith { $false } -Verifiable - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -MockWith { $false } -Verifiable - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -MockWith { $false } -Verifiable - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -MockWith { $false } -Verifiable - } + It 'Should throw the correct error message' { + Mock -CommandName Get-Module - Context 'When impersonate permissions are present for the login' { - It 'Should return true when the impersonate any login permissions are present for the login' { - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -MockWith { $true } -Verifiable - Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $true + { Import-SQLPSModule } | Should -Throw $script:localizedData.PowerShellSqlModuleNotFound - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Get-Module -ParameterFilter $mockGetModule_SQLPS_ParameterFilter -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Push-Location -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Pop-Location -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Import-Module -Exactly -Times 0 -Scope It } + } - It 'Should return true when the control server permissions are present for the login' { - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -MockWith { $true } -Verifiable - Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $true + Context 'When Import-Module fails to load the module' { + $mockExpectedModuleNameToImport = 'SqlServer' - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly - } + It 'Should throw the correct error message' { + $errorMessage = 'Mock Import-Module throwing a mocked error.' + Mock -CommandName Get-Module -MockWith $mockGetModuleSqlServer -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable + Mock -CommandName Import-Module -MockWith { + throw $errorMessage + } - It 'Should return true when the impersonate login permissions are present for the login' { - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -MockWith { $true } -Verifiable - Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $true + { Import-SQLPSModule } | Should -Throw ($script:localizedData.FailedToImportPowerShellSqlModule -f $mockExpectedModuleNameToImport) - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Get-Module -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It } + } - It 'Should return true when the control login permissions are present for the login' { - Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -MockWith { $true } -Verifiable - Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $true + # This is to test the tests (so the mock throws correctly) + Context 'When mock Import-Module is called with wrong module name' { + $mockExpectedModuleNameToImport = 'UnknownModule' - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 1 -Exactly + It 'Should throw the correct error message' { + Mock -CommandName Get-Module -MockWith $mockGetModuleSqlServer -ParameterFilter $mockGetModule_SqlServer_ParameterFilter -Verifiable + + { Import-SQLPSModule } | Should -Throw ($script:localizedData.FailedToImportPowerShellSqlModule -f 'SqlServer') + + Assert-MockCalled -CommandName Get-Module -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Push-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Pop-Location -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-Module -Exactly -Times 1 -Scope It } } - Context 'When impersonate permissions are missing for the login' { - It 'Should return false when the server permissions are missing for the login' { - Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $false + Assert-VerifiableMock + } - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 0 -Exactly - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 0 -Exactly + Describe 'DscResource.Common\Get-SqlInstanceMajorVersion' -Tag 'GetSqlInstanceMajorVersion' { + BeforeAll { + $mockSqlMajorVersion = 13 + $mockInstanceName = 'TEST' + + $mockGetItemProperty_MicrosoftSQLServer_InstanceNames_SQL = { + return @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name $mockInstanceName -Value $mockInstance_InstanceId -PassThru -Force + ) + ) } - It 'Should return false when the login permissions are missing for the login' { - Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $false + $mockGetItemProperty_MicrosoftSQLServer_FullInstanceId_Setup = { + return @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'Version' -Value "$($mockSqlMajorVersion).0.4001.0" -PassThru -Force + ) + ) + } - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 1 -Exactly - Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 1 -Exactly + $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL = { + $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' + } + + $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup = { + $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockInstance_InstanceId\Setup" } } - } - Describe 'Testing Connect-SQL' -Tag 'ConnectSql' { BeforeEach { - Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable - Mock -CommandName Import-SQLPSModule - Mock -CommandName New-Object ` - -MockWith $mockNewObject_MicrosoftDatabaseEngine ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter ` + Mock -CommandName Get-ItemProperty ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL ` + -MockWith $mockGetItemProperty_MicrosoftSQLServer_InstanceNames_SQL ` + -Verifiable + + Mock -CommandName Get-ItemProperty ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup ` + -MockWith $mockGetItemProperty_MicrosoftSQLServer_FullInstanceId_Setup ` -Verifiable } - Context 'When connecting to the default instance using Windows Authentication' { - It 'Should return the correct service instance' { - $mockExpectedDatabaseEngineServer = 'TestServer' - $mockExpectedDatabaseEngineInstance = 'MSSQLSERVER' + $mockInstance_InstanceId = "MSSQL$($mockSqlMajorVersion).$($mockInstanceName)" - $databaseEngineServerObject = Connect-SQL -ServerName $mockExpectedDatabaseEngineServer - $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly $mockExpectedDatabaseEngineServer + Context 'When calling Get-SqlInstanceMajorVersion' { + It 'Should return the correct major SQL version number' { + $result = Get-SqlInstanceMajorVersion -SQLInstanceName $mockInstanceName + $result | Should -Be $mockSqlMajorVersion - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter + Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL + + Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup } } - Context 'When connecting to the default instance using SQL Server Authentication' { - It 'Should return the correct service instance' { - $mockExpectedDatabaseEngineServer = 'TestServer' - $mockExpectedDatabaseEngineInstance = 'MSSQLSERVER' - $mockExpectedDatabaseEngineLoginSecure = $false + Context 'When calling Get-SqlInstanceMajorVersion and nothing is returned' { + It 'Should throw the correct error' { + Mock -CommandName Get-ItemProperty ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup ` + -MockWith { + return New-Object -TypeName Object + } -Verifiable - $databaseEngineServerObject = Connect-SQL -ServerName $mockExpectedDatabaseEngineServer -SetupCredential $mockSetupCredential -LoginType 'SqlLogin' - $databaseEngineServerObject.ConnectionContext.LoginSecure | Should -Be $false - $databaseEngineServerObject.ConnectionContext.Login | Should -Be $mockSetupCredentialUserName - $databaseEngineServerObject.ConnectionContext.SecurePassword | Should -Be $mockSetupCredentialSecurePassword - $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly $mockExpectedDatabaseEngineServer + $mockCorrectErrorMessage = ($script:localizedData.SqlServerVersionIsInvalid -f $mockInstanceName) + { Get-SqlInstanceMajorVersion -SQLInstanceName $mockInstanceName } | Should -Throw $mockCorrectErrorMessage - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter + Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_InstanceNames_SQL + + Assert-MockCalled -CommandName Get-ItemProperty -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockGetItemProperty_ParameterFilter_MicrosoftSQLServer_FullInstanceId_Setup } } - Context 'When connecting to the named instance using Windows Authentication' { - It 'Should return the correct service instance' { - $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME - $mockExpectedDatabaseEngineInstance = $mockInstanceName + Assert-VerifiableMock + } - $databaseEngineServerObject = Connect-SQL -InstanceName $mockExpectedDatabaseEngineInstance - $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + Describe 'DscResource.Common\Get-PrimaryReplicaServerObject' -Tag 'GetPrimaryReplicaServerObject' { + BeforeEach { + $mockServerObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.Server + $mockServerObject.DomainInstanceName = 'Server1' - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter - } + $mockAvailabilityGroup = New-Object -TypeName Microsoft.SqlServer.Management.Smo.AvailabilityGroup + $mockAvailabilityGroup.PrimaryReplicaServerName = 'Server1' } - Context 'When connecting to the named instance using SQL Server Authentication' { - It 'Should return the correct service instance' { - $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME - $mockExpectedDatabaseEngineInstance = $mockInstanceName - $mockExpectedDatabaseEngineLoginSecure = $false + $mockConnectSql = { + Param + ( + [Parameter()] + [System.String] + $ServerName, + + [Parameter()] + [System.String] + $InstanceName + ) - $databaseEngineServerObject = Connect-SQL -InstanceName $mockExpectedDatabaseEngineInstance -SetupCredential $mockSetupCredential -LoginType 'SqlLogin' - $databaseEngineServerObject.ConnectionContext.LoginSecure | Should -Be $false - $databaseEngineServerObject.ConnectionContext.Login | Should -Be $mockSetupCredentialUserName - $databaseEngineServerObject.ConnectionContext.SecurePassword | Should -Be $mockSetupCredentialSecurePassword - $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + $mock = @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'DomainInstanceName' -Value $ServerName -PassThru + ) + ) - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter - } + # Type the mock as a server object + $mock.PSObject.TypeNames.Insert(0,'Microsoft.SqlServer.Management.Smo.Server') + + return $mock } - Context 'When connecting to the named instance using Windows Authentication and different server name' { - It 'Should return the correct service instance' { - $mockExpectedDatabaseEngineServer = 'SERVER' - $mockExpectedDatabaseEngineInstance = $mockInstanceName + Mock -CommandName Connect-SQL -MockWith $mockConnectSql -Verifiable - $databaseEngineServerObject = Connect-SQL -ServerName $mockExpectedDatabaseEngineServer -InstanceName $mockExpectedDatabaseEngineInstance - $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + Context 'When the supplied server object is the primary replica' { + It 'Should return the same server object that was supplied' { + $result = Get-PrimaryReplicaServerObject -ServerObject $mockServerObject -AvailabilityGroup $mockAvailabilityGroup - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter + $result.DomainInstanceName | Should -Be $mockServerObject.DomainInstanceName + $result.DomainInstanceName | Should -Be $mockAvailabilityGroup.PrimaryReplicaServerName + + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 0 -Exactly } - } - Context 'When connecting to the named instance using Windows Authentication impersonation' { - It 'Should return the correct service instance' { - $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME - $mockExpectedDatabaseEngineInstance = $mockInstanceName + It 'Should return the same server object that was supplied when the PrimaryReplicaServerNameProperty is empty' { + $mockAvailabilityGroup.PrimaryReplicaServerName = '' - $testParameters = @{ - ServerName = $mockExpectedDatabaseEngineServer - InstanceName = $mockExpectedDatabaseEngineInstance - SetupCredential = $mockSetupCredential - } + $result = Get-PrimaryReplicaServerObject -ServerObject $mockServerObject -AvailabilityGroup $mockAvailabilityGroup - $databaseEngineServerObject = Connect-SQL @testParameters - $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" - $databaseEngineServerObject.ConnectionContext.ConnectAsUser | Should -Be $true - $databaseEngineServerObject.ConnectionContext.ConnectAsUserPassword | Should -BeExactly $mockSetupCredential.GetNetworkCredential().Password - $databaseEngineServerObject.ConnectionContext.ConnectAsUserName | Should -BeExactly $mockSetupCredential.GetNetworkCredential().UserName - $databaseEngineServerObject.ConnectionContext.ConnectAsUser | Should -Be $true + $result.DomainInstanceName | Should -Be $mockServerObject.DomainInstanceName + $result.DomainInstanceName | Should -Not -Be $mockAvailabilityGroup.PrimaryReplicaServerName - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 0 -Exactly } } - Context 'When connecting to the default instance using the correct service instance but does not return a correct Database Engine object' { - It 'Should throw the correct error' { - $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME - $mockExpectedDatabaseEngineInstance = $mockInstanceName + Context 'When the supplied server object is not the primary replica' { + It 'Should the server object of the primary replica' { + $mockAvailabilityGroup.PrimaryReplicaServerName = 'Server2' - Mock -CommandName New-Object ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter ` - -Verifiable + $result = Get-PrimaryReplicaServerObject -ServerObject $mockServerObject -AvailabilityGroup $mockAvailabilityGroup - $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToDatabaseEngineInstance -f $mockExpectedDatabaseEngineServer) - { Connect-SQL } | Should -Throw $mockCorrectErrorMessage + $result.DomainInstanceName | Should -Not -Be $mockServerObject.DomainInstanceName + $result.DomainInstanceName | Should -Be $mockAvailabilityGroup.PrimaryReplicaServerName - Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` - -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly } } - - Assert-VerifiableMock } - Describe 'Testing Test-SQLDscParameterState' -Tag TestSQLDscParameterState { - Context -Name 'When passing values' -Fixture { - It 'Should return true for two identical tables' { - $mockDesiredValues = @{ Example = 'test' } + Describe 'DscResource.Common\Test-AvailabilityReplicaSeedingModeAutomatic' -Tag 'TestAvailabilityReplicaSeedingModeAutomatic' { - $testParameters = @{ - CurrentValues = $mockDesiredValues - DesiredValues = $mockDesiredValues - } + BeforeEach { + $mockSqlVersion = 13 + $mockConnectSql = { + Param + ( + [Parameter()] + [System.String] + $ServerName, - Test-SQLDscParameterState @testParameters | Should -Be $true - } + [Parameter()] + [System.String] + $InstanceName + ) - It 'Should return false when a value is different for [System.String]' { - $mockCurrentValues = @{ Example = [System.String]'something' } - $mockDesiredValues = @{ Example = [System.String]'test' } + $mock = @( + ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name 'Version' -Value $mockSqlVersion -PassThru + ) + ) - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + # Type the mock as a server object + $mock.PSObject.TypeNames.Insert(0,'Microsoft.SqlServer.Management.Smo.Server') - Test-SQLDscParameterState @testParameters | Should -Be $false + return $mock } - It 'Should return false when a value is different for [System.Int32]' { - $mockCurrentValues = @{ Example = [System.Int32]1 } - $mockDesiredValues = @{ Example = [System.Int32]2 } - - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues + $mockSeedingMode = 'Manual' + $mockInvokeQuery = { + return @{ + Tables = @{ + Rows = @{ + seeding_mode_desc = $mockSeedingMode + } + } } - - Test-SQLDscParameterState @testParameters | Should -Be $false } - It 'Should return false when a value is different for [Int16]' { - $mockCurrentValues = @{ Example = [System.Int16]1 } - $mockDesiredValues = @{ Example = [System.Int16]2 } + Mock -CommandName Connect-SQL -MockWith $mockConnectSql -Verifiable + Mock -CommandName Invoke-Query -MockWith $mockInvokeQuery -Verifiable + } - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + $testAvailabilityReplicaSeedingModeAutomaticParams = @{ + SQLServer = 'Server1' + SQLInstanceName = 'MSSQLSERVER' + AvailabilityGroupName = 'Group1' + AvailabilityReplicaName = 'Replica2' + } - Test-SQLDscParameterState @testParameters | Should -Be $false - } + Context 'When the replica seeding mode is manual' { + # Test SQL 2012 and 2014. Not testing earlier versions because Availability Groups were introduced in SQL 2012. + foreach ( $instanceVersion in @(11,12) ) + { + It ( 'Should return $false when the instance version is {0}' -f $instanceVersion ) { + $mockSqlVersion = $instanceVersion - It 'Should return false when a value is different for [UInt16]' { - $mockCurrentValues = @{ Example = [System.UInt16]1 } - $mockDesiredValues = @{ Example = [System.UInt16]2 } + Test-AvailabilityReplicaSeedingModeAutomatic @testAvailabilityReplicaSeedingModeAutomaticParams | Should -Be $false - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 0 -Exactly } - - Test-SQLDscParameterState @testParameters | Should -Be $false } - It 'Should return false when a value is missing' { - $mockCurrentValues = @{ } - $mockDesiredValues = @{ Example = 'test' } + # Test SQL 2016 and later + foreach ( $instanceVersion in @(13,14) ) + { + It ( 'Should return $false when the instance version is {0} and the replica seeding mode is manual' -f $instanceVersion ) { + $mockSqlVersion = $instanceVersion - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + Test-AvailabilityReplicaSeedingModeAutomatic @testAvailabilityReplicaSeedingModeAutomaticParams | Should -Be $false - Test-SQLDscParameterState @testParameters | Should -Be $false + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly + } } + } - It 'Should return true when only a specified value matches, but other non-listed values do not' { - $mockCurrentValues = @{ Example = 'test'; SecondExample = 'true' } - $mockDesiredValues = @{ Example = 'test'; SecondExample = 'false' } + Context 'When the replica seeding mode is automatic' { + # Test SQL 2016 and later + foreach ( $instanceVersion in @(13,14) ) + { + It ( 'Should return $true when the instance version is {0} and the replica seeding mode is automatic' -f $instanceVersion ) { + $mockSqlVersion = $instanceVersion + $mockSeedingMode = 'Automatic' - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - ValuesToCheck = @('Example') + Test-AvailabilityReplicaSeedingModeAutomatic @testAvailabilityReplicaSeedingModeAutomaticParams | Should -Be $true + + Assert-MockCalled -CommandName Connect-SQL -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Invoke-Query -Scope It -Times 1 -Exactly } + } + } + } - Test-SQLDscParameterState @testParameters | Should -Be $true + Describe 'DscResource.Common\Test-ImpersonatePermissions' -Tag 'TestImpersonatePermissions' { + BeforeAll { + $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter = { + $Permissions -eq @('IMPERSONATE ANY LOGIN') } - It 'Should return false when only specified values do not match, but other non-listed values do ' { - $mockCurrentValues = @{ Example = 'test'; SecondExample = 'true' } - $mockDesiredValues = @{ Example = 'test'; SecondExample = 'false' } + $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter = { + $Permissions -eq @('CONTROL SERVER') + } - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - ValuesToCheck = @('SecondExample') - } + $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter = { + $Permissions -eq @('IMPERSONATE') + } - Test-SQLDscParameterState @testParameters | Should -Be $false + $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter = { + $Permissions -eq @('CONTROL') } - It 'Should return false when an empty hash table is used in the current values' { - $mockCurrentValues = @{ } - $mockDesiredValues = @{ Example = 'test'; SecondExample = 'false' } + $mockConnectionContextObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.ServerConnection + $mockConnectionContextObject.TrueLogin = 'Login1' - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + $mockServerObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.Server + $mockServerObject.ComputerNamePhysicalNetBIOS = 'Server1' + $mockServerObject.ServiceName = 'MSSQLSERVER' + $mockServerObject.ConnectionContext = $mockConnectionContextObject + } + + BeforeEach { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -MockWith { $false } -Verifiable + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -MockWith { $false } -Verifiable + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -MockWith { $false } -Verifiable + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -MockWith { $false } -Verifiable + } + + Context 'When impersonate permissions are present for the login' { + It 'Should return true when the impersonate any login permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $true - Test-SQLDscParameterState @testParameters | Should -Be $false + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly } - It 'Should return true when evaluating a table against a CimInstance' { - $mockCurrentValues = @{ Handle = '0'; ProcessId = '1000' } + It 'Should return true when the control server permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $true - $mockWin32ProcessProperties = @{ - Handle = 0 - ProcessId = 1000 - } + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly + } - $mockNewCimInstanceParameters = @{ - ClassName = 'Win32_Process' - Property = $mockWin32ProcessProperties - Key = 'Handle' - ClientOnly = $true - } + It 'Should return true when the impersonate login permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $true - $mockDesiredValues = New-CimInstance @mockNewCimInstanceParameters + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 1 -Exactly + } - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - ValuesToCheck = @('Handle','ProcessId') - } + It 'Should return true when the control login permissions are present for the login' { + Mock -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -MockWith { $true } -Verifiable + Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $true - Test-SQLDscParameterState @testParameters | Should -Be $true + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 1 -Exactly } + } - It 'Should return false when evaluating a table against a CimInstance and a value is wrong' { - $mockCurrentValues = @{ Handle = '1'; ProcessId = '1000' } + Context 'When impersonate permissions are missing for the login' { + It 'Should return false when the server permissions are missing for the login' { + Test-ImpersonatePermissions -ServerObject $mockServerObject | Should -Be $false - $mockWin32ProcessProperties = @{ - Handle = 0 - ProcessId = 1000 - } + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 0 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 0 -Exactly + } - $mockNewCimInstanceParameters = @{ - ClassName = 'Win32_Process' - Property = $mockWin32ProcessProperties - Key = 'Handle' - ClientOnly = $true - } + It 'Should return false when the login permissions are missing for the login' { + Test-ImpersonatePermissions -ServerObject $mockServerObject -SecurableName 'Login1' | Should -Be $false - $mockDesiredValues = New-CimInstance @mockNewCimInstanceParameters + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateAnyLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlServer_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ImpersonateLogin_ParameterFilter -Scope It -Times 1 -Exactly + Assert-MockCalled -CommandName Test-LoginEffectivePermissions -ParameterFilter $mockTestLoginEffectivePermissions_ControlLogin_ParameterFilter -Scope It -Times 1 -Exactly + } + } + } - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - ValuesToCheck = @('Handle','ProcessId') + Describe 'DscResource.Common\Connect-SQL' -Tag 'ConnectSql' { + BeforeAll { + $mockNewObject_MicrosoftDatabaseEngine = { + <# + $ArgumentList[0] will contain the ServiceInstance when calling mock New-Object. + But since the mock New-Object will also be called without arguments, we first + have to evaluate if $ArgumentList contains values. + #> + if( $ArgumentList.Count -gt 0) + { + $serverInstance = $ArgumentList[0] } - Test-SQLDscParameterState @testParameters | Should -Be $false - } - - It 'Should return true when evaluating a hash table containing an array' { - $mockCurrentValues = @{ Example = 'test'; SecondExample = @('1','2') } - $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + return New-Object -TypeName Object | + Add-Member -MemberType ScriptProperty -Name Status -Value { + if ($mockExpectedDatabaseEngineInstance -eq 'MSSQLSERVER') + { + $mockExpectedServiceInstance = $mockExpectedDatabaseEngineServer + } + else + { + $mockExpectedServiceInstance = "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + } - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + if ( $this.ConnectionContext.ServerInstance -eq $mockExpectedServiceInstance ) + { + return 'Online' + } + else + { + return $null + } + } -PassThru | + Add-Member -MemberType NoteProperty -Name ConnectionContext -Value ( + New-Object -TypeName Object | + Add-Member -MemberType NoteProperty -Name ServerInstance -Value $serverInstance -PassThru | + #Add-Member -MemberType ScriptProperty -Name LoginSecure -Value { [System.Boolean] $mockExpectedDatabaseEngineLoginSecure } -PassThru -Force | + Add-Member -MemberType NoteProperty -Name LoginSecure -Value $true -PassThru | + Add-Member -MemberType NoteProperty -Name Login -Value '' -PassThru | + Add-Member -MemberType NoteProperty -Name SecurePassword -Value $null -PassThru | + Add-Member -MemberType NoteProperty -Name ConnectAsUser -Value $false -PassThru | + Add-Member -MemberType NoteProperty -Name ConnectAsUserPassword -Value '' -PassThru | + Add-Member -MemberType NoteProperty -Name ConnectAsUserName -Value '' -PassThru | + Add-Member -MemberType ScriptMethod -Name Connect -Value { + if ($mockExpectedDatabaseEngineInstance -eq 'MSSQLSERVER') + { + $mockExpectedServiceInstance = $mockExpectedDatabaseEngineServer + } + else + { + $mockExpectedServiceInstance = "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + } + + if ($this.serverInstance -ne $mockExpectedServiceInstance) + { + throw ("Mock method Connect() was expecting ServerInstance to be '{0}', but was '{1}'." -f $mockExpectedServiceInstance, $this.serverInstance ) + } + + if ($mockThrowInvalidOperation) + { + throw 'Unable to connect.' + } + } -PassThru -Force + ) -PassThru -Force + } + + $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter = { + $TypeName -eq 'Microsoft.SqlServer.Management.Smo.Server' + } + + $mockThrowLocalizedMessage = { + throw $Message + } + + $mockSetupCredentialUserName = 'TestUserName12345' + $mockSetupCredentialPassword = 'StrongOne7.' + $mockSetupCredentialSecurePassword = ConvertTo-SecureString -String $mockSetupCredentialPassword -AsPlainText -Force + $mockSetupCredential = New-Object -TypeName PSCredential -ArgumentList ($mockSetupCredentialUserName, $mockSetupCredentialSecurePassword) + } - Test-SQLDscParameterState @testParameters | Should -Be $true - } + BeforeEach { + Mock -CommandName New-InvalidOperationException -MockWith $mockThrowLocalizedMessage -Verifiable + Mock -CommandName Import-SQLPSModule + Mock -CommandName New-Object ` + -MockWith $mockNewObject_MicrosoftDatabaseEngine ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter ` + -Verifiable + } - It 'Should return false when evaluating a hash table containing an array with wrong values' { - $mockCurrentValues = @{ Example = 'test'; SecondExample = @('A','B') } - $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + Context 'When connecting to the default instance using Windows Authentication' { + It 'Should return the correct service instance' { + $mockExpectedDatabaseEngineServer = 'TestServer' + $mockExpectedDatabaseEngineInstance = 'MSSQLSERVER' - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + $databaseEngineServerObject = Connect-SQL -ServerName $mockExpectedDatabaseEngineServer + $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly $mockExpectedDatabaseEngineServer - Test-SQLDscParameterState @testParameters | Should -Be $false + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter } + } - It 'Should return false when evaluating a hash table containing an array, but the CurrentValues are missing an array' { - $mockCurrentValues = @{ Example = 'test' } - $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + Context 'When connecting to the default instance using SQL Server Authentication' { + It 'Should return the correct service instance' { + $mockExpectedDatabaseEngineServer = 'TestServer' + $mockExpectedDatabaseEngineInstance = 'MSSQLSERVER' + $mockExpectedDatabaseEngineLoginSecure = $false - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + $databaseEngineServerObject = Connect-SQL -ServerName $mockExpectedDatabaseEngineServer -SetupCredential $mockSetupCredential -LoginType 'SqlLogin' + $databaseEngineServerObject.ConnectionContext.LoginSecure | Should -Be $false + $databaseEngineServerObject.ConnectionContext.Login | Should -Be $mockSetupCredentialUserName + $databaseEngineServerObject.ConnectionContext.SecurePassword | Should -Be $mockSetupCredentialSecurePassword + $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly $mockExpectedDatabaseEngineServer - Test-SQLDscParameterState @testParameters | Should -Be $false + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter } + } - It 'Should return false when evaluating a hash table containing an array, but the property i CurrentValues is $null' { - $mockCurrentValues = @{ Example = 'test'; SecondExample = $null } - $mockDesiredValues = @{ Example = 'test'; SecondExample = @('1','2') } + Context 'When connecting to the named instance using Windows Authentication' { + It 'Should return the correct service instance' { + $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME + $mockExpectedDatabaseEngineInstance = $mockInstanceName - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + $databaseEngineServerObject = Connect-SQL -InstanceName $mockExpectedDatabaseEngineInstance + $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" - Test-SQLDscParameterState @testParameters | Should -Be $false + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter } } - Context -Name 'When passing invalid types for DesiredValues' -Fixture { - It 'Should throw the correct error when DesiredValues is of wrong type' { - $mockCurrentValues = @{ Example = 'something' } - $mockDesiredValues = 'NotHashTable' + Context 'When connecting to the named instance using SQL Server Authentication' { + It 'Should return the correct service instance' { + $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME + $mockExpectedDatabaseEngineInstance = $mockInstanceName + $mockExpectedDatabaseEngineLoginSecure = $false - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - } + $databaseEngineServerObject = Connect-SQL -InstanceName $mockExpectedDatabaseEngineInstance -SetupCredential $mockSetupCredential -LoginType 'SqlLogin' + $databaseEngineServerObject.ConnectionContext.LoginSecure | Should -Be $false + $databaseEngineServerObject.ConnectionContext.Login | Should -Be $mockSetupCredentialUserName + $databaseEngineServerObject.ConnectionContext.SecurePassword | Should -Be $mockSetupCredentialSecurePassword + $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" - $mockCorrectErrorMessage = ($script:localizedData.PropertyTypeInvalidForDesiredValues -f $testParameters.DesiredValues.GetType().Name) - { Test-SQLDscParameterState @testParameters } | Should -Throw $mockCorrectErrorMessage + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter } + } - It 'Should write a warning when DesiredValues contain an unsupported type' { - Mock -CommandName Write-Warning -Verifiable - - # This is a dummy type to test with a type that could never be a correct one. - class MockUnknownType - { - [ValidateNotNullOrEmpty()] - [System.String] - $Property1 + Context 'When connecting to the named instance using Windows Authentication and different server name' { + It 'Should return the correct service instance' { + $mockExpectedDatabaseEngineServer = 'SERVER' + $mockExpectedDatabaseEngineInstance = $mockInstanceName - [ValidateNotNullOrEmpty()] - [System.String] - $Property2 + $databaseEngineServerObject = Connect-SQL -ServerName $mockExpectedDatabaseEngineServer -InstanceName $mockExpectedDatabaseEngineInstance + $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" - MockUnknownType() - { - } - } + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter + } + } - $mockCurrentValues = @{ Example = New-Object -TypeName MockUnknownType } - $mockDesiredValues = @{ Example = New-Object -TypeName MockUnknownType } + Context 'When connecting to the named instance using Windows Authentication impersonation' { + It 'Should return the correct service instance' { + $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME + $mockExpectedDatabaseEngineInstance = $mockInstanceName $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues + ServerName = $mockExpectedDatabaseEngineServer + InstanceName = $mockExpectedDatabaseEngineInstance + SetupCredential = $mockSetupCredential } - Test-SQLDscParameterState @testParameters | Should -Be $false + $databaseEngineServerObject = Connect-SQL @testParameters + $databaseEngineServerObject.ConnectionContext.ServerInstance | Should -BeExactly "$mockExpectedDatabaseEngineServer\$mockExpectedDatabaseEngineInstance" + $databaseEngineServerObject.ConnectionContext.ConnectAsUser | Should -Be $true + $databaseEngineServerObject.ConnectionContext.ConnectAsUserPassword | Should -BeExactly $mockSetupCredential.GetNetworkCredential().Password + $databaseEngineServerObject.ConnectionContext.ConnectAsUserName | Should -BeExactly $mockSetupCredential.GetNetworkCredential().UserName + $databaseEngineServerObject.ConnectionContext.ConnectAsUser | Should -Be $true - Assert-MockCalled -CommandName Write-Warning -Exactly -Times 1 + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter } } - Context -Name 'When passing an CimInstance as DesiredValue and ValuesToCheck is $null' -Fixture { + Context 'When connecting to the default instance using the correct service instance but does not return a correct Database Engine object' { It 'Should throw the correct error' { - $mockCurrentValues = @{ Example = 'something' } - - $mockWin32ProcessProperties = @{ - Handle = 0 - ProcessId = 1000 - } - - $mockNewCimInstanceParameters = @{ - ClassName = 'Win32_Process' - Property = $mockWin32ProcessProperties - Key = 'Handle' - ClientOnly = $true - } + $mockExpectedDatabaseEngineServer = $env:COMPUTERNAME + $mockExpectedDatabaseEngineInstance = $mockInstanceName - $mockDesiredValues = New-CimInstance @mockNewCimInstanceParameters + Mock -CommandName New-Object ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter ` + -Verifiable - $testParameters = @{ - CurrentValues = $mockCurrentValues - DesiredValues = $mockDesiredValues - ValuesToCheck = $null - } + $mockCorrectErrorMessage = ($script:localizedData.FailedToConnectToDatabaseEngineInstance -f $mockExpectedDatabaseEngineServer) + { Connect-SQL } | Should -Throw $mockCorrectErrorMessage - $mockCorrectErrorMessage = $script:localizedData.PropertyTypeInvalidForValuesToCheck - { Test-SQLDscParameterState @testParameters } | Should -Throw $mockCorrectErrorMessage + Assert-MockCalled -CommandName New-Object -Exactly -Times 1 -Scope It ` + -ParameterFilter $mockNewObject_MicrosoftDatabaseEngine_ParameterFilter } } Assert-VerifiableMock } - Describe 'Testing New-WarningMessage' -Tag NewWarningMessage { + Describe 'DscResource.Common\New-WarningMessage' -Tag 'NewWarningMessage' { Context -Name 'When writing a localized warning message' -Fixture { It 'Should write the error message without throwing' { Mock -CommandName Write-Warning -Verifiable @@ -1796,7 +2424,7 @@ InModuleScope $script:helperModuleName { Assert-VerifiableMock } - Describe 'Testing New-TerminatingError' -Tag NewWarningMessage { + Describe 'DscResource.Common\New-TerminatingError' -Tag 'NewTerminatingError' { Context -Name 'When building a localized error message' -Fixture { It 'Should return the correct error record with the correct error message' { $errorRecord = New-TerminatingError -ErrorType 'NoKeyFound' -FormatArgs 'Dummy error' @@ -1810,7 +2438,7 @@ InModuleScope $script:helperModuleName { It 'Should return the correct error record with a matching FullyQualifiedErrorId' { $errorRecord = New-TerminatingError -ErrorType 'NoKeyFound' -FormatArgs 'Dummy error' - $errorRecord.FullyQualifiedErrorId | Should -Be 'SqlServerDSCHelper.NoKeyFound' + $errorRecord.FullyQualifiedErrorId | Should -Be 'DscResource.NoKeyFound' } It 'Should return the correct error record with a matching FullyQualifiedErrorId when there is no calling module' { @@ -1844,7 +2472,7 @@ InModuleScope $script:helperModuleName { Assert-VerifiableMock } - Describe 'Testing Split-FullSQLInstanceName' { + Describe 'DscResource.Common\Split-FullSQLInstanceName' { Context 'When the "FullSQLInstanceName" parameter is not supplied' { It 'Should throw when the "FullSQLInstanceName" parameter is $null' { { Split-FullSQLInstanceName -FullSQLInstanceName $null } | Should -Throw @@ -1874,7 +2502,7 @@ InModuleScope $script:helperModuleName { } } - Describe 'Testing Test-ClusterPermissions' { + Describe 'DscResource.Common\Test-ClusterPermissions' { BeforeAll { Mock -CommandName Test-LoginEffectivePermissions -MockWith { $mockClusterServicePermissionsPresent @@ -1963,21 +2591,24 @@ InModuleScope $script:helperModuleName { } } - $mockGetService = { - return @{ - Name = $mockDynamicServiceName - DisplayName = $mockDynamicServiceDisplayName - DependentServices = @( - @{ - Name = $mockDynamicDependedServiceName - Status = 'Running' - DependentServices = @() + Describe 'DscResource.Common\Restart-ReportingServicesService' -Tag 'RestartReportingServicesService' { + BeforeAll { + $mockGetService = { + return @{ + Name = $mockDynamicServiceName + DisplayName = $mockDynamicServiceDisplayName + DependentServices = @( + @{ + Name = $mockDynamicDependedServiceName + Status = 'Running' + DependentServices = @() + } + ) } - ) + } + } - } - Describe 'Testing Restart-ReportingServicesService' { Context 'When restarting a Report Services default instance' { BeforeAll { $mockServiceName = 'ReportServer' @@ -2025,7 +2656,7 @@ InModuleScope $script:helperModuleName { } -Scope It -Exactly -Times 1 Assert-MockCalled -CommandName Stop-Service -Scope It -Exactly -Times 1 Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 2 - } + } } Context 'When restarting a Report Services named instance using a wait timer' { @@ -2052,11 +2683,11 @@ InModuleScope $script:helperModuleName { Assert-MockCalled -CommandName Stop-Service -Scope It -Exactly -Times 1 Assert-MockCalled -CommandName Start-Service -Scope It -Exactly -Times 2 Assert-MockCalled -CommandName Start-Sleep -Scope It -Exactly -Times 1 - } + } } } - Describe 'Testing Test-ActiveNode' { + Describe 'DscResource.Common\Test-ActiveNode' -Tag 'TestActiveNode' { BeforeAll { $mockServerObject = New-Object -TypeName Microsoft.SqlServer.Management.Smo.Server @@ -2101,15 +2732,17 @@ InModuleScope $script:helperModuleName { } } - Describe "Invoke-SqlScript" { - $invokeScriptFileParameters = @{ - ServerInstance = $env:COMPUTERNAME - InputFile = "set.sql" - } + Describe 'DscResource.Common\Invoke-SqlScript' -Tag 'InvokeSqlScript' { + BeforeAll { + $invokeScriptFileParameters = @{ + ServerInstance = $env:COMPUTERNAME + InputFile = "set.sql" + } - $invokeScriptQueryParameters = @{ - ServerInstance = $env:COMPUTERNAME - Query = "Test Query" + $invokeScriptQueryParameters = @{ + ServerInstance = $env:COMPUTERNAME + Query = "Test Query" + } } Context 'Invoke-SqlScript fails to import SQLPS module' { @@ -2125,55 +2758,71 @@ InModuleScope $script:helperModuleName { } Context 'Invoke-SqlScript is called with credentials' { - $passwordPlain = "password" - $user = "User" + BeforeAll { + $mockPasswordPlain = 'password' + $mockUsername = 'User' - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -ParameterFilter { - ($Username -eq $user) -and ($Password -eq $passwordPlain) - } + $password = ConvertTo-SecureString -String $mockPasswordPlain -AsPlainText -Force + $credential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList $mockUsername, $password - $password = ConvertTo-SecureString -String $passwordPlain -AsPlainText -Force - $cred = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList $user, $password + Mock -CommandName Import-SQLPSModule -MockWith {} + Mock -CommandName Invoke-Sqlcmd -ParameterFilter { + $Username -eq $mockUsername -and $Password -eq $mockPasswordPlain + } + } - It 'Should call Invoke-Sqlcmd with correct File parameterset parameters' { - $invokeScriptFileParameters.Add("Credential", $cred) + It 'Should call Invoke-Sqlcmd with correct File ParameterSet parameters' { + $invokeScriptFileParameters.Add('Credential', $credential) $null = Invoke-SqlScript @invokeScriptFileParameters Assert-MockCalled -CommandName Invoke-Sqlcmd -ParameterFilter { - ($Username -eq $user) -and ($Password -eq $passwordPlain) + $Username -eq $mockUsername -and $Password -eq $mockPasswordPlain } -Times 1 -Exactly -Scope It } - It 'Should call Invoke-Sqlcmd with correct Query parameterset parameters' { - $invokeScriptQueryParameters.Add("Credential", $cred) + It 'Should call Invoke-Sqlcmd with correct Query ParameterSet parameters' { + $invokeScriptQueryParameters.Add('Credential', $credential) $null = Invoke-SqlScript @invokeScriptQueryParameters Assert-MockCalled -CommandName Invoke-Sqlcmd -ParameterFilter { - ($Username -eq $user) -and ($Password -eq $passwordPlain) + $Username -eq $mockUsername -and $Password -eq $mockPasswordPlain } -Times 1 -Exactly -Scope It } } Context 'Invoke-SqlScript fails to execute the SQL scripts' { - $errorMessage = "Failed to run SQL Script" + $errorMessage = 'Failed to run SQL Script' Mock -CommandName Import-SQLPSModule -MockWith {} Mock -CommandName Invoke-Sqlcmd -MockWith { throw $errorMessage } - It 'Should throw the correct error from File parameterset Invoke-Sqlcmd' { + It 'Should throw the correct error from File ParameterSet Invoke-Sqlcmd' { { Invoke-SqlScript @invokeScriptFileParameters } | Should Throw $errorMessage } - It 'Should throw the correct error from Query parameterset Invoke-Sqlcmd' { + It 'Should throw the correct error from Query ParameterSet Invoke-Sqlcmd' { { Invoke-SqlScript @invokeScriptQueryParameters } | Should Throw $errorMessage } } } - Describe 'Testing Get-ServiceAccount'{ + Describe 'DscResource.Common\Get-ServiceAccount' -Tag 'GetServiceAccount' { + BeforeAll { + $mockLocalSystemAccountUserName = 'NT AUTHORITY\SYSTEM' + $mockLocalSystemAccountCredential = New-Object System.Management.Automation.PSCredential $mockLocalSystemAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) + + $mockManagedServiceAccountUserName = 'CONTOSO\msa$' + $mockManagedServiceAccountCredential = New-Object System.Management.Automation.PSCredential $mockManagedServiceAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) + + $mockDomainAccountUserName = 'CONTOSO\User1' + $mockDomainAccountCredential = New-Object System.Management.Automation.PSCredential $mockDomainAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) + + $mockLocalServiceAccountUserName = 'NT SERVICE\MyService' + $mockLocalServiceAccountCredential = New-Object System.Management.Automation.PSCredential $mockLocalServiceAccountUserName, (ConvertTo-SecureString "Password1" -AsPlainText -Force) + } + Context 'When getting service account' { It 'Should return NT AUTHORITY\SYSTEM' { $returnValue = Get-ServiceAccount -ServiceAccount $mockLocalSystemAccountCredential @@ -2204,7 +2853,15 @@ InModuleScope $script:helperModuleName { } } - Describe 'Testing Find-ExceptionByNumber'{ + Describe 'DscResource.Common\Find-ExceptionByNumber'{ + BeforeAll { + $mockInnerException = New-Object System.Exception "This is a mock inner excpetion object" + $mockInnerException | Add-Member -Name 'Number' -Value 2 -MemberType NoteProperty + + $mockException = New-Object System.Exception "This is a mock exception object", $mockInnerException + $mockException | Add-Member -Name 'Number' -Value 1 -MemberType NoteProperty + } + Context 'When searching Exception objects'{ It 'Should return true for main exception' { Find-ExceptionByNumber -ExceptionToSearch $mockException -ErrorNumber 1 | Should -Be $true @@ -2220,3 +2877,4 @@ InModuleScope $script:helperModuleName { } } } + diff --git a/Tests/Unit/DscResource.LocalizationHelper.Tests.ps1 b/Tests/Unit/DscResource.LocalizationHelper.Tests.ps1 new file mode 100644 index 000000000..2d054fab8 --- /dev/null +++ b/Tests/Unit/DscResource.LocalizationHelper.Tests.ps1 @@ -0,0 +1,203 @@ +<# + .SYNOPSIS + Automated unit test for helper functions in module DscResource.LocalizationHelper. + + .NOTES + To run this script locally, please make sure to first run the bootstrap + script. Read more at + https://github.com/PowerShell/SqlServerDsc/blob/dev/CONTRIBUTING.md#bootstrap-script-assert-testenvironment +#> + +Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') + +if (Test-SkipContinuousIntegrationTask -Type 'Unit') +{ + return +} + +# Import the DscResource.LocalizationHelper module to test +$script:resourceModulePath = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent +$script:modulesFolderPath = Join-Path -Path $script:resourceModulePath -ChildPath 'Modules\DscResource.LocalizationHelper' + +Import-Module -Name (Join-Path -Path $script:modulesFolderPath -ChildPath 'DscResource.LocalizationHelper.psm1') -Force + +InModuleScope 'DscResource.LocalizationHelper' { + Describe 'DscResource.LocalizationHelper\Get-LocalizedData' { + $mockTestPath = { + return $mockTestPathReturnValue + } + + $mockImportLocalizedData = { + $BaseDirectory | Should -Be $mockExpectedLanguagePath + } + + BeforeEach { + Mock -CommandName Test-Path -MockWith $mockTestPath -Verifiable + Mock -CommandName Import-LocalizedData -MockWith $mockImportLocalizedData -Verifiable + } + + Context 'When loading localized data for Swedish' { + $mockExpectedLanguagePath = 'sv-SE' + $mockTestPathReturnValue = $true + + It 'Should call Import-LocalizedData with sv-SE language' { + Mock -CommandName Join-Path -MockWith { + return 'sv-SE' + } -Verifiable + + { Get-LocalizedData -ResourceName 'DummyResource' } | Should -Not -Throw + + Assert-MockCalled -CommandName Join-Path -Exactly -Times 3 -Scope It + Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It + } + + $mockExpectedLanguagePath = 'en-US' + $mockTestPathReturnValue = $false + + It 'Should call Import-LocalizedData and fallback to en-US if sv-SE language does not exist' { + Mock -CommandName Join-Path -MockWith { + return $ChildPath + } -Verifiable + + { Get-LocalizedData -ResourceName 'DummyResource' } | Should -Not -Throw + + Assert-MockCalled -CommandName Join-Path -Exactly -Times 4 -Scope It + Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It + } + + Context 'When $ScriptRoot is set to a path' { + $mockExpectedLanguagePath = 'sv-SE' + $mockTestPathReturnValue = $true + + It 'Should call Import-LocalizedData with sv-SE language' { + Mock -CommandName Join-Path -MockWith { + return 'sv-SE' + } -Verifiable + + { Get-LocalizedData -ResourceName 'DummyResource' -ScriptRoot '.' } | Should -Not -Throw + + Assert-MockCalled -CommandName Join-Path -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It + } + + $mockExpectedLanguagePath = 'en-US' + $mockTestPathReturnValue = $false + + It 'Should call Import-LocalizedData and fallback to en-US if sv-SE language does not exist' { + Mock -CommandName Join-Path -MockWith { + return $ChildPath + } -Verifiable + + { Get-LocalizedData -ResourceName 'DummyResource' -ScriptRoot '.' } | Should -Not -Throw + + Assert-MockCalled -CommandName Join-Path -Exactly -Times 2 -Scope It + Assert-MockCalled -CommandName Test-Path -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Import-LocalizedData -Exactly -Times 1 -Scope It + } + } + } + + Context 'When loading localized data for English' { + Mock -CommandName Join-Path -MockWith { + return 'en-US' + } -Verifiable + + $mockExpectedLanguagePath = 'en-US' + $mockTestPathReturnValue = $true + + It 'Should call Import-LocalizedData with en-US language' { + { Get-LocalizedData -ResourceName 'DummyResource' } | Should -Not -Throw + } + } + + Assert-VerifiableMock + } + + Describe 'DscResource.LocalizationHelper\New-InvalidResultException' { + Context 'When calling with Message parameter only' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + + { New-InvalidResultException -Message $mockErrorMessage } | Should -Throw $mockErrorMessage + } + } + + Context 'When calling with both the Message and ErrorRecord parameter' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + $mockExceptionErrorMessage = 'Mocked exception error message' + + $mockException = New-Object -TypeName System.Exception -ArgumentList $mockExceptionErrorMessage + $mockErrorRecord = New-Object -TypeName System.Management.Automation.ErrorRecord -ArgumentList $mockException, $null, 'InvalidResult', $null + + { New-InvalidResultException -Message $mockErrorMessage -ErrorRecord $mockErrorRecord } | Should -Throw ('System.Exception: {0} ---> System.Exception: {1}' -f $mockErrorMessage, $mockExceptionErrorMessage) + } + } + + Assert-VerifiableMock + } + + Describe 'DscResource.LocalizationHelper\New-ObjectNotFoundException' { + Context 'When calling with Message parameter only' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + + { New-ObjectNotFoundException -Message $mockErrorMessage } | Should -Throw $mockErrorMessage + } + } + + Context 'When calling with both the Message and ErrorRecord parameter' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + $mockExceptionErrorMessage = 'Mocked exception error message' + + $mockException = New-Object -TypeName System.Exception -ArgumentList $mockExceptionErrorMessage + $mockErrorRecord = New-Object -TypeName System.Management.Automation.ErrorRecord -ArgumentList $mockException, $null, 'InvalidResult', $null + + { New-ObjectNotFoundException -Message $mockErrorMessage -ErrorRecord $mockErrorRecord } | Should -Throw ('System.Exception: {0} ---> System.Exception: {1}' -f $mockErrorMessage, $mockExceptionErrorMessage) + } + } + + Assert-VerifiableMock + } + + Describe 'DscResource.LocalizationHelper\New-InvalidOperationException' { + Context 'When calling with Message parameter only' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + + { New-InvalidOperationException -Message $mockErrorMessage } | Should -Throw $mockErrorMessage + } + } + + Context 'When calling with both the Message and ErrorRecord parameter' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + $mockExceptionErrorMessage = 'Mocked exception error message' + + $mockException = New-Object -TypeName System.Exception -ArgumentList $mockExceptionErrorMessage + $mockErrorRecord = New-Object -TypeName System.Management.Automation.ErrorRecord -ArgumentList $mockException, $null, 'InvalidResult', $null + + { New-InvalidOperationException -Message $mockErrorMessage -ErrorRecord $mockErrorRecord } | Should -Throw ('System.InvalidOperationException: {0} ---> System.Exception: {1}' -f $mockErrorMessage, $mockExceptionErrorMessage) + } + } + + Assert-VerifiableMock + } + + Describe 'DscResource.LocalizationHelper\New-InvalidArgumentException' { + Context 'When calling with both the Message and ArgumentName parameter' { + It 'Should throw the correct error' { + $mockErrorMessage = 'Mocked error' + $mockArgumentName = 'MockArgument' + + { New-InvalidArgumentException -Message $mockErrorMessage -ArgumentName $mockArgumentName } | Should -Throw ('Parameter name: {0}' -f $mockArgumentName) + } + } + + Assert-VerifiableMock + } +} diff --git a/Tests/Unit/MSFT_SqlAG.Tests.ps1 b/Tests/Unit/MSFT_SqlAG.Tests.ps1 index cc5d839a4..4ab30f8fe 100644 --- a/Tests/Unit/MSFT_SqlAG.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlAG.Tests.ps1 @@ -39,8 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - # Load the SMO stubs - Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') # Load the default SQL Module stub Import-SQLModuleStub @@ -59,7 +59,7 @@ try InModuleScope $script:dscResourceName { # This is relative to the path of the resource module script, not test test script. $script:moduleRoot = Split-Path -Path (Split-Path -Path $PSScriptRoot -Parent) -Parent - Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'Tests' -ChildPath (Join-Path -Path 'TestHelpers' -ChildPath 'CommonTestHelper.psm1'))) -Force -Global + Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'Tests' -ChildPath (Join-Path -Path 'TestHelpers' -ChildPath 'CommonTestHelper.psm1'))) -Force #region parameter mocks diff --git a/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 b/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 index 788ff9f74..24037c00c 100644 --- a/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlAGDatabase.Tests.ps1 @@ -29,11 +29,6 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR } Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'DSCResource.Tests' -ChildPath 'TestHelper.psm1')) -Force -Import-Module -Name (Join-Path -Path (Split-Path -Path $PSScriptRoot -Parent | Split-Path -Parent) -ChildPath 'SqlServerDscHelper.psm1') -Scope Global -Force -Import-Module -Name ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SQLPSStub.psm1 ) -Force -Global - -# Loading mocked classes -Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` @@ -44,6 +39,11 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') + + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlAGListener.Tests.ps1 b/Tests/Unit/MSFT_SqlAGListener.Tests.ps1 index a796d1e35..8033eae08 100644 --- a/Tests/Unit/MSFT_SqlAGListener.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlAGListener.Tests.ps1 @@ -39,8 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - # Loading stub cmdlets - Import-Module -Name ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SQLPSStub.psm1 ) -Force -Global + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 b/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 index 6233f9b98..df79ed5fd 100644 --- a/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlAGReplica.Tests.ps1 @@ -28,9 +28,7 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR & git @('clone', 'https://github.com/PowerShell/DscResource.Tests.git', (Join-Path -Path $script:moduleRoot -ChildPath '\DSCResource.Tests\')) } -Import-Module -Name ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SQLPSStub.psm1 ) -Force -Global Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'DSCResource.Tests' -ChildPath 'TestHelper.psm1')) -Force -Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` @@ -41,6 +39,11 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') + + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlDatabase.Tests.ps1 b/Tests/Unit/MSFT_SqlDatabase.Tests.ps1 index 2b993f5ae..feac8d33a 100644 --- a/Tests/Unit/MSFT_SqlDatabase.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlDatabase.Tests.ps1 @@ -39,7 +39,6 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - # Loading mocked classes } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlDatabaseDefaultLocation.Tests.ps1 b/Tests/Unit/MSFT_SqlDatabaseDefaultLocation.Tests.ps1 index fb16ce95c..793bac7f5 100644 --- a/Tests/Unit/MSFT_SqlDatabaseDefaultLocation.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlDatabaseDefaultLocation.Tests.ps1 @@ -27,9 +27,7 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR & git @('clone', 'https://github.com/PowerShell/DscResource.Tests.git', (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests')) } -Import-Module -Name ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SQLPSStub.psm1 ) -Force -Global Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath (Join-Path -Path 'DSCResource.Tests' -ChildPath 'TestHelper.psm1')) -Force -Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` @@ -40,6 +38,12 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') + + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global + } function Invoke-TestCleanup @@ -58,7 +62,10 @@ try $mockInstanceName = 'MSSQLSERVER' $mockSQLDataPath = 'C:\Program Files\Data\' $mockSqlLogPath = 'C:\Program Files\Log\' + + # Ending backslash is regression test for issue #1307. $mockSqlBackupPath = 'C:\Program Files\Backup\' + $mockSqlAlterDataPath = 'C:\Program Files\' $mockSqlAlterLogPath = 'C:\Program Files\' $mockSqlAlterBackupPath = 'C:\Program Files\' @@ -88,7 +95,8 @@ try Add-Member -MemberType NoteProperty -Name ComputerNamePhysicalNetBIOS -Value $mockServerName -PassThru -Force | Add-Member -MemberType NoteProperty -Name DefaultFile -Value $mockSqlDataPath -PassThru -Force | Add-Member -MemberType NoteProperty -Name DefaultLog -Value $mockSqlLogPath -PassThru -Force | - Add-Member -MemberType NoteProperty -Name BackupDirectory -Value $mockSqlBackupPath -PassThru -Force | + # Ending backslash is removed because of regression test for issue #1307. + Add-Member -MemberType NoteProperty -Name BackupDirectory -Value $mockSqlBackupPath.TrimEnd('\') -PassThru -Force | Add-Member -MemberType ScriptMethod -Name Alter -Value { if ($mockInvalidOperationForAlterMethod) { @@ -97,36 +105,30 @@ try $script:WasMethodAlterCalled = $true } -PassThru -Force } - - $testCases = @( - @{ - Type = 'Data' - Path = $mockSqlDataPath - AlterPath = $mockSqlAlterDataPath - ExpectedAlterPath = $mockExpectedAlterDataPath - InvalidPath = $mockInvalidPathForData - }, - @{ - Type = 'Log' - Path = $mockSqlLogPath - AlterPath = $mockSqlAlterLogPath - ExpectedAlterPath = $mockExpectedAlterLogPath - InvalidPath = $mockInvalidPathForLog - }, - @{ - Type = 'Backup' - Path = $mockSqlBackupPath - AlterPath = $mockSqlAlterBackupPath - ExpectedAlterPath = $mockExpectedAlterBackupPath - InvalidPath = $mockInvalidPathForBackup - } - ) #endregion Describe 'MSFT_SqlDatabaseDefaultLocation\Get-TargetResource' -Tag 'Get' { + BeforeAll { + $testCases = @( + @{ + Type = 'Data' + Path = $mockSqlDataPath + }, + @{ + Type = 'Log' + Path = $mockSqlLogPath + }, + @{ + Type = 'Backup' + # Ending backslash is removed because of regression test for issue #1307. + Path = $mockSqlBackupPath.TrimEnd('\') + } + ) + } + BeforeEach { Mock -CommandName Connect-SQL -MockWith $mockConnectSQL -Verifiable - Mock -CommandName Test-ActiveNode -Mockwith { + Mock -CommandName Test-ActiveNode -MockWith { param ( [PSObject] @@ -158,6 +160,25 @@ try } Describe 'MSFT_SqlDatabaseDefaultLocation\Test-TargetResource' -Tag 'Test' { + BeforeAll { + $testCases = @( + @{ + Type = 'Data' + Path = $mockSqlDataPath + AlterPath = $mockSqlAlterDataPath + }, + @{ + Type = 'Log' + Path = $mockSqlLogPath + AlterPath = $mockSqlAlterLogPath + }, + @{ + Type = 'Backup' + Path = $mockSqlBackupPath + AlterPath = $mockSqlAlterBackupPath + } + ) + } BeforeEach { Mock -CommandName Connect-SQL -MockWith $mockConnectSQL -Verifiable Mock -CommandName Test-ActiveNode -MockWith { @@ -236,6 +257,32 @@ try } Describe 'MSFT_SqlDatabaseDefaultLocation\Set-TargetResource' -Tag 'Set' { + BeforeAll { + $testCases = @( + @{ + Type = 'Data' + Path = $mockSqlDataPath + AlterPath = $mockSqlAlterDataPath + ExpectedAlterPath = $mockExpectedAlterDataPath + InvalidPath = $mockInvalidPathForData + }, + @{ + Type = 'Log' + Path = $mockSqlLogPath + AlterPath = $mockSqlAlterLogPath + ExpectedAlterPath = $mockExpectedAlterLogPath + InvalidPath = $mockInvalidPathForLog + }, + @{ + Type = 'Backup' + Path = $mockSqlBackupPath + AlterPath = $mockSqlAlterBackupPath + ExpectedAlterPath = $mockExpectedAlterBackupPath + InvalidPath = $mockInvalidPathForBackup + } + ) + } + BeforeEach { Mock -CommandName Connect-SQL -MockWith $mockConnectSQL -Verifiable Mock -CommandName Restart-SqlService -Verifiable diff --git a/Tests/Unit/MSFT_SqlDatabasePermission.Tests.ps1 b/Tests/Unit/MSFT_SqlDatabasePermission.Tests.ps1 index e81d15864..aba28b934 100644 --- a/Tests/Unit/MSFT_SqlDatabasePermission.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlDatabasePermission.Tests.ps1 @@ -40,7 +40,7 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { # Loading mocked classes - Add-Type -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests\Unit\Stubs\SMO.cs') + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlDatabaseRole.Tests.ps1 b/Tests/Unit/MSFT_SqlDatabaseRole.Tests.ps1 index ab05b2d2a..56c38c34f 100644 --- a/Tests/Unit/MSFT_SqlDatabaseRole.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlDatabaseRole.Tests.ps1 @@ -40,7 +40,7 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { # Loading mocked classes - Add-Type -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests\Unit\Stubs\SMO.cs') + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlRS.Tests.ps1 b/Tests/Unit/MSFT_SqlRS.Tests.ps1 index 49b25afe5..c451117ed 100644 --- a/Tests/Unit/MSFT_SqlRS.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlRS.Tests.ps1 @@ -38,7 +38,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - Import-Module -Name (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Global -Force + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlRSSetup.Tests.ps1 b/Tests/Unit/MSFT_SqlRSSetup.Tests.ps1 new file mode 100644 index 000000000..4e2d3b002 --- /dev/null +++ b/Tests/Unit/MSFT_SqlRSSetup.Tests.ps1 @@ -0,0 +1,958 @@ +<# + .SYNOPSIS + Automated unit test for MSFT_SqlRSSetup DSC resource. + + .NOTES + To run this script locally, please make sure to first run the bootstrap + script. Read more at + https://github.com/PowerShell/SqlServerDsc/blob/dev/CONTRIBUTING.md#bootstrap-script-assert-testenvironment +#> + +Import-Module -Name (Join-Path -Path $PSScriptRoot -ChildPath '..\TestHelpers\CommonTestHelper.psm1') + +if (Test-SkipContinuousIntegrationTask -Type 'Unit') +{ + return +} + +$script:dscModuleName = 'SqlServerDsc' +$script:dscResourceName = 'MSFT_SqlRSSetup' + +#region HEADER + +# Unit Test Template Version: 1.2.0 +$script:moduleRoot = Split-Path -Parent (Split-Path -Parent $PSScriptRoot) +if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests'))) -or ` + (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1'))) ) +{ + & git @('clone', 'https://github.com/PowerShell/DscResource.Tests.git', (Join-Path -Path $script:moduleRoot -ChildPath '\DSCResource.Tests\')) +} + +Import-Module (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1') -Force + +$TestEnvironment = Initialize-TestEnvironment ` + -DSCModuleName $script:dscModuleName ` + -DSCResourceName $script:dscResourceName ` + -TestType Unit + +#endregion HEADER + +function Invoke-TestSetup +{ +} + +function Invoke-TestCleanup +{ + Restore-TestEnvironment -TestEnvironment $TestEnvironment +} + +# Begin Testing +try +{ + Invoke-TestSetup + + InModuleScope $script:dscResourceName { + <# + .SYNOPSIS + Used to test arguments passed to Start-SqlSetupProcess while inside and It-block. + + This function must be called inside a Mock, since it depends being run inside an It-block. + + .PARAMETER Argument + A string containing all the arguments separated with space and each argument should start with '/'. + Only the first string in the array is evaluated. + + .PARAMETER ExpectedArgument + A hash table containing all the expected arguments. + #> + function Test-SetupArgument + { + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $Argument, + + [Parameter(Mandatory = $true)] + [System.Collections.Hashtable] + $ExpectedArgument + ) + + $argumentHashTable = @{} + + # Break the argument string into a hash table + ($Argument -split ' ?/') | ForEach-Object { + if ($_ -imatch '(\w+)="?([^\/]+)"?') + { + $key = $Matches[1] + $value = ($Matches[2] -replace '" "',' ') -replace '"','' + + $argumentHashTable.Add($key, $value) + } + elseif ($_ -imatch '(\w+)') + { + $key = $Matches[1] + $value = [System.Management.Automation.SwitchParameter] $true + + $argumentHashTable.Add($key, $value) + } + } + + $actualValues = $argumentHashTable.Clone() + + # Limit the output in the console when everything is fine. + if ($actualValues.Count -ne $ExpectedArgument.Count) + { + Write-Warning -Message 'Verified the setup argument count (expected vs actual)' + Write-Warning -Message ('Expected: {0}' -f ($ExpectedArgument.Keys -join ',')) + Write-Warning -Message ('Actual: {0}' -f ($actualValues.Keys -join ',')) + } + + # Start by checking whether we have the same number of parameters + $actualValues.Count | Should -Be $ExpectedArgument.Count ` + -Because ('the expected arguments was: {0}' -f ($ExpectedArgument.Keys -join ',')) + + Write-Verbose -Message 'Verified actual setup argument values against expected setup argument values' -Verbose + + foreach ($argumentKey in $ExpectedArgument.Keys) + { + $argumentKeyName = $actualValues.GetEnumerator() | + Where-Object -FilterScript { + $_.Name -eq $argumentKey + } | Select-Object -ExpandProperty 'Name' + + $argumentValue = $actualValues.$argumentKey + + $argumentKeyName | Should -Be $argumentKey + $argumentValue | Should -Be $ExpectedArgument.$argumentKey + } + } + + $mockInstanceName = 'SSRS' + $mockCurrentVersion = '14.0.6514.11481' + + # Default parameters that are used for the It-blocks. + $mockDefaultParameters = @{ + InstanceName = $mockInstanceName + IAcceptLicenseTerms = 'Yes' + SourcePath = '\\server\share\SQLServerReportingServices.exe' + } + + Describe "MSFT_SqlRSSetup\Get-TargetResource" -Tag 'Get' { + BeforeEach { + $mockGetTargetResourceParameters = $mockDefaultParameters.Clone() + } + + Context 'When the system is in the desired state' { + Context 'When there are no installed Reporting Services' { + BeforeAll { + Mock -CommandName Get-RegistryPropertyValue + } + + It 'Should return $null as the InstanceName' { + $result = Get-TargetResource @mockGetTargetResourceParameters + $result.InstanceName | Should -BeNullOrEmpty + + Assert-MockCalled -CommandName Get-RegistryPropertyValue -Exactly -Times 1 -Scope 'It' + } + + It 'Should return the same values as passed as parameters' { + $result = Get-TargetResource @mockGetTargetResourceParameters + $result.IAcceptLicenseTerms | Should -Be $mockGetTargetResourceParameters.IAcceptLicenseTerms + $result.SourcePath | Should -Be $mockGetTargetResourceParameters.SourcePath + + Assert-MockCalled -CommandName Get-RegistryPropertyValue -Exactly -Times 1 -Scope 'It' + } + + It 'Should return $null or $false for the rest of the properties' { + $getTargetResourceResult = Get-TargetResource @mockGetTargetResourceParameters + $getTargetResourceResult.Action | Should -BeNullOrEmpty + $getTargetResourceResult.SourceCredential | Should -BeNullOrEmpty + $getTargetResourceResult.ProductKey | Should -BeNullOrEmpty + $getTargetResourceResult.ForceRestart | Should -BeFalse + $getTargetResourceResult.EditionUpgrade | Should -BeFalse + $getTargetResourceResult.Edition | Should -BeNullOrEmpty + $getTargetResourceResult.LogPath | Should -BeNullOrEmpty + $getTargetResourceResult.InstallFolder | Should -BeNullOrEmpty + $getTargetResourceResult.ErrorDumpDirectory | Should -BeNullOrEmpty + $getTargetResourceResult.CurrentVersion | Should -BeNullOrEmpty + $getTargetResourceResult.ServiceName | Should -BeNullOrEmpty + + Assert-MockCalled -CommandName Get-RegistryPropertyValue -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When there is an installed Reporting Services' { + BeforeAll { + $mockGetRegistryPropertyValue_InstanceName = { + <# + Currently only the one instance name of 'SSRS' is supported, + and the same name is currently used for instance id. + #> + return $mockInstanceName + } + + $mockGetRegistryPropertyValue_InstanceName_ParameterFilter = { + $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\RS' ` + -and $Name -eq $mockInstanceName + } + + $mockInstallRootDirectory = 'C:\Program Files\Microsoft SQL Server Reporting Services' + $mockGetRegistryPropertyValue_InstallRootDirectory = { + return $mockInstallRootDirectory + } + + $mockGetRegistryPropertyValue_InstallRootDirectory_ParameterFilter = { + $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\SSRS\Setup' ` + -and $Name -eq 'InstallRootDirectory' + } + + $mockServiceName = 'SQLServerReportingServices' + $mockGetRegistryPropertyValue_ServiceName = { + return $mockServiceName + } + + $mockGetRegistryPropertyValue_ServiceName_ParameterFilter = { + $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\SSRS\Setup' ` + -and $Name -eq 'ServiceName' + } + + $mockErrorDumpDir = 'C:\Program Files\Microsoft SQL Server Reporting Services\SSRS\LogFiles' + $mockGetRegistryPropertyValue_ErrorDumpDir = { + return $mockErrorDumpDir + } + + $mockGetRegistryPropertyValue_ErrorDumpDir_ParameterFilter = { + $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\SSRS\CPE' ` + -and $Name -eq 'ErrorDumpDir' + } + + $mockGetPackage_CurrentVersion = { + return @{ + Version = $mockCurrentVersion + } + } + + Mock -CommandName Get-RegistryPropertyValue ` + -MockWith $mockGetRegistryPropertyValue_InstanceName ` + -ParameterFilter $mockGetRegistryPropertyValue_InstanceName_ParameterFilter + + Mock -CommandName Get-RegistryPropertyValue ` + -MockWith $mockGetRegistryPropertyValue_InstallRootDirectory ` + -ParameterFilter $mockGetRegistryPropertyValue_InstallRootDirectory_ParameterFilter + + Mock -CommandName Get-RegistryPropertyValue ` + -MockWith $mockGetRegistryPropertyValue_ServiceName ` + -ParameterFilter $mockGetRegistryPropertyValue_ServiceName_ParameterFilter + + Mock -CommandName Get-RegistryPropertyValue ` + -MockWith $mockGetRegistryPropertyValue_ErrorDumpDir ` + -ParameterFilter $mockGetRegistryPropertyValue_ErrorDumpDir_ParameterFilter + + # This is a workaround for the issue https://github.com/pester/Pester/issues/604. + function Get-Package + { + [CmdletBinding()] + param + ( + [Parameter(Mandatory = $true)] + [System.String] + $Name, + + [Parameter(Mandatory = $true)] + [System.String] + $ProviderName + ) + + throw '{0}: StubNotImplemented' -f $MyInvocation.MyCommand + } + + Mock -CommandName Get-Package -MockWith $mockGetPackage_CurrentVersion + } + + It 'Should return the correct InstanceName' { + $result = Get-TargetResource @mockGetTargetResourceParameters + $result.InstanceName | Should -Be $mockGetTargetResourceParameters.InstanceName + + Assert-MockCalled -CommandName Get-RegistryPropertyValue ` + -ParameterFilter $mockGetRegistryPropertyValue_InstanceName_ParameterFilter ` + -Exactly -Times 1 -Scope 'It' + } + + It 'Should return the same values as passed as parameters' { + $result = Get-TargetResource @mockGetTargetResourceParameters + $result.IAcceptLicenseTerms | Should -Be $mockGetTargetResourceParameters.IAcceptLicenseTerms + $result.SourcePath | Should -Be $mockGetTargetResourceParameters.SourcePath + } + + It 'Should return the correct values for the rest of the properties' { + $getTargetResourceResult = Get-TargetResource @mockGetTargetResourceParameters + $getTargetResourceResult.Action | Should -BeNullOrEmpty + $getTargetResourceResult.SourceCredential | Should -BeNullOrEmpty + $getTargetResourceResult.ProductKey | Should -BeNullOrEmpty + $getTargetResourceResult.ForceRestart | Should -BeFalse + $getTargetResourceResult.EditionUpgrade | Should -BeFalse + $getTargetResourceResult.Edition | Should -BeNullOrEmpty + $getTargetResourceResult.LogPath | Should -BeNullOrEmpty + $getTargetResourceResult.InstallFolder | Should -Be $mockInstallRootDirectory + $getTargetResourceResult.ErrorDumpDirectory | Should -Be $mockErrorDumpDir + $getTargetResourceResult.CurrentVersion | Should -Be $mockCurrentVersion + $getTargetResourceResult.ServiceName | Should -Be $mockServiceName + + Assert-MockCalled -CommandName Get-RegistryPropertyValue ` + -ParameterFilter $mockGetRegistryPropertyValue_InstallRootDirectory_ParameterFilter ` + -Exactly -Times 1 -Scope 'It' + + Assert-MockCalled -CommandName Get-RegistryPropertyValue ` + -ParameterFilter $mockGetRegistryPropertyValue_ServiceName_ParameterFilter ` + -Exactly -Times 1 -Scope 'It' + + Assert-MockCalled -CommandName Get-RegistryPropertyValue ` + -ParameterFilter $mockGetRegistryPropertyValue_ErrorDumpDir_ParameterFilter ` + -Exactly -Times 1 -Scope 'It' + + Assert-MockCalled -CommandName Get-Package -Exactly -Times 1 -Scope 'It' + } + + Context 'When there is an installed Reporting Services, but no installed package is found to determine version' { + BeforeEach { + Mock -CommandName Get-Package + Mock -CommandName Write-Warning + } + + It 'Should return the correct values for the rest of the properties' { + $getTargetResourceResult = Get-TargetResource @mockGetTargetResourceParameters + $getTargetResourceResult.CurrentVersion | Should -BeNullOrEmpty + + Assert-MockCalled -CommandName Write-Warning -Exactly -Times 1 -Scope 'It' + } + } + } + } + } + + Describe "MSFT_SqlRSSetup\Test-TargetResource" -Tag 'Test' { + BeforeEach { + $mockTestTargetResourceParameters = $mockDefaultParameters.Clone() + } + + Context 'When the system is in the desired state' { + Context 'When there are no installed Reporting Services' { + BeforeAll { + Mock -CommandName Get-TargetResource -MockWith { + return @{ + InstanceName = $null + } + } + } + + It 'Should return $true' { + $mockTestTargetResourceParameters['Action'] = 'Uninstall' + + $result = Test-TargetResource @mockTestTargetResourceParameters + $result | Should -BeTrue + + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When there is an installed Reporting Services' { + BeforeAll { + Mock -CommandName Get-TargetResource -MockWith { + return @{ + InstanceName = 'SSRS' + CurrentVersion = $mockCurrentVersion + } + } + + Mock -CommandName Get-FileProductVersion -MockWith { + return [System.Version] $mockCurrentVersion + } + } + + It 'Should return $true' { + $result = Test-TargetResource @mockTestTargetResourceParameters + $result | Should -BeTrue + + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Get-FileProductVersion -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When the installed Reporting Services is an older version that the installation media, but parameter VersionUpgrade is not used' { + BeforeAll { + Mock -CommandName Get-TargetResource -MockWith { + return @{ + InstanceName = 'SSRS' + CurrentVersion = $mockCurrentVersion + } + } + + Mock -CommandName Get-FileProductVersion -MockWith { + return [System.Version] '15.1.1.0' + } + } + + It 'Should return $false' { + # This is called without the parameter 'VersionUpgrade'. + $result = Test-TargetResource @mockTestTargetResourceParameters + $result | Should -BeTrue + + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Get-FileProductVersion -Exactly -Times 1 -Scope 'It' + } + } + } + + Context 'When the system is not in the desired state' { + Context 'When there should be no installed Reporting Services' { + BeforeAll { + Mock -CommandName Get-TargetResource -MockWith { + return @{ + InstanceName = 'SSRS' + } + } + } + + It 'Should return $false' { + $mockTestTargetResourceParameters['Action'] = 'Uninstall' + + $result = Test-TargetResource @mockTestTargetResourceParameters + $result | Should -BeFalse + + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When there are no installed Reporting Services' { + BeforeAll { + Mock -CommandName Get-TargetResource -MockWith { + return @{ + InstanceName = $null + } + } + + Mock -CommandName Get-FileProductVersion -MockWith { + return [System.Version] $mockCurrentVersion + } + } + + It 'Should return $false' { + $result = Test-TargetResource @mockTestTargetResourceParameters -Verbose + $result | Should -BeFalse + + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When the wrong version of Reporting Services is installed, and parameter VersionUpgrade is used' { + BeforeAll { + Mock -CommandName Get-TargetResource -MockWith { + return @{ + InstanceName = 'SSRS' + CurrentVersion = $mockCurrentVersion + } + } + + Mock -CommandName Get-FileProductVersion -MockWith { + return [System.Version] '15.1.1.0' + } + } + + It 'Should return $false' { + $mockTestTargetResourceParameters['VersionUpgrade'] = $true + + $result = Test-TargetResource @mockTestTargetResourceParameters -Verbose + $result | Should -BeFalse + + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Get-FileProductVersion -Exactly -Times 1 -Scope 'It' + } + } + } + } + + Describe "MSFT_SqlRSSetup\Set-TargetResource" -Tag 'Set' { + BeforeAll { + $mockProductKey = '1FAKE-2FAKE-3FAKE-4FAKE-5FAKE' + } + + BeforeEach { + $mockSetTargetResourceParameters = $mockDefaultParameters.Clone() + + # Reset global variable DSCMachineStatus before each test. + $global:DSCMachineStatus = 0 + } + + + Context 'When providing a missing SourcePath' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + + Mock -CommandName Test-Path -MockWith { + return $false + } + } + + It 'Should throw the correct error message' { + $errorMessage = $script:localizedData.SourcePathNotFound -f $mockSetTargetResourceParameters.SourcePath + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw $errorMessage + } + } + + Context 'When providing a correct path in SourcePath, but no executable' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + + Mock -CommandName Test-Path -MockWith { + return $true + } + + Mock -CommandName Get-Item -MockWith { + return @{ + Extension = '' + } + } + } + + It 'Should throw the correct error message' { + $errorMessage = $script:localizedData.SourcePathNotFound -f $mockSetTargetResourceParameters.SourcePath + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw $errorMessage + } + } + + Context 'When providing both the parameters ProductKey and Edition' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + $mockSetTargetResourceParameters['ProductKey'] = $mockProductKey + } + + It 'Should throw the correct error message' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw $script:localizedData.EditionInvalidParameter + } + } + + Context 'When providing neither the parameters ProductKey or Edition' { + It 'Should throw the correct error message' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw $script:localizedData.EditionMissingParameter + } + } + + Context 'When the system is not in the desired state' { + BeforeAll { + Mock -CommandName Test-Path -MockWith { + return $true + } + + Mock -CommandName Get-Item -MockWith { + return @{ + Extension = '.exe' + } + } + } + + Context 'When Reporting Services are installed with the minimum required parameters' { + BeforeEach { + $mockSetTargetResourceParameters['ProductKey'] = $mockProductKey + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + PID = $mockProductKey + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When Reporting Services should be uninstalled' { + BeforeEach { + $mockSetTargetResourceParameters['Action'] = 'Uninstall' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + Uninstall = [System.Management.Automation.SwitchParameter] $true + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When Reporting Services are installed with parameter Edition' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When Reporting Services are installed with parameters ProductKey, SuppressRestart, LogPath, EditionUpgrade, and InstallFolder' { + BeforeEach { + $mockSetTargetResourceParameters['ProductKey'] = $mockProductKey + $mockSetTargetResourceParameters['SuppressRestart'] = $true + $mockSetTargetResourceParameters['LogPath'] = 'log.txt' + $mockSetTargetResourceParameters['EditionUpgrade'] = $true + $mockSetTargetResourceParameters['InstallFolder'] = 'C:\Temp' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + PID = $mockProductKey + NoRestart = [System.Management.Automation.SwitchParameter] $true + Log = 'log.txt' + EditionUpgrade = [System.Management.Automation.SwitchParameter] $true + InstallFolder = 'C:\Temp' + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When Reporting Services are installed using parameter SourceCredential' { + BeforeAll { + $mockLocalPath = 'C:\LocalPath' + + $mockShareCredentialUserName = 'COMPANY\SqlAdmin' + $mockShareCredentialPassword = 'dummyPassW0rd' + $mockShareCredential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList @( + $mockShareCredentialUserName, + ($mockShareCredentialPassword | ConvertTo-SecureString -AsPlainText -Force) + ) + + Mock -CommandName Invoke-InstallationMediaCopy -MockWith { + return $mockLocalPath + } + } + + BeforeEach { + $mockSetTargetResourceParameters['ProductKey'] = $mockProductKey + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + PID = '1FAKE-2FAKE-3FAKE-4FAKE-5FAKE' + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks' { + $mockSetTargetResourceParameters['SourceCredential'] = $mockShareCredential + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + Assert-MockCalled -CommandName Invoke-InstallationMediaCopy -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + # Have to build the correct path (local path + executable). + $FilePath -eq (Join-Path -Path $mockLocalPath -ChildPath (Split-Path -Path $mockSetTargetResourceParameters.SourcePath -Leaf)) + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When the Reporting Services installation is successful with exit code 3010' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 3010 + } + } + + It 'Should call the correct mocks, and set $global:DSCMachineStatus to 1' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + # Should set the global DSCMachineStatus variable. + $global:DSCMachineStatus | Should -Be 1 + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + + Context 'When the Reporting Services installation is successful with exit code 3010, and called with parameter SuppressRestart' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + $mockSetTargetResourceParameters['SuppressRestart'] = $true + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + NoRestart = [System.Management.Automation.SwitchParameter] $true + } + } + + It 'Should call the correct mocks' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + + # Should not set the global DSCMachineStatus variable. + $global:DSCMachineStatus | Should -Be 0 + } + } + } + + Context 'When the Reporting Services installation is successful and ForceRestart is used' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + $mockSetTargetResourceParameters['ForceRestart'] = $true + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + } + + Mock -CommandName Test-PendingRestart + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks, and set $global:DSCMachineStatus to 1' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + # Should set the global DSCMachineStatus variable. + $global:DSCMachineStatus | Should -Be 1 + + Assert-MockCalled -CommandName Test-PendingRestart -Exactly -Times 0 -Scope 'It' + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When the Reporting Services installation is successful, and there are a pending restart' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + } + + Mock -CommandName Test-PendingRestart -MockWith { + return $true + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 0 + } + } + + It 'Should call the correct mocks, and set $global:DSCMachineStatus to 1' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Not -Throw + + # Should set the global DSCMachineStatus variable. + $global:DSCMachineStatus | Should -Be 1 + + Assert-MockCalled -CommandName Test-PendingRestart -Exactly -Times 1 -Scope 'It' + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + } + } + + Context 'When the Reporting Services installation fails' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + } + + Mock -CommandName Start-SqlSetupProcess -MockWith { + Test-SetupArgument -Argument $ArgumentList -ExpectedArgument $mockStartSqlSetupProcess_ExpectedArgumentList + + return 1 + } + } + + It 'Should throw the correct error message' { + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw $script:localizedData.SetupFailed + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + + # Should not set the global DSCMachineStatus variable. + $global:DSCMachineStatus | Should -Be 0 + } + + Context 'When the Reporting Services installation fails, and called with parameter LogPath' { + BeforeEach { + $mockSetTargetResourceParameters['Edition'] = 'Development' + $mockSetTargetResourceParameters['LogPath'] = 'TestDrive:\' + + $mockStartSqlSetupProcess_ExpectedArgumentList = @{ + Quiet = [System.Management.Automation.SwitchParameter] $true + IAcceptLicenseTerms = [System.Management.Automation.SwitchParameter] $true + Edition = 'Dev' + log = $mockSetTargetResourceParameters.LogPath + } + } + + It 'Should throw the correct error message' { + $errorMessage = $script:localizedData.SetupFailedWithLog -f $mockSetTargetResourceParameters.LogPath + + { Set-TargetResource @mockSetTargetResourceParameters } | Should -Throw $errorMessage + + Assert-MockCalled -CommandName Start-SqlSetupProcess -ParameterFilter { + $FilePath -eq $mockSetTargetResourceParameters.SourcePath + } -Exactly -Times 1 -Scope 'It' + + # Should not set the global DSCMachineStatus variable. + $global:DSCMachineStatus | Should -Be 0 + } + } + } + } + } + + Describe "MSFT_SqlRSSetup\Convert-EditionName" -Tag 'Helper' { + Context 'When converting edition names' { + $testCases = @( + @{ + InputName = 'Development' + OutputName = 'Dev' + } + @{ + InputName = 'Evaluation' + OutputName = 'Eval' + } + @{ + InputName = 'ExpressAdvanced' + OutputName = 'ExprAdv' + } + @{ + InputName = 'Dev' + OutputName = 'Development' + } + @{ + InputName = 'Eval' + OutputName = 'Evaluation' + } + @{ + InputName = 'ExprAdv' + OutputName = 'ExpressAdvanced' + } + ) + + It 'Should return the value when converting from value ' -TestCases $testCases { + param + ( + [Parameter()] + [System.String] + $InputName, + + [Parameter()] + [System.String] + $OutputName + ) + + Convert-EditionName -Name $InputName | Should -Be $OutputName + } + } + } + + Describe "MSFT_SqlRSSetup\Get-FileProductVersion" -Tag 'Helper' { + Context 'When converting edition names' { + $mockProductVersion = '14.0.0.0' + + BeforeAll { + Mock -CommandName Get-Item -MockWith { + return @{ + VersionInfo = @{ + ProductVersion = $mockProductVersion + } + } + } + } + + It 'Should return the correct product version' { + Get-FileProductVersion -Path 'TestDrive:\MockExecutable.exe' | Should -Be $mockProductVersion + } + } + } + } +} +finally +{ + Invoke-TestCleanup +} + diff --git a/Tests/Unit/MSFT_SqlScript.Tests.ps1 b/Tests/Unit/MSFT_SqlScript.Tests.ps1 index ebc6dced2..cb560a952 100644 --- a/Tests/Unit/MSFT_SqlScript.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlScript.Tests.ps1 @@ -33,8 +33,6 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR } Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1') -Force -Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath 'SqlServerDscHelper.psm1') - $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` @@ -44,8 +42,8 @@ $TestEnvironment = Initialize-TestEnvironment ` #endregion HEADER function Invoke-TestSetup { - Add-Type -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SqlPowerShellSqlExecutionException.cs') - Import-Module -Name (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Global -Force + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SqlPowerShellSqlExecutionException.cs') } function Invoke-TestCleanup { @@ -59,192 +57,143 @@ try Invoke-TestSetup InModuleScope $script:dscResourceName { - InModuleScope 'SqlServerDscHelper' { - $script:dscModuleName = 'SqlServerDsc' - $resourceName = 'MSFT_SqlScript' - - $testParameters = @{ - ServerInstance = $env:COMPUTERNAME - SetFilePath = "set.sql" - GetFilePath = "get.sql" - TestFilePath = "test.sql" - } + Describe 'MSFT_SqlScript\Get-TargetResource' { + BeforeAll { + $testParameters = @{ + ServerInstance = $env:COMPUTERNAME + SetFilePath = "set.sql" + GetFilePath = "get.sql" + TestFilePath = "test.sql" + } - $testParametersTimeout = @{ - ServerInstance = $env:COMPUTERNAME - SetFilePath = "set-timeout.sql" - GetFilePath = "get-timeout.sql" - TestFilePath = "test-timeout.sql" - QueryTimeout = 30 + $testParametersTimeout = @{ + ServerInstance = $env:COMPUTERNAME + SetFilePath = "set-timeout.sql" + GetFilePath = "get-timeout.sql" + TestFilePath = "test-timeout.sql" + QueryTimeout = 30 + } } - Describe "$resourceName\Get-TargetResource" { - - Context 'Get-TargetResource fails to import SQLPS module' { - $throwMessage = "Failed to import SQLPS module." + Context 'When Get-TargetResource returns script results successfully' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' + } - Mock -CommandName Import-SQLPSModule -MockWith { - throw $throwMessage - } + It 'Should return the expected results' { + $result = Get-TargetResource @testParameters - It 'Should throw the correct error from Import-Module' { - { Get-TargetResource @testParameters } | Should -Throw $throwMessage - } + $result.ServerInstance | Should -Be $testParameters.ServerInstance + $result.SetFilePath | Should -Be $testParameters.SetFilePath + $result.GetFilePath | Should -Be $testParameters.GetFilePath + $result.TestFilePath | Should -Be $testParameters.TestFilePath + $result | Should -BeOfType [System.Collections.Hashtable] } + } - Context 'Get-TargetResource returns script results successfully' { - Mock -CommandName Import-SQLPSModule - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } - - It 'Should return the expected results' { - $result = Get-TargetResource @testParameters - - $result.ServerInstance | Should -Be $testParameters.ServerInstance - $result.SetFilePath | Should -Be $testParameters.SetFilePath - $result.GetFilePath | Should -Be $testParameters.GetFilePath - $result.TestFilePath | Should -Be $testParameters.TestFilePath - $result | Should -BeOfType Hashtable - } + Context 'When Get-TargetResource returns script results successfully with query timeout' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' } - Context 'Get-TargetResource returns script results successfully with query timeout' { - Mock -CommandName Import-SQLPSModule - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } - - It 'Should return the expected results' { - $result = Get-TargetResource @testParametersTimeout - $result.ServerInstance | Should -Be $testParametersTimeout.ServerInstance - $result.SetFilePath | Should -Be $testParametersTimeout.SetFilePath - $result.GetFilePath | Should -Be $testParametersTimeout.GetFilePath - $result.TestFilePath | Should -Be $testParametersTimeout.TestFilePath - $result | Should -BeOfType Hashtable - } + It 'Should return the expected results' { + $result = Get-TargetResource @testParametersTimeout + $result.ServerInstance | Should -Be $testParametersTimeout.ServerInstance + $result.SetFilePath | Should -Be $testParametersTimeout.SetFilePath + $result.GetFilePath | Should -Be $testParametersTimeout.GetFilePath + $result.TestFilePath | Should -Be $testParametersTimeout.TestFilePath + $result | Should -BeOfType [System.Collections.Hashtable] } + } - Context 'Get-TargetResource throws an error when running the script in the GetFilePath parameter' { - $errorMessage = "Failed to run SQL Script" + Context 'When Get-TargetResource throws an error when running the script in the GetFilePath parameter' { + $errorMessage = "Failed to run SQL Script" - Mock -CommandName Import-SQLPSModule - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw $errorMessage - } + Mock -CommandName Invoke-SqlScript -MockWith { + throw $errorMessage + } - It 'Should throw the correct error from Invoke-Sqlcmd' { - { Get-TargetResource @testParameters } | Should -Throw $errorMessage - } + It 'Should throw the correct error from Invoke-Sqlcmd' { + { Get-TargetResource @testParameters } | Should -Throw $errorMessage } } + } - Describe "$resourceName\Set-TargetResource" { - - Context 'Set-TargetResource fails to import SQLPS module' { - $throwMessage = "Failed to import SQLPS module." - - Mock -CommandName Import-SQLPSModule -MockWith { throw $throwMessage } - - It 'Should throw the correct error from Import-Module' { - { Set-TargetResource @testParameters } | Should -Throw $throwMessage - } + Describe 'MSFT_SqlScript\Set-TargetResource' { + Context 'When Set-TargetResource runs script without issue' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' } - Context 'Set-TargetResource runs script without issue' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } - - It 'Should return the expected results' { - $result = Set-TargetResource @testParameters - $result | Should -Be '' - } + It 'Should return the expected results' { + $result = Set-TargetResource @testParameters + $result | Should -Be '' } + } - Context 'Set-TargetResource runs script without issue using timeout' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } - - It 'Should return the expected results' { - $result = Set-TargetResource @testParametersTimeout - $result | Should -Be '' - } + Context 'When Set-TargetResource runs script without issue using timeout' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' } - Context 'Set-TargetResource throws an error when running the script in the SetFilePath parameter' { - $errorMessage = "Failed to run SQL Script" - - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw $errorMessage - } - - It 'Should throw the correct error from Invoke-Sqlcmd' { - { Set-TargetResource @testParameters } | Should -Throw $errorMessage - } + It 'Should return the expected results' { + $result = Set-TargetResource @testParametersTimeout + $result | Should -Be '' } } - Describe "$resourceName\Test-TargetResource" { - Context 'Test-TargetResource fails to import SQLPS module' { - $throwMessage = 'Failed to import SQLPS module.' + Context 'When Set-TargetResource throws an error when running the script in the SetFilePath parameter' { + $errorMessage = "Failed to run SQL Script" - Mock -CommandName Import-SQLPSModule -MockWith { - throw $throwMessage - } + Mock -CommandName Invoke-SqlScript -MockWith { + throw $errorMessage + } - It 'Should throw the correct error from Import-Module' { - { Set-TargetResource @testParameters } | Should -Throw $throwMessage - } + It 'Should throw the correct error from Invoke-Sqlcmd' { + { Set-TargetResource @testParameters } | Should -Throw $errorMessage } + } + } - Context 'Test-TargetResource runs script without issue' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith {} + Describe 'MSFT_SqlScript\Test-TargetResource' { + Context 'When Test-TargetResource runs script without issue' { + Mock -CommandName Invoke-SqlScript - It 'Should return true' { - $result = Test-TargetResource @testParameters - $result | Should -Be $true - } + It 'Should return true' { + $result = Test-TargetResource @testParameters + $result | Should -Be $true } + } - Context 'Test-TargetResource runs script without issue with timeout' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith {} + Context 'When Test-TargetResource runs script without issue with timeout' { + Mock -CommandName Invoke-SqlScript - It 'Should return true' { - $result = Test-TargetResource @testParametersTimeout - $result | Should -Be $true - } + It 'Should return true' { + $result = Test-TargetResource @testParametersTimeout + $result | Should -Be $true } + } - Context 'Test-TargetResource throws the exception SqlPowerShellSqlExecutionException when running the script in the TestFilePath parameter' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw New-Object -TypeName Microsoft.SqlServer.Management.PowerShell.SqlPowerShellSqlExecutionException - } + Context 'When Test-TargetResource throws the exception SqlPowerShellSqlExecutionException when running the script in the TestFilePath parameter' { + Mock -CommandName Invoke-SqlScript -MockWith { + throw New-Object -TypeName Microsoft.SqlServer.Management.PowerShell.SqlPowerShellSqlExecutionException + } - It 'Should return false' { - $result = Test-TargetResource @testParameters - $result | Should -Be $false - } + It 'Should return false' { + $result = Test-TargetResource @testParameters + $result | Should -Be $false } + } - Context 'Test-TargetResource throws an unexpected error when running the script in the TestFilePath parameter' { - $errorMessage = "Failed to run SQL Script" + Context 'When Test-TargetResource throws an unexpected error when running the script in the TestFilePath parameter' { + $errorMessage = "Failed to run SQL Script" - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw $errorMessage - } + Mock -CommandName Invoke-SqlScript -MockWith { + throw $errorMessage + } - It 'Should throw the correct error from Invoke-Sqlcmd' { - { Test-TargetResource @testParameters } | Should -Throw $errorMessage - } + It 'Should throw the correct error from Invoke-Sqlcmd' { + { Test-TargetResource @testParameters } | Should -Throw $errorMessage } } } diff --git a/Tests/Unit/MSFT_SqlScriptQuery.Tests.ps1 b/Tests/Unit/MSFT_SqlScriptQuery.Tests.ps1 index d8b49d689..9eeb4d83f 100644 --- a/Tests/Unit/MSFT_SqlScriptQuery.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlScriptQuery.Tests.ps1 @@ -33,7 +33,6 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR } Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1') -Force -Import-Module -Name (Join-Path -Path $script:moduleRoot -ChildPath 'SqlServerDscHelper.psm1') $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` @@ -43,8 +42,11 @@ $TestEnvironment = Initialize-TestEnvironment ` #endregion HEADER function Invoke-TestSetup { - Add-Type -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SqlPowerShellSqlExecutionException.cs') - Import-Module -Name (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Global -Force + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SqlPowerShellSqlExecutionException.cs') + + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup { @@ -58,192 +60,143 @@ try Invoke-TestSetup InModuleScope $script:dscResourceName { - InModuleScope 'SqlServerDscHelper' { - $script:dscModuleName = 'SqlServerDsc' - $resourceName = 'MSFT_SqlScriptQuery' - - $testParameters = @{ - ServerInstance = $env:COMPUTERNAME - GetQuery = "GetQuery;" - TestQuery = "TestQuery;" - SetQuery = "SetQuery;" - } - - $testParametersTimeout = @{ - ServerInstance = $env:COMPUTERNAME - GetQuery = "GetQuery;" - TestQuery = "TestQuery;" - SetQuery = "SetQuery;" - QueryTimeout = 30 - } - - Describe "$resourceName\Get-TargetResource" { - - Context 'Get-TargetResource fails to import SQLPS module' { - $throwMessage = "Failed to import SQLPS module." - - Mock -CommandName Import-SQLPSModule -MockWith { - throw $throwMessage - } - - It 'Should throw the correct error from Import-Module' { - { Get-TargetResource @testParameters } | Should -Throw $throwMessage - } + Describe 'MSFT_SqlScriptQuery\Get-TargetResource' { + BeforeAll { + $testParameters = @{ + ServerInstance = $env:COMPUTERNAME + GetQuery = "GetQuery;" + TestQuery = "TestQuery;" + SetQuery = "SetQuery;" } - Context 'Get-TargetResource returns script results successfully' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } - - It 'Should return the expected results' { - $result = Get-TargetResource @testParameters - - $result.ServerInstance | Should -Be $testParameters.ServerInstance - $result.GetQuery | Should -Be $testParameters.GetQuery - $result.SetQuery | Should -Be $testParameters.SetQuery - $result.TestQuery | Should -Be $testParameters.TestQuery - $result | Should BeOfType Hashtable - } + $testParametersTimeout = @{ + ServerInstance = $env:COMPUTERNAME + GetQuery = "GetQuery;" + TestQuery = "TestQuery;" + SetQuery = "SetQuery;" + QueryTimeout = 30 } + } - Context 'Get-TargetResource returns script results successfully with query timeout' { - Mock -CommandName Import-SQLPSModule - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } - - It 'Should return the expected results' { - $result = Get-TargetResource @testParametersTimeout - $result.ServerInstance | Should -Be $testParametersTimeout.ServerInstance - $result.GetQuery | Should -Be $testParameters.GetQuery - $result.SetQuery | Should -Be $testParameters.SetQuery - $result.TestQuery | Should -Be $testParameters.TestQuery - $result | Should BeOfType Hashtable - } + Context 'Get-TargetResource returns script results successfully' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' } - Context 'Get-TargetResource throws an error when running the script in the GetQuery parameter' { - $errorMessage = "Failed to run SQL Script" - - Mock -CommandName Import-SQLPSModule - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw $errorMessage - } + It 'Should return the expected results' { + $result = Get-TargetResource @testParameters - It 'Should throw the correct error from Invoke-Sqlcmd' { - { Get-TargetResource @testParameters } | Should -Throw $errorMessage - } + $result.ServerInstance | Should -Be $testParameters.ServerInstance + $result.GetQuery | Should -Be $testParameters.GetQuery + $result.SetQuery | Should -Be $testParameters.SetQuery + $result.TestQuery | Should -Be $testParameters.TestQuery + $result | Should BeOfType Hashtable } } - Describe "$resourceName\Set-TargetResource" { - - Context 'Set-TargetResource fails to import SQLPS module' { - $throwMessage = "Failed to import SQLPS module." - - Mock -CommandName Import-SQLPSModule -MockWith { throw $throwMessage } + Context 'Get-TargetResource returns script results successfully with query timeout' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' + } - It 'Should throw the correct error from Import-Module' { - { Set-TargetResource @testParameters } | Should -Throw $throwMessage - } + It 'Should return the expected results' { + $result = Get-TargetResource @testParametersTimeout + $result.ServerInstance | Should -Be $testParametersTimeout.ServerInstance + $result.GetQuery | Should -Be $testParameters.GetQuery + $result.SetQuery | Should -Be $testParameters.SetQuery + $result.TestQuery | Should -Be $testParameters.TestQuery + $result | Should BeOfType Hashtable } + } - Context 'Set-TargetResource runs script without issue' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } + Context 'Get-TargetResource throws an error when running the script in the GetQuery parameter' { + $errorMessage = "Failed to run SQL Script" - It 'Should return the expected results' { - $result = Set-TargetResource @testParameters - $result | Should -Be '' - } + Mock -CommandName Invoke-SqlScript -MockWith { + throw $errorMessage } - Context 'Set-TargetResource runs script without issue using timeout' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - return '' - } + It 'Should throw the correct error from Invoke-Sqlcmd' { + { Get-TargetResource @testParameters } | Should -Throw $errorMessage + } + } + } - It 'Should return the expected results' { - $result = Set-TargetResource @testParametersTimeout - $result | Should -Be '' - } + Describe 'MSFT_SqlScriptQuery\Set-TargetResource' { + Context 'Set-TargetResource runs script without issue' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' } - Context 'Set-TargetResource throws an error when running the script in the SetFilePath parameter' { - $errorMessage = "Failed to run SQL Script" + It 'Should return the expected results' { + $result = Set-TargetResource @testParameters + $result | Should -Be '' + } + } - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw $errorMessage - } + Context 'Set-TargetResource runs script without issue using timeout' { + Mock -CommandName Invoke-SqlScript -MockWith { + return '' + } - It 'Should throw the correct error from Invoke-Sqlcmd' { - { Set-TargetResource @testParameters } | Should -Throw $errorMessage - } + It 'Should return the expected results' { + $result = Set-TargetResource @testParametersTimeout + $result | Should -Be '' } } - Describe "$resourceName\Test-TargetResource" { - Context 'Test-TargetResource fails to import SQLPS module' { - $throwMessage = 'Failed to import SQLPS module.' + Context 'Set-TargetResource throws an error when running the script in the SetFilePath parameter' { + $errorMessage = "Failed to run SQL Script" - Mock -CommandName Import-SQLPSModule -MockWith { - throw $throwMessage - } + Mock -CommandName Invoke-SqlScript -MockWith { + throw $errorMessage + } - It 'Should throw the correct error from Import-Module' { - { Set-TargetResource @testParameters } | Should -Throw $throwMessage - } + It 'Should throw the correct error from Invoke-Sqlcmd' { + { Set-TargetResource @testParameters } | Should -Throw $errorMessage } + } + } - Context 'Test-TargetResource runs script without issue' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith {} + Describe 'MSFT_SqlScriptQuery\Test-TargetResource' { + Context 'Test-TargetResource runs script without issue' { + Mock -CommandName Invoke-SqlScript - It 'Should return true' { - $result = Test-TargetResource @testParameters - $result | Should -Be $true - } + It 'Should return true' { + $result = Test-TargetResource @testParameters + $result | Should -Be $true } + } - Context 'Test-TargetResource runs script without issue with timeout' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith {} + Context 'Test-TargetResource runs script without issue with timeout' { + Mock -CommandName Invoke-SqlScript - It 'Should return true' { - $result = Test-TargetResource @testParametersTimeout - $result | Should -Be $true - } + It 'Should return true' { + $result = Test-TargetResource @testParametersTimeout + $result | Should -Be $true } + } - Context 'Test-TargetResource throws the exception SqlPowerShellSqlExecutionException when running the script in the TestFilePath parameter' { - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw New-Object -TypeName Microsoft.SqlServer.Management.PowerShell.SqlPowerShellSqlExecutionException - } + Context 'Test-TargetResource throws the exception SqlPowerShellSqlExecutionException when running the script in the TestFilePath parameter' { + Mock -CommandName Invoke-SqlScript -MockWith { + throw New-Object -TypeName Microsoft.SqlServer.Management.PowerShell.SqlPowerShellSqlExecutionException + } - It 'Should return false' { - $result = Test-TargetResource @testParameters - $result | Should -Be $false - } + It 'Should return false' { + $result = Test-TargetResource @testParameters + $result | Should -Be $false } + } - Context 'Test-TargetResource throws an unexpected error when running the script in the TestFilePath parameter' { - $errorMessage = "Failed to run SQL Script" + Context 'Test-TargetResource throws an unexpected error when running the script in the TestFilePath parameter' { + $errorMessage = "Failed to run SQL Script" - Mock -CommandName Import-SQLPSModule -MockWith {} - Mock -CommandName Invoke-Sqlcmd -MockWith { - throw $errorMessage - } + Mock -CommandName Invoke-SqlScript -MockWith { + throw $errorMessage + } - It 'Should throw the correct error from Invoke-Sqlcmd' { - { Test-TargetResource @testParameters } | Should -Throw $errorMessage - } + It 'Should throw the correct error from Invoke-Sqlcmd' { + { Test-TargetResource @testParameters } | Should -Throw $errorMessage } } } diff --git a/Tests/Unit/MSFT_SqlServerEndpoint.Tests.ps1 b/Tests/Unit/MSFT_SqlServerEndpoint.Tests.ps1 index da3b6a81a..86843d44b 100644 --- a/Tests/Unit/MSFT_SqlServerEndpoint.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerEndpoint.Tests.ps1 @@ -39,7 +39,7 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { # Loading mocked classes - Add-Type -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests\Unit\Stubs\SMO.cs') + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup { diff --git a/Tests/Unit/MSFT_SqlServerEndpointPermission.Tests.ps1 b/Tests/Unit/MSFT_SqlServerEndpointPermission.Tests.ps1 index 000bbcca4..3202edffb 100644 --- a/Tests/Unit/MSFT_SqlServerEndpointPermission.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerEndpointPermission.Tests.ps1 @@ -40,7 +40,7 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { # Loading mocked classes - Add-Type -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests\Unit\Stubs\SMO.cs') + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServerEndpointState.Tests.ps1 b/Tests/Unit/MSFT_SqlServerEndpointState.Tests.ps1 index de1eafc9d..7f90c553e 100644 --- a/Tests/Unit/MSFT_SqlServerEndpointState.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerEndpointState.Tests.ps1 @@ -39,8 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - # Loading stub cmdlets - Import-Module -Name ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SQLPSStub.psm1 ) -Force -Global + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServerLogin.Tests.ps1 b/Tests/Unit/MSFT_SqlServerLogin.Tests.ps1 index a45fb9f61..218f6ea5e 100644 --- a/Tests/Unit/MSFT_SqlServerLogin.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerLogin.Tests.ps1 @@ -43,8 +43,11 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - Import-Module -Name ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SQLPSStub.psm1 ) -Force - Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') + + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServerMaxDop.Tests.ps1 b/Tests/Unit/MSFT_SqlServerMaxDop.Tests.ps1 index 68a947c48..bdc09f063 100644 --- a/Tests/Unit/MSFT_SqlServerMaxDop.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerMaxDop.Tests.ps1 @@ -30,9 +30,6 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR Import-Module (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1') -Force -# Loading mocked classes -Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) - $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` -DSCResourceName $script:dscResourceName ` @@ -42,6 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 b/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 index 2b46e80c1..7b88588a3 100644 --- a/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerMemory.Tests.ps1 @@ -30,9 +30,6 @@ if ( (-not (Test-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'DSCR Import-Module (Join-Path -Path $script:moduleRoot -ChildPath 'DSCResource.Tests\TestHelper.psm1') -Force -# Loading mocked classes -Add-Type -Path ( Join-Path -Path ( Join-Path -Path $PSScriptRoot -ChildPath Stubs ) -ChildPath SMO.cs ) - $TestEnvironment = Initialize-TestEnvironment ` -DSCModuleName $script:dscModuleName ` -DSCResourceName $script:dscResourceName ` @@ -42,6 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 b/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 index 4ed04687e..92007b2c8 100644 --- a/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerPermission.Tests.ps1 @@ -40,7 +40,7 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { # Loading mocked classes - Add-Type -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests\Unit\Stubs\SMO.cs') + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServerReplication.Tests.ps1 b/Tests/Unit/MSFT_SqlServerReplication.Tests.ps1 index 40a51ba12..cc7080d2f 100644 --- a/Tests/Unit/MSFT_SqlServerReplication.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerReplication.Tests.ps1 @@ -38,11 +38,10 @@ $TestEnvironment = Initialize-TestEnvironment ` -TestType Unit #endregion HEADER -# Begin Testing + try { InModuleScope $script:dscResourceName { - Describe 'Helper functions' { Context 'Get-SqlServerMajorVersion' { diff --git a/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 b/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 index 30dbcc64d..bf1d90e51 100644 --- a/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServerSecureConnection.Tests.ps1 @@ -39,7 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - Import-Module -Name (Join-Path -Path (Join-Path -Path (Join-Path -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests') -ChildPath 'Unit') -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Global -Force + # Importing SQLPS stubs + Import-Module -Name (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SQLPSStub.psm1') -Force -Global } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlServiceAccount.Tests.ps1 b/Tests/Unit/MSFT_SqlServiceAccount.Tests.ps1 index 811b68143..a7d4eaf45 100644 --- a/Tests/Unit/MSFT_SqlServiceAccount.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlServiceAccount.Tests.ps1 @@ -39,8 +39,8 @@ $TestEnvironment = Initialize-TestEnvironment ` function Invoke-TestSetup { - # Compile the SMO stubs for use by the unit tests. - Add-Type -Path (Join-Path -Path $script:moduleRoot -ChildPath 'Tests\Unit\Stubs\SMO.cs') + # Loading mocked classes + Add-Type -Path (Join-Path -Path (Join-Path -Path $PSScriptRoot -ChildPath 'Stubs') -ChildPath 'SMO.cs') } function Invoke-TestCleanup diff --git a/Tests/Unit/MSFT_SqlSetup.Tests.ps1 b/Tests/Unit/MSFT_SqlSetup.Tests.ps1 index 84c79d68a..fdaaef18f 100644 --- a/Tests/Unit/MSFT_SqlSetup.Tests.ps1 +++ b/Tests/Unit/MSFT_SqlSetup.Tests.ps1 @@ -96,26 +96,36 @@ try $value = ($Matches[2] -replace '" "',' ') -replace '"','' } - $null = $argumentHashTable.Add($key, $value) + $argumentHashTable.Add($key, $value) } } $actualValues = $argumentHashTable.Clone() + # Limit the output in the console when everything is fine. + if ($actualValues.Count -ne $ExpectedArgument.Count) + { + Write-Warning -Message 'Verified the setup argument count (expected vs actual)' + Write-Warning -Message ('Expected: {0}' -f ($ExpectedArgument.Keys -join ',')) + Write-Warning -Message ('Actual: {0}' -f ($actualValues.Keys -join ',')) + } + # Start by checking whether we have the same number of parameters - Write-Verbose 'Verifying setup argument count (expected vs actual)' -Verbose - Write-Verbose -Message ('Expected: {0}' -f ($ExpectedArgument.Keys -join ',') ) -Verbose - Write-Verbose -Message ('Actual: {0}' -f ($actualValues.Keys -join ',')) -Verbose + $actualValues.Count | Should -Be $ExpectedArgument.Count ` + -Because ('the expected arguments was: {0}' -f ($ExpectedArgument.Keys -join ',')) - $actualValues.Count | Should -Be $ExpectedArgument.Count + Write-Verbose -Message 'Verified actual setup argument values against expected setup argument values' -Verbose - Write-Verbose 'Verifying actual setup arguments against expected setup arguments' -Verbose foreach ($argumentKey in $ExpectedArgument.Keys) { - $argumentKeyName = $argumentHashTable.GetEnumerator() | Where-Object -FilterScript { $_.Name -eq $argumentKey } | Select-Object -ExpandProperty Name + $argumentKeyName = $actualValues.GetEnumerator() | + Where-Object -FilterScript { + $_.Name -eq $argumentKey + } | Select-Object -ExpandProperty 'Name' + $argumentKeyName | Should -Be $argumentKey - $argumentValue = $argumentHashTable.$argumentKey + $argumentValue = $actualValues.$argumentKey $argumentValue | Should -Be $ExpectedArgument.$argumentKey } } @@ -694,59 +704,10 @@ try ) } - $mockRobocopyExecutableName = 'Robocopy.exe' - $mockRobocopyExecutableVersionWithoutUnbufferedIO = '6.2.9200.00000' - $mockRobocopyExecutableVersionWithUnbufferedIO = '6.3.9600.16384' - $mockRobocopyExecutableVersion = '' # Set dynamically during runtime - $mockRobocopyArgumentSilent = '/njh /njs /ndl /nc /ns /nfl' - $mockRobocopyArgumentCopySubDirectoriesIncludingEmpty = '/e' - $mockRobocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource = '/purge' - $mockRobocopyArgumentUseUnbufferedIO = '/J' - $mockRobocopyArgumentSourcePath = 'C:\Source\SQL2016' - $mockRobocopyArgumentDestinationPath = 'D:\Temp' - $mockRobocopyArgumentSourcePathWithSpaces = 'C:\Source\SQL2016 STD SP1' - $mockRobocopyArgumentDestinationPathWithSpaces = 'D:\Temp\DSC SQL2016' - - $mockGetCommand = { - return @( - ( - New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'Name' -Value $mockRobocopyExecutableName -PassThru | - Add-Member -MemberType ScriptProperty -Name FileVersionInfo -Value { - return @( ( New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'ProductVersion' -Value $mockRobocopyExecutableVersion -PassThru -Force - ) ) - } -PassThru -Force - ) - ) - } - - $mockStartSqlSetupProcessExpectedArgument = '' # Set dynamically during runtime - $mockStartSqlSetupProcessExitCode = 0 # Set dynamically during runtime - - $mockStartSqlSetupProcess_Robocopy = { - if ( $ArgumentList -cne $mockStartSqlSetupProcessExpectedArgument ) - { - throw "Expected arguments was not the same as the arguments in the function call.`nExpected: '$mockStartSqlSetupProcessExpectedArgument' `n But was: '$ArgumentList'" - } - - return New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'ExitCode' -Value 0 -PassThru -Force - } - - $mockStartSqlSetupProcess_Robocopy_WithExitCode = { - return New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'ExitCode' -Value $mockStartSqlSetupProcessExitCode -PassThru -Force - } - $mockSourcePathUNCWithoutLeaf = '\\server\share' $mockSourcePathGuid = 'cc719562-0f46-4a16-8605-9f8a47c70402' - $mockNewGuid = { - return New-Object -TypeName Object | - Add-Member -MemberType NoteProperty -Name 'Guid' -Value $mockSourcePathGuid -PassThru -Force - } - $mockGetTemporaryFolder = { + $mockNewTemporaryFolder = { return $mockSourcePathUNC } @@ -967,11 +928,11 @@ try These are written with both lower-case and upper-case to make sure we support that. The feature list must be written in the order it is returned by the function Get-TargetResource. #> - $defaultFeatures = 'SQLEngine,Replication,Dq,Dqc,FullText,Rs,As,Is,Bol,Conn,Bc,Sdk,Mds,Ssms,Adv_Ssms' + $mockDefaultFeatures = 'SQLEngine,Replication,Dq,Dqc,FullText,Rs,As,Is,Bol,Conn,Bc,Sdk,Mds,Ssms,Adv_Ssms' # Default parameters that are used for the It-blocks $mockDefaultParameters = @{ - Features = $defaultFeatures + Features = $mockDefaultFeatures } $MockSqlSvcStartupType = 'Automatic' @@ -987,7 +948,7 @@ try Features = 'SQLEngine' } - Describe "SqlSetup\Get-TargetResource" -Tag 'Get' { + Describe 'SqlSetup\Get-TargetResource' -Tag 'Get' { #region Setting up TestDrive:\ # Local path to TestDrive:\ @@ -1101,8 +1062,8 @@ try } -MockWith $mockEmptyHashtable -Verifiable } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Connect-UncPath -Verifiable + Mock -CommandName Disconnect-UncPath -Verifiable Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable Mock -CommandName Get-ItemProperty -ParameterFilter { @@ -1130,8 +1091,8 @@ try $result = Get-TargetResource @testParameters $result.InstanceName | Should -Be $testParameters.InstanceName - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It @@ -1219,8 +1180,8 @@ try ) } -Verifiable - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Connect-UncPath -Verifiable + Mock -CommandName Disconnect-UncPath -Verifiable Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable #region Mock Get-CimInstance @@ -1281,8 +1242,8 @@ try $result = Get-TargetResource @testParameters $result.InstanceName | Should -Be $testParameters.InstanceName - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It @@ -1442,8 +1403,8 @@ try } -MockWith $mockEmptyHashtable -Verifiable } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Connect-UncPath -Verifiable + Mock -CommandName Disconnect-UncPath -Verifiable Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable Mock -CommandName Get-ItemProperty -ParameterFilter { @@ -1471,8 +1432,8 @@ try $result = Get-TargetResource @testParameters $result.InstanceName | Should -Be $testParameters.InstanceName - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It @@ -1577,8 +1538,8 @@ try } -MockWith $mockGetItemProperty_UninstallProducts2014 -Verifiable } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Connect-UncPath -Verifiable + Mock -CommandName Disconnect-UncPath -Verifiable Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable #region Mock Get-CimInstance @@ -1685,8 +1646,8 @@ try } -MockWith $mockGetItemProperty_UninstallProducts2014 -Verifiable } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Connect-UncPath -Verifiable + Mock -CommandName Disconnect-UncPath -Verifiable Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable #region Mock Get-CimInstance @@ -1747,8 +1708,8 @@ try $result = Get-TargetResource @testParameters $result.InstanceName | Should -Be $testParameters.InstanceName - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 0 -Scope It + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It @@ -1931,8 +1892,8 @@ try } -MockWith $mockGetItemProperty_UninstallProducts2014 -Verifiable } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable + Mock -CommandName Connect-UncPath -Verifiable + Mock -CommandName Disconnect-UncPath -Verifiable Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable #region Mock Get-CimInstance @@ -1993,8 +1954,8 @@ try $result = Get-TargetResource @testParameters $result.InstanceName | Should -Be $testParameters.InstanceName - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Connect-UncPath -Exactly -Times 1 -Scope It + Assert-MockCalled -CommandName Disconnect-UncPath -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It @@ -2555,621 +2516,197 @@ try } Describe "SqlSetup\Test-TargetResource" -Tag 'Test' { - #region Setting up TestDrive:\ - - # Local path to TestDrive:\ - $mockSourcePath = $TestDrive.FullName - - # UNC path to TestDrive:\ - $testDrive_DriveShare = (Split-Path -Path $mockSourcePath -Qualifier) -replace ':','$' - $mockSourcePathUNC = Join-Path -Path "\\localhost\$testDrive_DriveShare" -ChildPath (Split-Path -Path $mockSourcePath -NoQualifier) - - #endregion Setting up TestDrive:\ - - BeforeAll { - # General mocks - Mock -CommandName Get-SqlMajorVersion -MockWith $mockGetSqlMajorVersion -Verifiable - Mock -CommandName Connect-SQL -MockWith $mockConnectSQL -Verifiable - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName -or $Name -eq $mockNamedInstance_InstanceName) - } -MockWith $mockGetItemProperty_SQL -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - ( - $Path -eq "HKLM:\SYSTEM\CurrentControlSet\Services\$mockDefaultInstance_AnalysisServiceName" -or - $Path -eq "HKLM:\SYSTEM\CurrentControlSet\Services\$mockNamedInstance_AnalysisServiceName" - ) -and - $Name -eq 'ImagePath' - } -MockWith $mockGetItemProperty_ServicesAnalysis -Verifiable - - # Mocking SharedDirectory - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UserData\S-1-5-18\Components\FEE2E540D20152D4597229B6CFBC0A69' - } -MockWith $mockGetItemProperty_SharedDirectory -Verifiable - - Mock -CommandName Get-Item -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UserData\S-1-5-18\Components\FEE2E540D20152D4597229B6CFBC0A69' - } -MockWith $mockGetItem_SharedDirectory -Verifiable - - # Mocking SharedWowDirectory - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UserData\S-1-5-18\Components\A79497A344129F64CA7D69C56F5DD8B4' - } -MockWith $mockGetItemProperty_SharedWowDirectory -Verifiable - - Mock -CommandName Get-Item -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UserData\S-1-5-18\Components\A79497A344129F64CA7D69C56F5DD8B4' - } -MockWith $mockGetItem_SharedWowDirectory -Verifiable - } - - BeforeEach { - Mock -CommandName Connect-SQLAnalysis -MockWith $mockConnectSQLAnalysis -Verifiable - } - - # For this test we only need to test one SQL Server version. Mocking SQL Server 2016 for the 'not in the desired state' test. - $mockSqlMajorVersion = 13 - - $mockDefaultInstance_InstanceId = "$($mockSqlDatabaseEngineName)$($mockSqlMajorVersion).$($mockDefaultInstance_InstanceName)" - - $mockSqlInstallPath = "C:\Program Files\Microsoft SQL Server\$($mockDefaultInstance_InstanceId)\MSSQL" - $mockDynamicSqlBackupPath = "C:\Program Files\Microsoft SQL Server\$($mockDefaultInstance_InstanceId)\MSSQL\Backup" - $mockDynamicSqlTempDatabasePath = '' - $mockDynamicSqlTempDatabaseLogPath = '' - $mockSqlDefaultDatabaseFilePath = "C:\Program Files\Microsoft SQL Server\$($mockDefaultInstance_InstanceId)\MSSQL\DATA\" - $mockSqlDefaultDatabaseLogPath = "C:\Program Files\Microsoft SQL Server\$($mockDefaultInstance_InstanceId)\MSSQL\DATA\" - - # This sets administrators dynamically in the mock Connect-SQLAnalysis. - $mockDynamicSqlAnalysisAdmins = $mockSqlAnalysisAdmins - Context 'When the system is not in the desired state' { - BeforeEach { - $testParameters = $mockDefaultParameters - $testParameters += @{ - InstanceName = $mockDefaultInstance_InstanceName - SourceCredential = $null - SourcePath = $mockSourcePath - } - - # Mock all SSMS products here to make sure we don't return any when testing SQL Server 2016 - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockGetItemProperty_UninstallProducts -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -MockWith $mockGetItemProperty_InstanceId_ConfigurationState -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\DQ\*" - } -MockWith $mockGetItemProperty_DQFeature -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" - } -MockWith $mockGetItemProperty_SqlVersion_ConfigurationState -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -MockWith $mockGetItemProperty_Setup -Verifiable - } - - It 'Should return that the desired state is absent when no products are installed' { - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable - - $result = Test-TargetResource @testParameters - $result | Should -Be $false - - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - } - - It 'Should return that the desired state is asbent when SSMS product is missing' { - Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable - - #region Mock Get-CimInstance - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_DatabaseService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_AgentService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_FullTextService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_ReportingService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" - } -MockWith $mockGetCimInstance_DefaultInstance_IntegrationService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_AnalysisService -Verifiable - - # If Get-CimInstance is used in any other way than those mocks with a ParameterFilter, then throw and error - Mock -CommandName Get-CimInstance -MockWith { - throw "Mock Get-CimInstance was called with unexpected parameters. ClassName=$ClassName, Filter=$Filter" - } -Verifiable - #endregion Mock Get-CimInstance - - # Change the default features for this test. - $testParameters.Features = 'SSMS' - - $result = Test-TargetResource @testParameters - $result | Should -Be $false - - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - #region Assert Get-CimInstance - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" - } -Exactly -Times 1 -Scope It - #endregion Assert Get-CimInstance - } - - It 'Should return that the desired state is asbent when ADV_SSMS product is missing' { - Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable - - #region Mock Get-CimInstance - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_DatabaseService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_AgentService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_FullTextService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_ReportingService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" - } -MockWith $mockGetCimInstance_DefaultInstance_IntegrationService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_AnalysisService -Verifiable - - # If Get-CimInstance is used in any other way than those mocks with a ParameterFilter, then throw and error - Mock -CommandName Get-CimInstance -MockWith { - throw "Mock Get-CimInstance was called with unexpected parameters. ClassName=$ClassName, Filter=$Filter" - } -Verifiable - #endregion Mock Get-CimInstance - - # Change the default features for this test. - $testParameters.Features = 'ADV_SSMS' - - $result = Test-TargetResource @testParameters - $result | Should -Be $false - - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - #region Assert Get-CimInstance - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" - } -Exactly -Times 1 -Scope It - #endregion Assert Get-CimInstance - } - - It 'Should return that the desired state is absent when a clustered instance cannot be found' { - $testClusterParameters = $testParameters.Clone() + Context 'When no features are installed' { + BeforeAll { + $testParameters = $mockDefaultParameters.Clone() + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + } - $testClusterParameters += @{ - FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName - FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress - FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } + } -Verifiable } - $result = Test-TargetResource @testClusterParameters + It 'Should return $false' { + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeFalse - $result | Should -Be $false + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + } } - # This is a test for regression testing of issue #432 - It 'Should return false if a SQL Server failover cluster is missing features' { - $mockCurrentInstanceName = $mockDefaultInstance_InstanceName + Context 'When a clustered instance cannot be found' { + BeforeAll { + $testParameters = $mockDefaultParameters.Clone() + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName + FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress + FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + } - Mock -CommandName Get-TargetResource -MockWith { - return @{ - Features = 'SQLENGINE' # Must be upper-case since Get-TargetResource returns upper-case. - FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName - FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress - FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + FailoverClusterGroupName = $null + FailoverClusterNetworkName = $null + FailoverClusterIPAddress = $null } - } -Verifiable + } -Verifiable + } - $testClusterParameters = $testParameters.Clone() - $testClusterParameters['Features'] = 'SQLEngine,AS' + It 'Should return $false' { + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeFalse - $testClusterParameters += @{ - FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName - FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress - FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' } - - $result = Test-TargetResource @testClusterParameters - $result | Should -Be $false } - } - # For this test we only need to test one SQL Server version. Mocking SQL Server 2014 for the 'in the desired state' test. - $mockSqlMajorVersion = 12 + Context 'When a SQL Server failover cluster is missing features' { + BeforeAll { + $testParameters = $mockDefaultParameters.Clone() + $testParameters['Features'] = 'SQLEngine,AS' + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName + FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress + FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + } - Context "When the system is in the desired state" { - BeforeEach { - $testParameters = $mockDefaultParameters - $testParameters += @{ - InstanceName = $mockDefaultInstance_InstanceName - SourceCredential = $null - SourcePath = $mockSourcePath + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = 'SQLENGINE' # Must be upper-case since Get-TargetResource returns upper-case. + FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName + FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress + FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + } + } -Verifiable } - # Mock all SSMS products. - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockGetItemProperty_UninstallProducts -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) - } -MockWith $mockGetItemProperty_UninstallProducts2008R2 -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) - } -MockWith $mockGetItemProperty_UninstallProducts2012 -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockGetItemProperty_UninstallProducts2014 -Verifiable - - Mock -CommandName Get-Service -MockWith $mockGetService_DefaultInstance -Verifiable - - #region Mock Get-CimInstance - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_DatabaseService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_AgentService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_FullTextService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_ReportingService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" - } -MockWith $mockGetCimInstance_DefaultInstance_IntegrationService -Verifiable - - Mock -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" - } -MockWith $mockGetCimInstance_DefaultInstance_AnalysisService -Verifiable - - # If Get-CimInstance is used in any other way than those mocks with a ParameterFilter, then throw and error - Mock -CommandName Get-CimInstance -MockWith { - throw "Mock Get-CimInstance was called with unexpected parameters. ClassName=$ClassName, Filter=$Filter" - } -Verifiable - #endregion Mock Get-CimInstance - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -MockWith $mockGetItemProperty_InstanceId_ConfigurationState -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\DQ\*" - } -MockWith $mockGetItemProperty_DQFeature -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" - } -MockWith $mockGetItemProperty_SqlVersion_ConfigurationState -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" - } -MockWith $mockGetItemProperty_ClientComponentsFull_FeatureList -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -MockWith $mockGetItemProperty_Setup -Verifiable - } + # This is a test for regression testing of issue #432 + It 'Should return $false' { + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeFalse - It 'Should return that the desired state is present' { - $result = Test-TargetResource @testParameters - $result | Should -Be $true - - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - #region Assert Get-CimInstance - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_DatabaseServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AgentServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_FullTextServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_ReportingServiceName'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$(($mockDefaultInstance_IntegrationServiceName -f $mockSqlMajorVersion))'" - } -Exactly -Times 1 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -ParameterFilter { - $ClassName -eq 'Win32_Service' -and - $Filter -eq "Name = '$mockDefaultInstance_AnalysisServiceName'" - } -Exactly -Times 1 -Scope It - #endregion Assert Get-CimInstance + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + } } + } - It 'Should return that the desired state is present when the correct clustered instance was found' { - $mockCurrentInstanceName = $mockDefaultInstance_InstanceName - - Mock -CommandName Connect-SQL -MockWith $mockConnectSQLCluster -Verifiable + Context "When the system is in the desired state" { + Context 'When all features are installed' { + BeforeAll { + $testParameters = $mockDefaultParameters.Clone() + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + } - Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterResource -Verifiable -ParameterFilter { - $Filter -eq "Type = 'SQL Server'" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = $mockDefaultFeatures + } + } -Verifiable } - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSClusterResourceGroup_DefaultInstance -Verifiable -ParameterFilter { $ResultClassName -eq 'MSCluster_ResourceGroup' } - - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSClusterResource_DefaultInstance -Verifiable -ParameterFilter { $ResultClassName -eq 'MSCluster_Resource' } - - $testClusterParameters = $testParameters.Clone() + It 'Should return $true' { + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeTrue - $testClusterParameters += @{ - FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName - FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress - FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' } - - $result = Test-TargetResource @testClusterParameters - - $result | Should -Be $true - - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 1 -Scope It -ParameterFilter { $Filter -eq "Type = 'SQL Server'" } - Assert-MockCalled -CommandName Get-CimAssociatedInstance -Exactly -Times 3 -Scope It } - It 'Should not return false after a clustered install due to the presence of a variable called "FailoverClusterDisks"' { - $mockCurrentInstanceName = $mockDefaultInstance_InstanceName - - Mock -CommandName Connect-SQL -MockWith $mockConnectSQLCluster -Verifiable + Context 'When the correct clustered instance was found' { + BeforeAll { + $testParameters = $mockDefaultParameters.Clone() + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName + FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress + FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + } - Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterResource -Verifiable -ParameterFilter { - $Filter -eq "Type = 'SQL Server'" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = $mockDefaultFeatures + FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName + FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress + FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + } + } -Verifiable } - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSClusterResourceGroup_DefaultInstance -Verifiable -ParameterFilter { $ResultClassName -eq 'MSCluster_ResourceGroup' } - - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSClusterResource_DefaultInstance -Verifiable -ParameterFilter { $ResultClassName -eq 'MSCluster_Resource' } - - $testClusterParameters = $testParameters.Clone() + It 'Should return $true' { + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeTrue - $testClusterParameters += @{ - FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName - FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress - FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' } - $mockDynamicSqlDataDirectoryPath = $mockSqlDataDirectoryPath - $mockDynamicSqlUserDatabasePath = $mockSqlUserDatabasePath - $mockDynamicSqlUserDatabaseLogPath = $mockSqlUserDatabaseLogPath - $mockDynamicSqlTempDatabasePath = $mockSqlTempDatabasePath - $mockDynamicSqlTempDatabaseLogPath = $mockSqlTempDatabaseLogPath - $mockDynamicSqlBackupPath = $mockSqlBackupPath + # Regression test when the variables were detected differently. + It 'Should not return false after a clustered install due to the presence of a variable called "FailoverClusterDisks"' { + # These are needed to populate paths when calling (& $mockClusterDiskMap). + $mockDynamicSqlDataDirectoryPath = $mockSqlDataDirectoryPath + $mockDynamicSqlUserDatabasePath = $mockSqlUserDatabasePath + $mockDynamicSqlUserDatabaseLogPath = $mockSqlUserDatabaseLogPath + $mockDynamicSqlTempDatabasePath = $mockSqlTempDatabasePath + $mockDynamicSqlTempDatabaseLogPath = $mockSqlTempDatabaseLogPath + $mockDynamicSqlBackupPath = $mockSqlBackupPath - New-Variable -Name 'FailoverClusterDisks' -Value (& $mockClusterDiskMap)['UserData'] + New-Variable -Name 'FailoverClusterDisks' -Value (& $mockClusterDiskMap)['UserData'] - $result = Test-TargetResource @testClusterParameters + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeTrue - $result | Should -Be $true + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' + } } - # This is a test for regression testing of issue #432 - It 'Should return true if a SQL Server failover cluster has all features and is in desired state' { - $mockCurrentInstanceName = $mockDefaultInstance_InstanceName + # This is a test for regression testing of issue #432. + Context 'When the SQL Server failover cluster has all features and is in desired state' { + BeforeAll { + $testParameters = $mockDefaultParameters.Clone() + $testParameters['Features'] = 'SQLENGINE,AS' + $testParameters += @{ + InstanceName = $mockDefaultInstance_InstanceName + SourceCredential = $null + SourcePath = $mockSourcePath + FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName + FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress + FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + } - Mock -CommandName Get-TargetResource -MockWith { - return @{ - Features = 'SQLENGINE,AS' # Must be upper-case since Get-TargetResource returns upper-case. + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = 'SQLEngine,AS' FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName } - } -Verifiable + } -Verifiable + } - $testClusterParameters = $testParameters.Clone() - $testClusterParameters['Features'] = 'SQLEngine,AS' + It 'Should return $true' { + $result = Test-TargetResource @testParameters -Verbose + $result | Should -BeTrue - $testClusterParameters += @{ - FailoverClusterGroupName = $mockDefaultInstance_FailoverClusterGroupName - FailoverClusterIPAddress = $mockDefaultInstance_FailoverClusterIPAddress - FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName + Assert-MockCalled -CommandName Get-TargetResource -Exactly -Times 1 -Scope 'It' } - - $result = Test-TargetResource @testClusterParameters - $result | Should -Be $true } } @@ -3234,11 +2771,9 @@ try return $true } -Verifiable - <#1 - These mock should not have Verifiable because they are used to test so we never - call them in Assert-MockCalled. - #> - Mock -CommandName Connect-SQL -MockWith $mockConnectSQL + Mock -CommandName Test-PendingRestart -MockWith { + return $false + } # Mock PsDscRunAsCredential context. $PsDscContext = @{ @@ -3247,8 +2782,6 @@ try } BeforeEach { - Mock -CommandName Connect-SQLAnalysis -MockWith $mockConnectSQLAnalysis - $testParameters = $mockDefaultParameters.Clone() $testParameters += @{ SQLSysAdminAccounts = 'COMPANY\User1','COMPANY\SQLAdmins' @@ -3273,23 +2806,11 @@ try Context "When setup process fails with exit code " { BeforeEach { Mock -CommandName Start-SqlSetupProcess -MockWith $mockStartSqlSetupProcess_WithDynamicExitCode -Verifiable - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable - Mock -CommandName Copy-ItemWithRobocopy -Verifiable - Mock -CommandName Get-TemporaryFolder -MockWith $mockGetTemporaryFolder -Verifiable - Mock -CommandName New-Guid -MockWith $mockNewGuid -Verifiable - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } + } -Verifiable $testParameters += @{ InstanceName = $mockDefaultInstance_InstanceName @@ -3327,23 +2848,13 @@ try Context "When SQL Server version is $mockSqlMajorVersion and the system is not in the desired state for a default instance" { BeforeEach { - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable - Mock -CommandName Copy-ItemWithRobocopy -Verifiable - Mock -CommandName Get-TemporaryFolder -MockWith $mockGetTemporaryFolder -Verifiable - Mock -CommandName New-Guid -MockWith $mockNewGuid -Verifiable - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockEmptyHashtable -Verifiable + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } + } -Verifiable - Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable + Mock -CommandName Invoke-InstallationMediaCopy -MockWith $mockNewTemporaryFolder } It 'Should set the system in the desired state when feature is SQLENGINE' { @@ -3368,11 +2879,11 @@ try AsSvcStartupType = $mockAsSvcStartupType IsSvcStartupType = $mockIsSvcStartupType RsSvcStartupType = $mockRsSvcStartupType - SqlTempdbFileCount = 2 - SqlTempdbFileSize = 128 - SqlTempdbFileGrowth = 128 - SqlTempdbLogFileSize = 128 - SqlTempdbLogFileGrowth = 128 + SqlTempDbFileCount = 2 + SqlTempDbFileSize = 128 + SqlTempDbFileGrowth = 128 + SqlTempDbLogFileSize = 128 + SqlTempDbLogFileGrowth = 128 } if ( $mockSqlMajorVersion -in (13,14) ) @@ -3401,38 +2912,16 @@ try AsSvcStartupType = $mockAsSvcStartupType IsSvcStartupType = $mockIsSvcStartupType RsSvcStartupType = $mockRsSvcStartupType - SqlTempdbFileCount = 2 - SqlTempdbFileSize = 128 - SqlTempdbFileGrowth = 128 - SqlTempdbLogFileSize = 128 - SqlTempdbLogFileGrowth = 128 + SqlTempDbFileCount = 2 + SqlTempDbFileSize = 128 + SqlTempDbFileGrowth = 128 + SqlTempDbLogFileSize = 128 + SqlTempDbLogFileGrowth = 128 } { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-TemporaryFolder -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName New-Guid -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Copy-ItemWithRobocopy -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - + Assert-MockCalled -CommandName Invoke-InstallationMediaCopy -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Import-SQLPSModule -Exactly -Times 1 -Scope It @@ -3487,24 +2976,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3528,25 +2999,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3566,40 +3018,13 @@ try $testParameters.Features = $testParameters.Features -replace ',SSMS,ADV_SSMS','' } - # Mocking SharedDirectory (when previously installed and should not be installed again). - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UserData\S-1-5-18\Components\FEE2E540D20152D4597229B6CFBC0A69' - } -MockWith $mockGetItemProperty_SharedDirectory -Verifiable - - # Mocking SharedWowDirectory (when previously installed and should not be installed again). - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UserData\S-1-5-18\Components\A79497A344129F64CA7D69C56F5DD8B4' - } -MockWith $mockGetItemProperty_SharedWowDirectory -Verifiable - - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable - Mock -CommandName Copy-ItemWithRobocopy -Verifiable - Mock -CommandName Get-TemporaryFolder -MockWith $mockGetTemporaryFolder -Verifiable - Mock -CommandName New-Guid -MockWith $mockNewGuid -Verifiable - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } } -Verifiable - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" - } -Verifiable + Mock -CommandName Invoke-InstallationMediaCopy -MockWith $mockNewTemporaryFolder -Verifiable } It 'Should set the system in the desired state when feature is SQLENGINE' { @@ -3616,30 +3041,7 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 2 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 2 -Scope It - Assert-MockCalled -CommandName Get-TemporaryFolder -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName New-Guid -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Copy-ItemWithRobocopy -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - + Assert-MockCalled -CommandName Invoke-InstallationMediaCopy -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3675,25 +3077,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3711,23 +3094,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3749,30 +3115,13 @@ try $testParameters.Features = $testParameters.Features -replace ',SSMS,ADV_SSMS','' } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable - Mock -CommandName Copy-ItemWithRobocopy -Verifiable - Mock -CommandName Get-TemporaryFolder -MockWith $mockGetTemporaryFolder -Verifiable - Mock -CommandName New-Guid -MockWith $mockNewGuid -Verifiable - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } } -Verifiable - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" - } -Verifiable + Mock -CommandName Invoke-InstallationMediaCopy -MockWith $mockNewTemporaryFolder -Verifiable } It 'Should set the system in the desired state when feature is SQLENGINE' { @@ -3789,30 +3138,7 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName New-SmbMapping -Exactly -Times 2 -Scope It - Assert-MockCalled -CommandName Remove-SmbMapping -Exactly -Times 2 -Scope It - Assert-MockCalled -CommandName Get-TemporaryFolder -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName New-Guid -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Copy-ItemWithRobocopy -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - + Assert-MockCalled -CommandName Invoke-InstallationMediaCopy -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Import-SQLPSModule -Exactly -Times 1 -Scope It @@ -3849,25 +3175,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3885,23 +3192,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -3931,24 +3221,10 @@ try $testParameters.Features = $testParameters.Features -replace ',SSMS,ADV_SSMS','' } - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - Mock -CommandName Get-CimInstance -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" - } -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } } -Verifiable } @@ -3966,25 +3242,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -4020,24 +3277,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -4055,23 +3294,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It - - Assert-MockCalled -CommandName Get-CimInstance -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudio2014_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2008R2_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2012_ProductIdentifyingNumber) -or - $Path -eq (Join-Path -Path $mockRegistryUninstallProductsPath -ChildPath $mockSqlServerManagementStudioAdvanced2014_ProductIdentifyingNumber) - } -Exactly -Times 6 -Scope It - Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It } @@ -4093,44 +3315,10 @@ try FailoverClusterNetworkName = $mockDefaultInstance_FailoverClusterNetworkName } - Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterResourceGroup_AvailableStorage -ParameterFilter { - $Filter -eq "Name = 'Available Storage'" - } -Verifiable - - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSCluster_ResourceGroupToResource -ParameterFilter { - ($Association -eq 'MSCluster_ResourceGroupToResource') -and ($ResultClassName -eq 'MSCluster_Resource') - } -Verifiable - - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSCluster_ResourceToPossibleOwner -ParameterFilter { - $Association -eq 'MSCluster_ResourceToPossibleOwner' - } -Verifiable - - Mock -CommandName Get-CimAssociatedInstance -MockWith $mockGetCimAssociatedInstance_MSCluster_DiskPartition -ParameterFilter { - $ResultClassName -eq 'MSCluster_DiskPartition' - } -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterNetwork -ParameterFilter { - ($Namespace -eq 'root/MSCluster') -and ($ClassName -eq 'MSCluster_Network') -and ($Filter -eq 'Role >= 2') - } -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockGetCIMInstance_MSCluster_ClusterSharedVolume -ParameterFilter { - $ClassName -eq 'MSCluster_ClusterSharedVolume' - } -Verifiable - - Mock -CommandName Get-CimInstance -MockWith $mockGetCIMInstance_MSCluster_ClusterSharedVolumeToResource -ParameterFilter { - $ClassName -eq 'MSCluster_ClusterSharedVolumeToResource' - } -Verifiable - - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - } - - BeforeEach { - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" - } -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } } -Verifiable } @@ -4181,6 +3369,12 @@ try SQLBackupDir = $mockDynamicSqlBackupPath } + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } + } -Verifiable + Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterResourceGroup_AvailableStorage -ParameterFilter { $Filter -eq "Name = 'Available Storage'" } -Verifiable @@ -4208,31 +3402,6 @@ try Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterNetwork -ParameterFilter { ($Namespace -eq 'root/MSCluster') -and ($ClassName -eq 'MSCluster_Network') -and ($Filter -eq 'Role >= 2') } -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -MockWith $mockGetItemProperty_InstanceId_ConfigurationState -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\DQ\*" - } -MockWith $mockGetItemProperty_DQFeature -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" - } -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -MockWith $mockGetItemProperty_Setup -Verifiable - - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - } - - BeforeEach { - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" - } -Verifiable } It 'Should pass proper parameters to setup' { @@ -4497,29 +3666,13 @@ try Action = 'PrepareFailoverCluster' } - Mock -CommandName New-SmbMapping -Verifiable - Mock -CommandName Remove-SmbMapping -Verifiable - Mock -CommandName Copy-ItemWithRobocopy -Verifiable - Mock -CommandName Get-TemporaryFolder -MockWith $mockGetTemporaryFolder -Verifiable - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\ConfigurationState" - } -MockWith $mockGetItemProperty_InstanceId_ConfigurationState -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\DQ\*" - } -MockWith $mockGetItemProperty_DQFeature -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } } -Verifiable - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$mockDefaultInstance_InstanceId\Setup" -and $Name -eq 'SqlProgramDir' - } -MockWith $mockGetItemProperty_Setup -Verifiable - - Mock -CommandName Start-SqlSetupProcess -MockWith $mockStartSqlSetupProcess -Verifiable + Mock -CommandName Invoke-InstallationMediaCopy -MockWith $mockNewTemporaryFolder -Verifiable Mock -CommandName Get-CimInstance -ParameterFilter { ($Namespace -eq 'root/MSCluster') -and ($ClassName -eq 'MSCluster_ResourceGroup') -and ($Filter -eq "Name = 'Available Storage'") @@ -4559,13 +3712,6 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw - Assert-MockCalled -CommandName Connect-SQL -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Connect-SQLAnalysis -Exactly -Times 0 -Scope It - Assert-MockCalled -CommandName Get-Service -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq 'HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\Instance Names\SQL' -and - ($Name -eq $mockDefaultInstance_InstanceName) - } -Exactly -Times 0 -Scope It Assert-MockCalled -CommandName Start-SqlSetupProcess -Exactly -Times 1 -Scope It Assert-MockCalled -CommandName Test-TargetResource -Exactly -Times 1 -Scope It @@ -4618,12 +3764,10 @@ try SQLBackupDir = $mockDynamicSqlBackupPath } - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\Tools\Setup\Client_Components_Full" - } -Verifiable - - Mock -CommandName Get-ItemProperty -ParameterFilter { - $Path -eq "HKLM:\SOFTWARE\Microsoft\Microsoft SQL Server\$($mockSqlMajorVersion)0\ConfigurationState" + Mock -CommandName Get-TargetResource -MockWith { + return @{ + Features = '' + } } -Verifiable Mock -CommandName Get-CimInstance -MockWith $mockGetCimInstance_MSClusterResourceGroup_AvailableStorage -ParameterFilter { @@ -4653,8 +3797,6 @@ try Mock -CommandName Get-CimInstance -MockWith $mockGetCIMInstance_MSCluster_ClusterSharedVolumeToResource -ParameterFilter { $ClassName -eq 'MSCluster_ClusterSharedVolumeToResource' } -Verifiable - - Mock -CommandName Get-Service -MockWith $mockEmptyHashtable -Verifiable } It 'Should throw an error when one or more paths are not resolved to clustered storage' { @@ -4667,7 +3809,6 @@ try } It 'Should properly map paths to clustered disk resources' { - $mockStartSqlSetupProcessExpectedArgument = $mockStartSqlSetupProcessExpectedArgumentClusterDefault.Clone() $mockStartSqlSetupProcessExpectedArgument += @{ Action = 'CompleteFailoverCluster' @@ -4731,7 +3872,6 @@ try } It 'Should build a valid IP address string for a single address' { - $mockStartSqlSetupProcessExpectedArgument = $mockStartSqlSetupProcessExpectedArgumentClusterDefault.Clone() $mockStartSqlSetupProcessExpectedArgument += @{ FailoverClusterIPAddresses = $mockDefaultInstance_FailoverClusterIPAddressParameter_SingleSite @@ -4800,221 +3940,11 @@ try { Set-TargetResource @testParameters } | Should -Not -Throw } } - } Assert-VerifiableMock } - # Tests only the parts of the code that does not already get tested thru the other tests. - Describe 'Copy-ItemWithRobocopy' -Tag 'Helper' { - Context 'When Copy-ItemWithRobocopy is called it should return the correct arguments' { - BeforeEach { - Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable - Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy -Verifiable - $mockRobocopyArgumentSourcePathQuoted = '"{0}"' -f $mockRobocopyArgumentSourcePath - $mockRobocopyArgumentDestinationPathQuoted = '"{0}"' -f $mockRobocopyArgumentDestinationPath - } - - - It 'Should use Unbuffered IO when copying' { - $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO - - $mockStartSqlSetupProcessExpectedArgument = - $mockRobocopyArgumentSourcePathQuoted, - $mockRobocopyArgumentDestinationPathQuoted, - $mockRobocopyArgumentCopySubDirectoriesIncludingEmpty, - $mockRobocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource, - $mockRobocopyArgumentUseUnbufferedIO, - $mockRobocopyArgumentSilent -join ' ' - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should not use Unbuffered IO when copying' { - $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithoutUnbufferedIO - - $mockStartSqlSetupProcessExpectedArgument = - $mockRobocopyArgumentSourcePathQuoted, - $mockRobocopyArgumentDestinationPathQuoted, - $mockRobocopyArgumentCopySubDirectoriesIncludingEmpty, - $mockRobocopyArgumentDeletesDestinationFilesAndDirectoriesNotExistAtSource, - '', - $mockRobocopyArgumentSilent -join ' ' - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - } - - Context 'When Copy-ItemWithRobocopy throws an exception it should return the correct error messages' { - BeforeEach { - $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO - - Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable - Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy_WithExitCode -Verifiable - } - - It 'Should throw the correct error message when error code is 8' { - $mockStartSqlSetupProcessExitCode = 8 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Throw "Robocopy reported errors when copying files. Error code: $mockStartSqlSetupProcessExitCode." - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should throw the correct error message when error code is 16' { - $mockStartSqlSetupProcessExitCode = 16 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Throw "Robocopy reported errors when copying files. Error code: $mockStartSqlSetupProcessExitCode." - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should throw the correct error message when error code is greater than 7 (but not 8 or 16)' { - $mockStartSqlSetupProcessExitCode = 9 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Throw "Robocopy reported that failures occurred when copying files. Error code: $mockStartSqlSetupProcessExitCode." - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - } - - Context 'When Copy-ItemWithRobocopy is called and finishes successfully it should return the correct exit code' { - BeforeEach { - $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO - - Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable - Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy_WithExitCode -Verifiable - } - - AfterEach { - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should finish successfully with exit code 1' { - $mockStartSqlSetupProcessExitCode = 1 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should finish successfully with exit code 2' { - $mockStartSqlSetupProcessExitCode = 2 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should finish successfully with exit code 3' { - $mockStartSqlSetupProcessExitCode = 3 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePath - DestinationPath = $mockRobocopyArgumentDestinationPath - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - } - } - Context 'When Copy-ItemWithRobocopy is called with spaces in paths and finishes successfully it should return the correct exit code' { - BeforeEach { - $mockRobocopyExecutableVersion = $mockRobocopyExecutableVersionWithUnbufferedIO - - Mock -CommandName Get-Command -MockWith $mockGetCommand -Verifiable - Mock -CommandName Start-Process -MockWith $mockStartSqlSetupProcess_Robocopy_WithExitCode -Verifiable - } - - AfterEach { - Assert-MockCalled -CommandName Get-Command -Exactly -Times 1 -Scope It - Assert-MockCalled -CommandName Start-Process -Exactly -Times 1 -Scope It - } - - It 'Should finish successfully with exit code 1' { - $mockStartSqlSetupProcessExitCode = 1 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePathWithSpaces - DestinationPath = $mockRobocopyArgumentDestinationPathWithSpaces - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - - } - - It 'Should finish successfully with exit code 2' { - $mockStartSqlSetupProcessExitCode = 2 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePathWithSpaces - DestinationPath = $mockRobocopyArgumentDestinationPathWithSpaces - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - - } - - It 'Should finish successfully with exit code 3' { - $mockStartSqlSetupProcessExitCode = 3 - - $copyItemWithRobocopyParameter = @{ - Path = $mockRobocopyArgumentSourcePathWithSpaces - DestinationPath = $mockRobocopyArgumentDestinationPathWithSpaces - } - - { Copy-ItemWithRobocopy @copyItemWithRobocopyParameter } | Should -Not -Throw - } - } - } - Describe 'Get-ServiceAccountParameters' -Tag 'Helper' { $serviceTypes = @('SQL','AGT','IS','RS','AS','FT') @@ -5083,45 +4013,6 @@ try } } - Describe 'Get-TemporaryFolder' -Tag 'Helper' { - BeforeAll { - $mockExpectedTempPath = [IO.Path]::GetTempPath() - } - - Context 'When using Get-TemporaryFolder' { - It 'Should return the correct temporary path' { - Get-TemporaryFolder | Should -BeExactly $mockExpectedTempPath - } - } - } - - Describe 'Start-SqlSetupProcess' -Tag 'Helper' { - Context 'When starting a process successfully' { - It 'Should return exit code 0' { - $startSqlSetupProcessParameters = @{ - FilePath = 'powershell.exe' - ArgumentList = '-Command &{Start-Sleep -Seconds 2}' - Timeout = 30 - } - - $processExitCode = Start-SqlSetupProcess @startSqlSetupProcessParameters - $processExitCode | Should -BeExactly 0 - } - } - - Context 'When starting a process and the process does not finish before the timeout period' { - It 'Should throw an error message' { - $startSqlSetupProcessParameters = @{ - FilePath = 'powershell.exe' - ArgumentList = '-Command &{Start-Sleep -Seconds 3}' - Timeout = 2 - } - - { Start-SqlSetupProcess @startSqlSetupProcessParameters } | Should -Throw - } - } - } - Describe 'Get-InstalledSharedFeatures' -Tag 'Helper' { Context 'When there are no shared features installed' { BeforeAll { From dfd32b3ca3a364996a2dd9908f77dfb4f7f5f507 Mon Sep 17 00:00:00 2001 From: Katie Kragenbrink Date: Wed, 3 Apr 2019 12:21:14 -0700 Subject: [PATCH 10/10] Releasing version 12.4.0.0 --- CHANGELOG.md | 2 + SqlServerDsc.psd1 | 134 ++++++++++++++++++++++++++-------------------- 2 files changed, 77 insertions(+), 59 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 827851b12..cc668c8d3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,8 @@ ## Unreleased +## 12.4.0.0 + - Changes to SqlServerDsc - Added new resources. - SqlRSSetup diff --git a/SqlServerDsc.psd1 b/SqlServerDsc.psd1 index 97bb461cc..7135da128 100644 --- a/SqlServerDsc.psd1 +++ b/SqlServerDsc.psd1 @@ -1,6 +1,6 @@ @{ # Version number of this module. - moduleVersion = '12.3.0.0' + moduleVersion = '12.4.0.0' # ID used to uniquely identify this module GUID = '693ee082-ed36-45a7-b490-88b07c86b42f' @@ -50,65 +50,80 @@ # ReleaseNotes of this module ReleaseNotes = '- Changes to SqlServerDsc - - Reverting the change that was made as part of the - [issue 1260](https://github.com/PowerShell/SqlServerDsc/issues/1260) - in the previous release, as it only mitigated the issue, it did not - solve the issue. - - Removed the container testing since that broke the integration tests, - possible due to using excessive amount of memory on the AppVeyor build - worker. This will make the unit tests to take a bit longer to run - ([issue 1260](https://github.com/PowerShell/SqlServerDsc/issues/1260)). - - The unit tests and the integration tests are now run in two separate - build workers in AppVeyor. One build worker runs the integration tests, - while a second build worker runs the unit tests. The build workers runs - in parallel on paid accounts, but sequentially on free accounts - ([issue 1260](https://github.com/PowerShell/SqlServerDsc/issues/1260)). - - Clean up error handling in some of the integration tests that was - part of a workaround for a bug in Pester. The bug is resolved, and - the error handling is not again built into Pester. - - Speeding up the AppVeyor tests by splitting the common tests in a - separate build job. - - Updated the appveyor.yml to have the correct build step, and also - correct run the build step only in one of the jobs. - - Update integration tests to use the new integration test template. - - Added SqlAgentOperator resource. -- Changes to SqlServiceAccount - - Fixed Get-ServiceObject when searching for Integration Services service. - Unlike the rest of SQL Server services, the Integration Services service - cannot be instanced, however you can have multiple versions installed. - Get-Service object would return the correct service name that you - are looking for, but it appends the version number at the end. Added - parameter VersionNumber so the search would return the correct - service name. - - Added code to allow for using Managed Service Accounts. - - Now the correct service type string value is returned by the function - `Get-TargetResource`. Previously one value was passed in as a parameter - (e.g. `DatabaseEngine`), but a different string value as returned - (e.g. `SqlServer`). Now `Get-TargetResource` return the same values - that can be passed as values in the parameter `ServiceType` - ([issue 981](https://github.com/PowerShell/SqlServerDsc/issues/981)). -- Changes to SqlServerLogin - - Fixed issue in Test-TargetResource to valid password on disabled accounts - ([issue 915](https://github.com/PowerShell/SqlServerDsc/issues/915)). - - Now when adding a login of type SqlLogin, and the SQL Server login mode - is set to `"Integrated"`, an error is correctly thrown - ([issue 1179](https://github.com/PowerShell/SqlServerDsc/issues/1179)). + - Added new resources. + - SqlRSSetup + - Added helper module DscResource.Common from the repository + DscResource.Template. + - Moved all helper functions from SqlServerDscHelper.psm1 to DscResource.Common. + - Renamed Test-SqlDscParameterState to Test-DscParameterState. + - New-TerminatingError error text for a missing localized message now matches + the output even if the "missing localized message" localized message is + also missing. + - Added helper module DscResource.LocalizationHelper from the repository + DscResource.Template, this replaces the helper module CommonResourceHelper.psm1. + - Cleaned up unit tests, mostly around loading cmdlet stubs and loading + classes stubs, but also some tests that were using some odd variants. + - Fix all integration tests according to issue [PowerShell/DscResource.Template14](https://github.com/PowerShell/DscResource.Template/issues/14). +- Changes to SqlServerMemory + - Updated Cim Class to Win32_ComputerSystem (instead of Win32_PhysicalMemory) + because the correct memory size was not being detected correctly on Azure VMs + ([issue 914](https://github.com/PowerShell/SqlServerDsc/issues/914)). - Changes to SqlSetup - - Updated the integration test to stop the named instance while installing - the other instances to mitigate - [issue 1260](https://github.com/PowerShell/SqlServerDsc/issues/1260). - - Add parameters to configure the Tempdb files during the installation of - the instance. The new parameters are SqlTempdbFileCount, SqlTempdbFileSize, - SqlTempdbFileGrowth, SqlTempdbLogFileSize and SqlTempdbLogFileGrowth - ([issue 1167](https://github.com/PowerShell/SqlServerDsc/issues/1167)). -- Changes to SqlServerEndpoint - - Add the optional parameter Owner. The default owner remains the login used - for the creation of the endpoint - ([issue 1251](https://github.com/PowerShell/SqlServerDsc/issues/1251)). - [Maxime Daniou (@mdaniou)](https://github.com/mdaniou) - - Add integration tests - ([issue 744](https://github.com/PowerShell/SqlServerDsc/issues/744)). - [Maxime Daniou (@mdaniou)](https://github.com/mdaniou) + - Split integration tests into two jobs, one for running integration tests + for SQL Server 2016 and another for running integration test for + SQL Server 2017 ([issue 858](https://github.com/PowerShell/SqlServerDsc/issues/858)). + - Localized messages for Master Data Services no longer start and end with + single quote. + - When installing features a verbose message is written if a feature is found + to already be installed. It no longer quietly removes the feature from the + `/FEATURES` argument. + - Cleaned up a bit in the tests, removed excessive piping. + - Fixed minor typo in examples. + - A new optional parameter `FeatureFlag` parameter was added to control + breaking changes. Functionality added under a feature flag can be + toggled on or off, and could be changed later to be the default. + This way we can also make more of the new functionalities the default + in the same breaking change release ([issue 1105](https://github.com/PowerShell/SqlServerDsc/issues/1105)). + - Added a new way of detecting if the shared feature CONN (Client Tools + Connectivity, and SQL Client Connectivity SDK), BC (Client Tools + Backwards Compatibility), and SDK (Client Tools SDK) is installed or + not. The new functionality is used when the parameter `FeatureFlag` + is set to `"DetectionSharedFeatures"` ([issue 1105](https://github.com/PowerShell/SqlServerDsc/issues/1105)). + - Added a new helper function `Get-InstalledSharedFeatures` to move out + some of the code from the `Get-TargetResource` to make unit testing + easier and faster. + - Changed the logic of "Build the argument string to be passed to setup" to + not quote the value if root directory is specified + ([issue 1254](https://github.com/PowerShell/SqlServerDsc/issues/1254)). + - Moved some resource specific helper functions to the new helper module + DscResource.Common so they can be shared with the new resource SqlRSSetup. + - Improved verbose messages in Test-TargetResource function to more + clearly tell if features are already installed or not. + - Refactored unit tests for the functions Test-TargetResource and + Set-TargetResource to improve testing speed. + - Modified the Test-TargetResource and Set-TargetResource to not be + case-sensitive when comparing feature names. *This was handled + correctly in real-world scenarios, but failed when running the unit + tests (and testing casing).* +- Changes to SqlAGDatabase + - Fix MatchDatabaseOwner to check for CONTROL SERVER, IMPERSONATE LOGIN, or + CONTROL LOGIN permission in addition to IMPERSONATE ANY LOGIN. + - Update and fix MatchDatabaseOwner help text. +- Changes to SqlAG + - Updated documentation on the behaviour of defaults as they only apply when + creating a group. +- Changes to SqlAGReplica + - AvailabilityMode, BackupPriority, and FailoverMode defaults only apply when + creating a replica not when making changes to an existing replica. Explicit + parameters will still change existing replicas ([issue 1244](https://github.com/PowerShell/SqlServerDsc/issues/1244)). + - ReadOnlyRoutingList now gets updated without throwing an error on the first + run ([issue 518](https://github.com/PowerShell/SqlServerDsc/issues/518)). + - Test-Resource fixed to report whether ReadOnlyRoutingList desired state + has been reached correctly ([issue 1305](https://github.com/PowerShell/SqlServerDsc/issues/1305)). +- Changes to SqlDatabaseDefaultLocation + - No longer does the Test-TargetResource fail on the second test run + when the backup file path was changed, and the path was ending with + a backslash ([issue 1307](https://github.com/PowerShell/SqlServerDsc/issues/1307)). ' @@ -135,4 +150,5 @@ +