-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDeploymentReadinessChecker.psm1
290 lines (227 loc) · 14.7 KB
/
DeploymentReadinessChecker.psm1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
#Requires -Version 4
#Requires -Modules 'Pester'
Function Test-DeploymentReadiness {
<#
.SYNOPSIS
Validates that one or more computers meet the prerequisites for a software deployment/upgrade.
.DESCRIPTION
Validates that one or more computers meet the prerequisites for a software deployment/upgrade.
It generates a NUnit-style test result file for each computer and creates a visual, dynamic HTLM report encompassing data from all the test results.
The list of computers to check is specified via the ComputerName parameter.
The deployment or upgrade prerequisites are specified in a Pester-based validation script located in the sub-directory \ReadinessValidationScript.
Test-DeploymentReadiness can only invoke one validation script at a time, even if there are multiple scripts named *.Tests.ps1 in the ReadinessValidationScript sub-directory.
.PARAMETER ComputerName
To specify one or more computers against which the prerequisites checks will be performed.
If the validation script has a ComputerName parameter, the function passes one computer at a time to its ComputerName parameter, via the Script parameter of Invoke-Pester.
.PARAMETER Credential
To specify the credentials to connect remotely to the target computers.
If the validation script has a Credential parameter, the function passes the value of its own Credential parameter to the validation script, via the Script parameter of Invoke-Pester.
.PARAMETER OutputPath
To specify in which directory the output test results files and the summary report should be located.
If the directory doesn't exist, it will be created.
If not specified, the defaut output path is the current directory.
.PARAMETER TestParameters
If the test script used to validate the prerequisites take parameters, their names and values can be specified as a hashtable via this parameter.
Then, the function will pass these into the Script parameter of Invoke-Pester, when calling the validation script.
To see the format of the hashtable for this parameter, please refer to the examples by running : Get-Help Test-DeploymentReadiness -Examples
.PARAMETER Tag
If the Pester validation script contains Describe blocks with tags, only the tests in Describe blocks with the specified Tag parameter value(s) are run.
Wildcard characters and Tag values that include spaces or whitespace characters are not supported.
.PARAMETER ExcludeTag
If the Pester validation script contains Describe blocks with tags, tests in Describe blocks with the specified Tag parameter values are omitted.
Wildcard characters and Tag values that include spaces or whitespace characters are not supported.
Just like the ExcludeTag parameter of Invoke-Pester, when you specify multiple ExcludeTag values, this omits tests that have any of the listed tags (it ORs the tags).
.PARAMETER ValidationScript
This is a dynamic parameter which is made available (and mandatory) whenever there is more than one test script in the sub-folder \ReadinessValidationScript\.
This is because Test-DeploymentReadiness can only invoke one validation script at a time, so if there is more than one, the user has to specify which one.
This parameter expects the name (not the full path) of one of the test file present in <ModuleFolder>\ReadinessValidationScript\.
If no value is specified when there is more than one validation script available, the error message will tell the user what are the possible values.
(See the last example in the Examples section of the help.)
.EXAMPLE
Test-DeploymentReadiness -ComputerName (Get-Content .\Computers_List.txt) -Credential (Get-Credential) -OutputPath $env:USERPROFILE\Desktop\DeploymentReadinessReport
Validates that all the computers with the name listed in the file Computers_list.txt meet the prerequisites specified in a validation script located in the sub-directory \ReadinessValidationScript.
.EXAMPLE
$TestParams = @{ DeploymentServerName = $DeploymentServerName; ManagementServerName = $ManagementServerName }
Test-DeploymentReadiness -ComputerName 'Server1','Server2' -Credential (Get-Credential) -TestParameters $TestParams
Validates that all the computers with the name listed in the file Computers_list.txt meet the prerequisites specified in a validation script located in the sub-directory \ReadinessValidationScript.
It uses a hashtable ($TestParams) to pass parameter names and values to the validation script.
.EXAMPLE
'Computer1','Computer2','Computer3' | Test-DeploymentReadiness -Credential (Get-Credential) -OutputPath $env:USERPROFILE\Desktop\DeploymentReadinessReport
Validates that all the computers specified via pipeline input meet the prerequisites specified in a validation script located in the sub-directory \ReadinessValidationScript.
.EXAMPLE
Test-DeploymentReadiness -ComputerName (Get-Content .\Computers_List.txt) -Credential (Get-Credential) -OutputPath $env:USERPROFILE\Desktop\DeploymentReadinessReport
cmdlet Test-DeploymentReadiness at command pipeline position 1
Supply values for the following parameters:
(Type !? for Help.)
ValidationScript:
Test-DeploymentReadiness : Cannot validate argument on parameter
'ValidationScript'. The argument "" does not belong to the set
"ClientDeployment.Tests.ps1,Example.Tests.ps1" specified by the ValidateSet
attribute. Supply an argument that is in the set and then try the command again.
At line:1 char:1
+ Test-DeploymentReadiness -ComputerName 'Devops-test-dscnuc' -OutputPa ...
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ CategoryInfo : InvalidData: (:) [Test-DeploymentReadiness], ParameterBindingValidationException
+ FullyQualifiedErrorId : ParameterArgumentValidationError,Test-DeploymentReadiness
In this case, there is more than one validation script in the sub-folder \ReadinessValidationScript\, so the user has to specify the name of the validation script via the ValidationScript parameter.
Note that the error message provides the set of possible values ("ClientDeployment.Tests.ps1" and "Example.Tests.ps1", here).
.NOTES
Author : Mathieu Buisson
.LINK
https://github.com/MathieuBuisson/DeploymentReadinessChecker
#>
[cmdletbinding()]
param(
[Parameter(ValueFromPipeline=$True, Mandatory=$True, Position=0)]
[string[]]$ComputerName,
[Parameter(Position=1)]
[pscredential]$Credential,
[Parameter(Position=2)]
[string]$OutputPath = $($pwd.ProviderPath),
[Parameter(Position=3)]
[hashtable]$TestParameters,
[Parameter(Position=4)]
[Alias('Tags')]
[string[]]$Tag,
[Parameter(Position=5)]
[Alias('ExcludeTags')]
[string[]]$ExcludeTag
)
DynamicParam {
# The ValidationScript parameter is made available (and mandatory) only there is more than one test script in the sub-folder \ReadinessValidationScript\.
If ( (Get-ChildItem -Path "$PSScriptRoot\ReadinessValidationScript\" -Filter '*.Tests.ps1').Count -gt 1 ) {
$ParameterName = 'ValidationScript'
# Creating a parameter dictionary
$RuntimeParameterDictionary = New-Object -TypeName System.Management.Automation.RuntimeDefinedParameterDictionary
# Creating an empty collection of parameter attributes
$AttributeCollection = New-Object -TypeName System.Collections.ObjectModel.Collection[System.Attribute]
# Setting parameter attributes and values
$ValidationScriptAttribute = New-Object -TypeName System.Management.Automation.ParameterAttribute
$ValidationScriptAttribute.Mandatory = $True
$ValidationScriptAttribute.Position = 6
$ValidationScriptAttribute.HelpMessage = "There was more than one test script found in $PSScriptRoot\ReadinessValidationScript\. `r`nPlease specify the name of the test script to use. `r`nTip : Use Tab completion to see the possible script names."
# Adding the parameter attributes to the attributes collection
$AttributeCollection.Add($ValidationScriptAttribute)
# Generating dynamic values for a ValidateSet
$SetValues = Get-ChildItem "$PSScriptRoot\ReadinessValidationScript" -Filter '*.Tests.ps1' | Select-Object -ExpandProperty Name
$ValidateSetAttribute = New-Object -TypeName System.Management.Automation.ValidateSetAttribute($SetValues)
# Adding the ValidateSet to the attributes collection
$AttributeCollection.Add($ValidateSetAttribute)
# Creating the dynamic parameter
$RuntimeParameter = New-Object -TypeName System.Management.Automation.RuntimeDefinedParameter($ParameterName, [string], $AttributeCollection)
$RuntimeParameterDictionary.Add($ParameterName, $RuntimeParameter)
return $RuntimeParameterDictionary
}
}
Begin {
If ( -not(Test-Path -Path $OutputPath) ) {
New-Item -ItemType Directory -Path $OutputPath -Force
}
# Checking if the validation script has a ComputerName parameter
If ( $PSBoundParameters.ContainsKey('ValidationScript') ) {
$ValidationScriptFile = Join-Path -Path "$PSScriptRoot\ReadinessValidationScript" -ChildPath $PSBoundParameters.ValidationScript
}
Else {
$ValidationScriptFile = (Get-ChildItem -Path "$PSScriptRoot\ReadinessValidationScript\" -Filter '*.Tests.ps1').FullName
}
Write-Verbose "The validation script file is : $ValidationScriptFile"
$ScriptInfo = Get-Command $ValidationScriptFile
[System.Boolean]$HasComputerNameParameter = $ScriptInfo.Parameters.Keys -contains 'ComputerName'
Write-Verbose "Does the Pester validation script have a ComputerName parameter ? $($HasComputerNameParameter)."
# Checking if credentials to connect to target computers were specified
If ( $PSBoundParameters.ContainsKey('Credential') ) {
$CredentialSpecified = $True
}
# Checking if the validation script has a Credential parameter
[System.Boolean]$HasCredentialParameter = $ScriptInfo.Parameters.Keys -contains 'Credential'
Write-Verbose "Does the Pester validation script have a Credential parameter ? $($HasCredentialParameter)."
# Setting tag filtering parameters to pass to Invoke-Pester if the Tag or ExcludeTag parameter is specified
If ( $PSBoundParameters.ContainsKey('Tag') -or $PSBoundParameters.ContainsKey('ExcludeTag') ) {
[hashtable]$TagFilteringParameters = @{}
Write-Verbose 'Tag filtering is ON.'
If ( $PSBoundParameters.ContainsKey('Tag') ) {
$TagFilteringParameters.Add('Tag', $Tag)
}
If ( $PSBoundParameters.ContainsKey('ExcludeTag') ) {
$TagFilteringParameters.Add('ExcludeTag', $ExcludeTag)
}
}
}
Process {
# If the validation script has a Credential parameter, the function passes the value of
# its own Credential parameter to the validation script, via the Script parameter of Invoke-Pester.
If ( $CredentialSpecified -and $HasCredentialParameter ) {
If ( $PSBoundParameters.ContainsKey('TestParameters') ) {
If ( $TestParameters.Credential ) {
$TestParameters.Credential = $Credential
}
Else {
$TestParameters.Add('Credential', $Credential)
}
}
Else {
$TestParameters = @{ Credential = $Credential }
}
}
Foreach ( $Computer in $ComputerName ) {
# Cleaning up any leading or trailing space
$Computer = $Computer.Trim()
# If the validation script has a ComputerName parameter, the function passes one computer at a
# time to the validation script's ComputerName parameter, via the Script parameter of Invoke-Pester.
If ( $HasComputerNameParameter ) {
If ( $TestParameters ) {
If ( $TestParameters.ComputerName ) {
$TestParameters.ComputerName = $Computer
}
Else {
$TestParameters.Add('ComputerName', $Computer)
}
}
Else {
$TestParameters = @{ ComputerName = $Computer }
}
}
# Building the hashtable to pass parameters to the Pester validation script via the Script parameter of Invoke-Pester
If ( $TestParameters ) {
Foreach ( $Key in $TestParameters.Keys ) {
Write-Verbose "Parameter passed to the validation script. Key : $Key, Value : $($TestParameters.$Key)"
}
$ScriptParameters = @{
Path = $ValidationScriptFile
Parameters = $TestParameters
}
}
Else {
$ScriptParameters = @{
Path = $ValidationScriptFile
}
}
Write-Verbose "Running Pester validation script against computer : $Computer"
If ( $TagFilteringParameters ) {
Invoke-Pester -Script $ScriptParameters -OutputFile "$OutputPath\$Computer.xml" -OutputFormat NUnitXml @TagFilteringParameters
}
Else {
Invoke-Pester -Script $ScriptParameters -OutputFile "$OutputPath\$Computer.xml" -OutputFormat NUnitXml
}
}
}
End {
Invoke-ReportUnit -OutputPath $OutputPath
}
}
# Wrapper function to call ReportUnit.exe, this is mainly to be able to mock ReportUnit calls
Function Invoke-ReportUnit ($OutputPath) {
$ReportUnitPath = "$PSScriptRoot\ReportUnit\ReportUnit.exe"
$Null = & $ReportUnitPath $OutputPath
If ( $LASTEXITCODE -eq 0 ) {
Write-Host "`r`nThe deployment readiness report has been successfully created."
Write-Host "To view the report, please open the following file : $(Join-Path -Path $OutputPath -ChildPath 'Index.html')"
# It maybe be useful to output the file containing the overview report to the pipeline, in case the user wants to do something with it.
Get-ChildItem -Path (Join-Path -Path $OutputPath -ChildPath 'Index.html')
}
Else {
Write-Error "An error occurred when ReportUnit was generating HTML reports from the Pester test results. To troubleshoot this, try running '$PSScriptRoot\ReportUnit\ReportUnit.exe' manually to see the actual error message."
}
}
New-Alias -Name 'tdr' -Value 'Test-DeploymentReadiness'
Export-ModuleMember -Function 'Test-DeploymentReadiness' -Alias 'tdr'