forked from microsoft/AL-Go-Actions
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathIncrementVersionNumber.psm1
295 lines (260 loc) · 11.4 KB
/
IncrementVersionNumber.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
291
292
293
294
295
<#
.Synopsis
Changes a version setting value in a settings file.
.Description
Changes a version setting value in a settings file.
If the setting does not exist in the settings file, the function does nothing, unless the Force parameter is specified.
.Parameter settingsFilePath
Path to a JSON file containing the settings.
.Parameter settingName
Name of the setting to change. The setting must be a version number.
.Parameter newValue
New value of the setting. Allowed values are: +1 (increment major version number), +0.1 (increment minor version number), or a version number in the format Major.Minor (e.g. 1.0 or 1.2
.Parameter Force
If specified, the function will create the setting if it does not exist in the settings file.
#>
function Set-VersionInSettingsFile {
param(
[Parameter(Mandatory = $true)]
[string] $settingsFilePath,
[Parameter(Mandatory = $true)]
[string] $settingName,
[Parameter(Mandatory = $true)]
[string] $newValue,
[switch] $Force
)
#region Validate parameters
if (-not (Test-Path $settingsFilePath)) {
throw "Settings file ($settingsFilePath) not found."
}
Write-Host "Reading settings from $settingsFilePath"
try {
$settingsJson = Get-Content $settingsFilePath -Encoding UTF8 -Raw | ConvertFrom-Json
}
catch {
throw "Settings file ($settingsFilePath) is malformed: $_"
}
$settingExists = [bool] ($settingsJson.PSObject.Properties.Name -eq $settingName)
if ((-not $settingExists) -and (-not $Force)) {
Write-Host "Setting $settingName not found in $settingsFilePath"
return
}
# Add the setting if it does not exist
if (-not $settingExists) {
$settingsJson | Add-Member -MemberType NoteProperty -Name $settingName -Value $null
}
$oldVersion = [System.Version] $settingsJson.$settingName
# Validate new version value
if ($newValue.StartsWith('+')) {
# Handle incremental version number
# Defensive check. Should never happen.
$allowedIncrementalVersionNumbers = @('+1', '+0.1')
if (-not $allowedIncrementalVersionNumbers.Contains($newValue)) {
throw "Incremental version number $newValue is not allowed. Allowed incremental version numbers are: $($allowedIncrementalVersionNumbers -join ', ')"
}
# Defensive check. Should never happen.
if($null -eq $oldVersion) {
throw "The setting $settingName does not exist in the settings file. It must exist to be able to increment the version number."
}
}
else {
# Handle absolute version number
$versionNumberFormat = '^\d+\.\d+$' # Major.Minor
if (-not ($newValue -match $versionNumberFormat)) {
throw "Version number $newValue is not in the correct format. The version number must be in the format Major.Minor (e.g. 1.0 or 1.2)"
}
}
#endregion
$versionNumbers = @() # an array to hold the version numbers: major, minor, build, revision
switch($newValue) {
'+1' {
# Increment major version number
$versionNumbers += $oldVersion.Major + 1
$versionNumbers += 0
}
'+0.1' {
# Increment minor version number
$versionNumbers += $oldVersion.Major
$versionNumbers += $oldVersion.Minor + 1
}
default {
# Absolute version number
$versionNumbers += $newValue.Split('.')
}
}
# Include build and revision numbers if they exist in the old version number
if ($oldVersion -and ($oldVersion.Build -ne -1)) {
$versionNumbers += 0 # Always set the build number to 0
if ($oldVersion.Revision -ne -1) {
$versionNumbers += 0 # Always set the revision number to 0
}
}
# Construct the new version number. Cast to System.Version to validate if the version number is valid.
$newVersion = [System.Version] "$($versionNumbers -join '.')"
if($newVersion -lt $oldVersion) {
throw "The new version number ($newVersion) is less than the old version number ($oldVersion). The version number must be incremented."
}
if($newVersion -eq $oldVersion) {
Write-Host "The setting $settingName is already set to $newVersion in $settingsFilePath"
return
}
if($null -eq $oldVersion) {
Write-Host "Setting setting $settingName to $newVersion in $settingsFilePath"
}
else {
Write-Host "Changing $settingName from $oldVersion to $newVersion in $settingsFilePath"
}
$settingsJson.$settingName = $newVersion.ToString()
$settingsJson | Set-JsonContentLF -Path $settingsFilePath
}
<#
.Synopsis
Checks if a setting exists in a settings file.
.Description
Checks if a setting exists in a settings file.
.Parameter settingsFilePath
Path to a JSON file containing the settings.
.Parameter settingName
Name of the setting to check.
#>
function Test-SettingExists {
param(
[Parameter(Mandatory = $true)]
[string] $settingsFilePath,
[Parameter(Mandatory = $true)]
[string] $settingName
)
if (-not (Test-Path $settingsFilePath)) {
throw "Settings file ($settingsFilePath) not found."
}
Write-Host "Reading settings from $settingsFilePath"
try {
$settingsJson = Get-Content $settingsFilePath -Encoding UTF8 -Raw | ConvertFrom-Json
}
catch {
throw "Settings file ($settingsFilePath) is malformed: $_"
}
$settingExists = [bool] ($settingsJson.PSObject.Properties.Name -eq $settingName)
return $settingExists
}
<#
.Synopsis
Changes the version number of a project.
.Description
Changes the version number of a project.
The version number is changed in the project settings file (value for 'repoVersion') and in the app.json files of all apps in the project, as well as all references to the apps in the dependencies of the app.json files.
.Parameter baseFolder
Base folder of the repository.
.Parameter project
Name of the project (relative to the base folder).
.Parameter newValue
New version number. If the version number starts with a +, the new version number will be added to the old version number. Else the new version number will replace the old version number.
#>
function Set-VersionInAppManifests($projectPath, $projectSettings, $newValue) {
# Check if the project uses repoVersion versioning strategy
$useRepoVersion = (($projectSettings.PSObject.Properties.Name -eq "versioningStrategy") -and (($projectSettings.versioningStrategy -band 16) -eq 16))
if ($useRepoVersion) {
$newValue = $projectSettings.repoVersion
}
$allAppFolders = @($projectSettings.appFolders) + @($projectSettings.testFolders) + @($projectSettings.bcptTestFolders)
# Set version in app.json files
$allAppFolders | ForEach-Object {
$appFolder = Join-Path $projectPath $_
$appJson = Join-Path $appFolder "app.json"
Set-VersionInSettingsFile -settingsFilePath $appJson -settingName 'version' -newValue $newValue
}
}
<#
.Synopsis
Changes the version number of dependencies in app.json files.
.Description
Changes the version number of dependencies in app.json files.
The version number of the dependencies is changed to the version number of the app that the dependency refers to. If the app is not found, the version number of the dependency is not changed.
.Parameter appFolders
Array of paths to the app folders. Each app folder must contain an app.json file. The apps are used to get the version number of the dependencies.
#>
function Set-DependenciesVersionInAppManifests {
param(
[Parameter(Mandatory = $true)]
[string[]] $appFolders
)
# Get all apps info: app ID and app version
$appsInfos = @($appFolders | ForEach-Object {
$appJson = Join-Path $_ "app.json"
$app = Get-Content -Path $appJson -Encoding UTF8 -Raw | ConvertFrom-Json
return [PSCustomObject]@{
Id = $app.id
Version = $app.version
}
})
# Update dependencies in app.json files
$appFolders | ForEach-Object {
$appJsonPath = Join-Path $_ "app.json"
$appJson = Get-Content -Path $appJsonPath -Encoding UTF8 -Raw | ConvertFrom-Json
$dependencies = $appJson.dependencies
$dependencies | ForEach-Object {
$dependency = $_
$appInfo = $appsInfos | Where-Object { $_.Id -eq $dependency.id }
if ($appInfo) {
Write-Host "Updating dependency app $($dependency.id) in $appJsonPath from $($dependency.version) to $($appInfo.Version)"
$dependency.version = $appInfo.Version
}
}
$appJson | Set-JsonContentLF -Path $appJsonPath
}
}
<#
.Synopsis
Sets the version number of a Power Platform solution.
.Description
Sets the version number of a Power Platform solution.
The version number is changed in the Solution.xml file of the Power Platform solution.
.Parameter powerPlatformSolutionPath
Path to the Power Platform solution.
.Parameter newValue
New version number. If the version number starts with a +, the new version number will be added to the old version number. Else the new version number will replace the old version number.
Allowed values are: +1 (increment major version number), +0.1 (increment minor version number), or a full version number (e.g. major.minor.build.revision).
#>
function Set-PowerPlatformSolutionVersion {
param(
[Parameter(Mandatory = $true)]
[string] $powerPlatformSolutionPath,
[Parameter(Mandatory = $true)]
[string] $newValue
)
Write-Host "Updating Power Platform solution version"
$files = Get-ChildItem -Path $powerPlatformSolutionPath -filter 'Solution.xml' -Recurse -File | Where-Object { $_.Directory.Name -eq "other" }
if (-not $files) {
Write-Host "Power Platform solution file not found"
}
else {
foreach ($file in $files) {
$xml = [xml](Get-Content -Encoding UTF8 -Path $file.FullName)
if ($newValue.StartsWith('+')) {
# Increment version
$versionNumbers = $xml.SelectNodes("//Version")[0].InnerText.Split(".")
switch($newValue) {
'+1' {
# Increment major version number
$versionNumbers[0] = "$(([int]$versionNumbers[0])+1)"
}
'+0.1' {
# Increment minor version number
$versionNumbers[1] = "$(([int]$versionNumbers[1])+1)"
}
default {
throw "Unexpected version number $newValue"
}
}
$newVersion = [string]::Join(".", $versionNumbers)
}
else {
$newVersion = $newValue
}
Write-Host "Updating $($file.FullName) with new version $newVersion"
$xml.SelectNodes("//Version")[0].InnerText = $newVersion
$xml.Save($file.FullName)
}
}
}
Export-ModuleMember -Function Set-VersionInSettingsFile, Set-VersionInAppManifests, Set-DependenciesVersionInAppManifests, Set-PowerPlatformSolutionVersion, Test-SettingExists