forked from pester/Pester
-
Notifications
You must be signed in to change notification settings - Fork 0
/
build.ps1
341 lines (285 loc) · 13.2 KB
/
build.ps1
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
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
#! /usr/bin/pwsh
<#
.SYNOPSIS
Used to build and import the Pester module during Pester development.
The code that should be excluded from the build is wrapped in this if:
# PESTER_BUILD
if (-not (Get-Variable -Name "PESTER_BUILD" -ValueOnly -ErrorAction Ignore)) {
...Your code here...
}
# end PESTER_BUILD
This will exclude the code from execution when dot-sourcing, but keep it when running the
file directly. When inlining into single file, the regex below will omit the wrapped code.
The # endif comment is significant, because it allows the regex to identify the correct end }.
You can also include the code in build. Which will run the code only when dot-sources into Pester.psm1.
# PESTER_BUILD
if ((Get-Variable -Name "PESTER_BUILD" -ValueOnly -ErrorAction Ignore)) {
...Your code here...
}
# end PESTER_BUILD
Or any other if that uses the variable.
Any such blocks will be excluded entrily when inlining.
.PARAMETER Load
Imports the built Pester module in a separate PowerShell session
and measure how fast it imports. If the module cannot be imported it throws
an error.
.PARAMETER Clean
Cleans the build folder ./bin and rebuilds the assemblies.
.PARAMETER Configuration
The configuration used to build the assemblies. Only used in combination with -Clean. Defaults to Release.
.PARAMETER LockedRestore
Restore nuget packages using the nuget lock file. Useful only for CI.
.PARAMETER Inline
Inline all files into Pester.psm1, instead of dot-sourcing. This is how the real build is done,
but inlinig the files is annoying for local development.
$env:PESTER_BUILD_INLINE=1 environment variable is used to force inlining in files that don't provide
the -Inline parameter. When this gets stuck, and you see your module inline even when it should not,
use -Inline:$false to reset it to 0.
.PARAMETER Inline
Builds the module as a single Pester.psm1-file, instead of dot-sourcing the files. This is the mode used in release build.
#>
[CmdletBinding()]
param (
[switch] $Load,
[switch] $Clean,
[ValidateSet('Debug', 'Release')]
[string] $Configuration = 'Release',
[switch] $LockedRestore,
[switch] $Inline,
[switch] $Import
)
$ErrorActionPreference = 'Stop'
Get-Module Pester | Remove-Module
if ($Clean -and $PSVersionTable.PSVersion -lt [version]'5.1') {
throw 'Clean build of Pester requires PowerShell 5.1 or greater. If you have already compiled the assemblies and only modified powershell-files, try calling ./build.ps1 without -Clean.'
}
if ($Clean -and (Test-Path "$PSScriptRoot/bin")) {
Remove-Item "$PSScriptRoot/bin" -Recurse -Force
}
if ($Clean) {
# Using Import-LocalizedData over Test-ModuleManifest because the latter requires psm1 and
# PesterConfiguration.Format.xml to exists which are both generated later in build script
$manifest = Import-LocalizedData -FileName 'Pester.psd1' -BaseDirectory "$PSScriptRoot/src"
if (-not $LockedRestore) {
dotnet restore "$PSScriptRoot/src/csharp/Pester.sln"
}
else {
dotnet restore "$PSScriptRoot/src/csharp/Pester.sln" --locked-mode
}
dotnet build "$PSScriptRoot/src/csharp/Pester.sln" --no-restore --configuration $Configuration -p:VersionPrefix="$($manifest.ModuleVersion)" -p:VersionSuffix="$($manifest.PrivateData.PSData.Prerelease)"
if (0 -ne $LASTEXITCODE) {
throw 'build failed!'
}
}
if ($Clean) {
# Update PesterConfiguration help in about_PesterConfiguration
if ($PSVersionTable.PSVersion.Major -ge 6) {
$null = [Reflection.Assembly]::LoadFrom("$PSScriptRoot/src/csharp/Pester/bin/$Configuration/net8.0/Pester.dll")
}
else {
$null = [Reflection.Assembly]::LoadFrom("$PSScriptRoot/src/csharp/Pester/bin/$Configuration/net462/Pester.dll")
}
function Format-NicelyMini ($value) {
if ($value -is [bool]) {
if ($value) {
'$true'
}
else {
'$false'
}
}
if ($value -is [int] -or $value -is [decimal]) {
return $value
}
if ($value -is [string]) {
if ([String]::IsNullOrEmpty($value)) {
return '$null'
}
else {
return "'$value'"
}
}
# does not work with [object[]] when building for some reason
if ($value -is [System.Collections.IEnumerable]) {
if (0 -eq $value.Count) {
return '@()'
}
$v = foreach ($i in $value) {
Format-NicelyMini $i
}
return "@($($v -join ', '))"
}
}
# generate help for config object and insert it
$configObject = [PesterConfiguration]::Default
$eol = [Environment]::NewLine
$generatedConfig = foreach ($p in $configObject.PSObject.Properties.Name) {
$section = $configObject.($p)
"${p}:"
foreach ($r in $section.PSObject.Properties.Name) {
$option = $section.$r
$default = Format-NicelyMini $option.Default
$type = $option.Default.GetType() -as [string]
" ${r}: $($option.Description)$eol Type: ${type}$eol Default value: ${default}$eol"
}
}
$helpPath = "$PSScriptRoot/src/en-US/about_PesterConfiguration.help.txt"
# in older versions utf8 means with BOM
$enc = if ($PSVersionTable.PSVersion.Major -ge 7) { 'utf8BOM' } else { 'utf8' }
$helpContent = Get-Content $helpPath -Encoding $enc
# Get section margin from the topic name on line 2
$margin = if ($helpContent[1] -match '^(?<margin>\s*)about_') { $matches.margin } else { '' }
$sbf = [System.Text.StringBuilder]''
$generated = $false
foreach ($line in $helpContent) {
if (-not $generated -and $line -match '^SECTIONS AND OPTIONS\s*$') {
$generated = $true
$null = $sbf.AppendLine("$line$eol")
foreach ($l in @($generatedConfig -split $eol)) {
$m = if ($l) { $margin } else { $null }
$null = $sbf.AppendLine("$m$l")
}
}
elseif ($generated -and ($line -cmatch '^SEE ALSO\s*$')) {
$generated = $false
}
if (-not $generated) {
$null = $sbf.AppendLine($line)
}
}
Set-Content -Encoding $enc -Value $sbf.ToString().TrimEnd() -Path $helpPath
}
function Copy-Content ($Content) {
foreach ($c in $content) {
$source, $destination = $c
$null = New-Item -Force $destination -ItemType Directory
Get-ChildItem $source -File | Copy-Item -Destination $destination
}
}
$content = @(
, ("$PSScriptRoot/src/en-US/*.txt", "$PSScriptRoot/bin/en-US/")
, ("$PSScriptRoot/src/Pester.ps1", "$PSScriptRoot/bin/")
, ("$PSScriptRoot/src/Pester.psd1", "$PSScriptRoot/bin/")
, ("$PSScriptRoot/src/Pester.Format.ps1xml", "$PSScriptRoot/bin/")
)
if ($Clean) {
$content += @(
, ("$PSScriptRoot/src/schemas/JUnit4/*.xsd", "$PSScriptRoot/bin/schemas/JUnit4/")
, ("$PSScriptRoot/src/schemas/NUnit25/*.xsd", "$PSScriptRoot/bin/schemas/NUnit25/")
, ("$PSScriptRoot/src/schemas/NUnit3/*.xsd", "$PSScriptRoot/bin/schemas/NUnit3/")
, ("$PSScriptRoot/src/schemas/Cobertura/*.dtd", "$PSScriptRoot/bin/schemas/Cobertura/")
, ("$PSScriptRoot/src/schemas/JaCoCo/*.dtd", "$PSScriptRoot/bin/schemas/JaCoCo/")
, ("$PSScriptRoot/src/csharp/Pester/bin/$Configuration/net462/Pester.dll", "$PSScriptRoot/bin/bin/net462/")
, ("$PSScriptRoot/src/csharp/Pester/bin/$Configuration/net8.0/Pester.dll", "$PSScriptRoot/bin/bin/net8.0/")
)
}
Copy-Content -Content $content
if ($Clean) {
# Generate PesterConfiguration.Format.ps1xml to ensure list view for all sections
$configSections = $configObject.GetType().Assembly.GetExportedTypes() | Where-Object { $_.BaseType -eq [Pester.ConfigurationSection] }
# Get internal ctor as public ctor always returns instanceId = zero guid prior to PS v7.1.0
$formatViewCtor = [System.Management.Automation.FormatViewDefinition].GetConstructors('Instance,NonPublic')
# Generate listcontrol views for all configuration sections
$typeDefs = foreach ($section in $configSections) {
$builder = [System.Management.Automation.ListControl]::Create().StartEntry()
$section.GetProperties() | Where-Object { $_.PropertyType.IsSubclassOf([Pester.Option]) } | ForEach-Object {
$builder.AddItemProperty($_.Name) > $null
}
$listControl = $builder.EndEntry().EndList()
$ViewDef = $formatViewCtor.Invoke(($section.FullName, $listControl, [guid]::NewGuid())) -as [System.Collections.Generic.List[System.Management.Automation.FormatViewDefinition]]
[System.Management.Automation.ExtendedTypeDefinition]::new($section.FullName, $ViewDef)
}
# Create view for Option to ensure Table and hide IsModified
$builder = [System.Management.Automation.TableControl]::Create().StartRowDefinition()
[Pester.Option[bool]].GetProperties() | Where-Object Name -NotIn 'IsModified' | ForEach-Object {
$builder.AddPropertyColumn($_.Name, [System.Management.Automation.Alignment]::Undefined, $null) > $null
}
$tableControl = $builder.EndRowDefinition().EndTable()
$ViewDef = $formatViewCtor.Invoke(('Pester.Option', $tableControl, [guid]::NewGuid())) -as [System.Collections.Generic.List[System.Management.Automation.FormatViewDefinition]]
$typeDefs += [System.Management.Automation.ExtendedTypeDefinition]::new('Pester.Option', $ViewDef)
# Export all formatdata
Export-FormatData -InputObject $typeDefs -Path "$PSScriptRoot/bin/PesterConfiguration.Format.ps1xml"
}
if (-not $PSBoundParameters.ContainsKey('Inline')) {
# Force inlining by env variable, build.ps1 is used in
# multiple places and passing the $inline everywhere is
# difficult.
# Only read this option here. Don't write it.
if ($env:PESTER_BUILD_INLINE -eq '1') {
$Inline = $true
}
else {
$Inline = $false
}
}
else {
# We provided Inline explicitly, write the option. This assumes that
# you don't use -Inline:$false in any of the test scripts, otherwise the
# test script would reset the option incorrectly.
$env:PESTER_BUILD_INLINE = [string][int][bool] $Inline
}
$null = New-Item "$PSScriptRoot/bin" -ItemType Directory -Force
$script = @(
"$PSScriptRoot/src/functions/Pester.SafeCommands.ps1"
"$PSScriptRoot/src/Pester.Types.ps1"
"$PSScriptRoot/src/Pester.State.ps1"
"$PSScriptRoot/src/Pester.Utility.ps1"
"$PSScriptRoot/src/Pester.Runtime.ps1"
"$PSScriptRoot/src/TypeClass.ps1"
"$PSScriptRoot/src/Format.ps1"
# TODO: the original version of Format2 from assert, because it does not format strings and other stuff in Pester specific way. I've used this regex (Format-Collection|Format-Object|Format-Null|Format-Boolean|Format-ScriptBlock|Format-Number|Format-Hashtable|Format-Dictionary|Format-Nicely|Get-DisplayProperty|Get-ShortType|Format-Type), '$12' to replace in VS Code.
"$PSScriptRoot/src/Format2.ps1"
"$PSScriptRoot/src/Pester.RSpec.ps1"
"$PSScriptRoot/src/Main.ps1"
"$PSScriptRoot/src/functions/assert/*/*"
"$PSScriptRoot/src/functions/assertions/*"
"$PSScriptRoot/src/functions/*"
"$PSScriptRoot/src/Module.ps1"
)
$files = Get-ChildItem $script -File | Select-Object -Unique
$sb = [System.Text.StringBuilder]""
if (-not $inline) {
# define this on the top of the module to skip
# the code that is wrapped in this if in different source files
$null = $sb.AppendLine('$PESTER_BUILD=1')
}
foreach ($f in $files) {
if ($inline) {
$lines = Get-Content $f
$relativePath = ($f.FullName -replace ([regex]::Escape($PSScriptRoot))).TrimStart('\').TrimStart('/')
$null = $sb.AppendLine("# file $relativePath")
$noBuild = $false
foreach ($l in $lines) {
# when inlining the code skip everything wrapped in # PESTER_BUILD, # end PESTER_BUILD
#
if ($l -match '^.*#\s*PESTER_BUILD') {
# start skipping lines
$noBuild = $true
}
if (-not $noBuild) {
# append lines when we are not skipping them
$null = $sb.AppendLine($l)
}
if ($l -match '^.*#\s*end\s*PESTER_BUILD') {
# stop skipping lines
$noBuild = $false
}
}
}
else {
# when not inlining just dot-source the file
if ($f.FullName -notlike "*.ps1") {
throw "$($f.FullName) is not a ps1 file"
}
$null = $sb.AppendLine(". '$($f.FullName)'")
}
}
$sb.ToString() | Set-Content "$PSScriptRoot/bin/Pester.psm1" -Encoding UTF8
$powershell = Get-Process -Id $PID | Select-Object -ExpandProperty Path
if ($Load) {
& $powershell -c "'Load: ' + (Measure-Command { Import-Module '$PSScriptRoot/bin/Pester.psd1' -ErrorAction Stop}).TotalMilliseconds + 'ms'"
if (0 -ne $LASTEXITCODE) {
throw "load failed!"
}
}
if ($Import) { Import-Module "$PSScriptRoot/bin/Pester.psd1" -Force }