resource/script/Build.ps1
[CmdletBinding(DefaultParameterSetName = 'IncrementBuild')] param( [switch] $IncrementMajor, [switch] $IncrementMinor, [switch] $IncrementRevision ) Begin { $EPB = $ErrorActionPreference try { $ErrorActionPreference = 'Stop' $BuildDefinition = (Join-Path -Path $PSScriptRoot -ChildPath '.\Build.psd1' -Resolve) # 1. Load the Build Settings Write-Verbose "Loading Build settings from $BuildDefinition" $Build = Import-PowerShellDataFile -Path $BuildDefinition Write-Verbose "Building into $($Build.Destination)" # 2. Populate build settings with calculated values Write-Verbose "Populating build settings" $Build.CurrentYMD = Get-Date -Format 'yyyy.MM.dd' # 2.1 Create the destination directory (where the module code is placed) if(-not (Test-Path -Path $Build.Destination)) { Write-Debug "Creating the destination directory $($Build.Destination)" New-Item -Path $Build.Destination -ItemType Container | Out-null } # 2.2 Resolve source and destination root directories (get their fully qualified path) $Build.Destination = Join-Path -Path $PSScriptRoot -ChildPath $Build.Destination -Resolve $Build.Source = Join-Path -Path $PSScriptRoot -ChildPath $Build.Source -Resolve Write-Debug "Source: $($Build.Source)" Write-Debug "Destination: $($Build.Destination)" # 2.3 Add directory structure if required @('init','tools','class','public','private') | Foreach-Object { if(-not (Test-Path (Join-Path -Path $Build.Source -ChildPath $_) )) { Write-Debug "Creating the directory $(Join-Path -Path $Build.Source -ChildPath $_)" New-Item -Path $Build.Source -Name $_ -ItemType Directory | Out-Null } } # 2.4 Load and manage the build number $Build.VersionPath = Join-Path -Path $PSScriptRoot -ChildPath "Version.psd1" if(-not (Test-Path -Path $Build.VersionPath)) { Write-Debug "Creating the version file $($Build.VersionPath)" $Item = New-Item -Path $Build.VersionPath -ItemType File '@{Major=0;Minor=0;Build=0;Revision=0}' | Set-Content $Item } $Version = Import-PowerShellDataFile -Path $Build.VersionPath if($IncrementMajor.IsPresent -and $IncrementMajor) { $Version.Major ++ $Version.Minor = 0 $Version.Revision = 0 } elseif($IncrementMinor.IsPresent -and $IncrementMinor) { $Version.Minor ++ $Version.Revision = 0 } elseif($IncrementRevision.IsPresent -and $IncrementRevision) { $Version.Revision ++ } $Version.Build ++ $Build.Version = [version]::new("$($Version.Major).$($Version.Minor).$($Version.Build).$($Version.Revision)") Write-Verbose "Version: $($Build.Version)" # 2.5 Create the versionned destination directory $Destination = Join-Path (Join-Path $Build.Destination $Build.Name) $Build.Version if((Test-Path -Path $Destination)) { Write-Warning 'Destination exists!' Remove-Item -Force -Path $Destination -Confirm } Write-Debug "Creating the versionned destination directory $Destination" New-Item -Path $Destination -ItemType Container | Out-null $Build.Destination = Resolve-Path -Path $Destination # 2.5.2 Create the tools subdirectory in the module Write-Debug "Creating the tools subdirectory in the module" New-Item -Path (Join-Path $Destination 'tools') -ItemType Container | Out-null # 2.6 Calculate the fully qualified path of the Module's code file $Build.ManifestPath = Join-Path -Path $Build.Destination -ChildPath "$($Build.Name).psd1" Write-Debug "Creating the module's manifest file $($Build.ManifestPath)" New-Item -Path $Build.ManifestPath -ItemType File | Out-null # 2.7 Calculate the fully qualified path of the Module's manifest $Build.ModulePath = Join-Path -Path $Build.Destination -ChildPath "$($Build.Name).psm1" Write-Debug "Creating the module's code file $($Build.ModulePath)" New-Item -Path $Build.ModulePath -ItemType File | Out-null # 2.8 Calculate the fully qualified path of the Module's init script file (this script will be executed, in the client's scope while module is getting loaded) $Build.InitScriptPath = Join-Path -Path $Build.Destination -ChildPath "tools\init.ps1" Write-Debug "Creating the module's init script file $($Build.InitScriptPath)" New-Item -Path $Build.InitScriptPath -ItemType File | Out-null # 2.9 Calculate the fully qualified path of the Module's class script file (there are some limitation for classes exposed in this way, see https://stackoverflow.com/questions/31051103/how-to-export-a-class-in-a-powershell-v5-module) $Build.ClassScriptPath = Join-Path -Path $Build.Destination -ChildPath "tools\classes.ps1" Write-Debug "Creating the module's class script file $($Build.ClassScriptPath)" New-Item -Path $Build.ClassScriptPath -ItemType File | Out-null } catch { Write-Error $_ } finally { $ErrorActionPreference = $EPB } } End { Write-Verbose "Incrementing the build number" # Increment the build number, and save Clear-Content -Path $Build.VersionPath '@{' | Set-Content -Path $Build.VersionPath " Major = $($Build.Version.Major)" | Add-Content -Path $Build.VersionPath " Minor = $($Build.Version.Minor)" | Add-Content -Path $Build.VersionPath " Revision = $($Build.Version.Revision)" | Add-Content -Path $Build.VersionPath " Build = $($Build.Version.Build)" | Add-Content -Path $Build.VersionPath '}' | Add-Content -Path $Build.VersionPath Write-Debug "New version number: $($Build.Version)" } Process { # Stores the list of public functions exposed by the module $FunctionsToExport = @() # Stores the list of TypeData (for module's classes) exposed by the module $TypesToExport = @() # Stores the list of FormatData (for module's classes) exposed by the module $FormatsToExport = @() # Load dependencies INTO THE BUILD process, if required Write-Debug "Processing .Net dependencies" $Build.AssemblyDependencies | ForEach-Object { $AssemblyName = $_ -replace '\.[^\.]+$' $ImportScript = @" if (-not ("$($_)" -as [Type])) { Add-Type -Assembly $($AssemblyName) } "@ $ImportScript | Add-Content -Path $Build.InitScriptPath } # Register module dependencies Write-Debug "Processing module dependencies" $Build.ModuleDependencies | ForEach-Object { "#Requires -Module $($_)" | Add-Content -Path $Build.ModulePath # Add the constraint also to the init script "#Requires -Module $($_)" | Add-Content -Path $Build.InitScriptPath } # 1. Build the Module's code file Write-Verbose "Building the module's code file" # 1.0 Copy the README file Get-ChildItem -Path $Build.Source -File -Filter '*.md' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.md' ) { Write-Debug "Copying the README file $($_)" $_ | Copy-Item -Destination $Build.Destination } } # 1.1 Add code on top of the module, before any Function definition Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'init') -File -Filter '*.ps1' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.ps1' ) { Write-Debug "Adding code from $($_)" Get-Content -Path $_.FullName | Add-Content -Path $Build.ModulePath } } # 1.2 Add private functions (not exposed by the module) Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'private') -File -Filter '*.ps1' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.ps1' ) { $FunctionName = $_.Name -replace '\.ps1$' Write-Debug "Adding private function $($FunctionName) from $($_)" "Function $($FunctionName) {" | Add-Content -Path $Build.ModulePath Get-Content -Path $_.FullName | Add-Content -Path $Build.ModulePath '}' | Add-Content -Path $Build.ModulePath } } # 1.3 Add public functions (exposed by the module) Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'public') -File -Filter '*.ps1' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.ps1' ) { $FunctionName = $_.Name -replace '\.ps1$' Write-Debug "Adding public function $($FunctionName) from $($_)" "Function $($FunctionName) {" | Add-Content -Path $Build.ModulePath Get-Content -Path $_.FullName | Add-Content -Path $Build.ModulePath '}' | Add-Content -Path $Build.ModulePath $FunctionsToExport += $FunctionName } } # 1.4 Add class definitions # 1.4.a Add class' code to the module's class file Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'class') -File -Filter '*.ps1' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.ps1' ) { Write-Debug "Adding class code from $($_)" Get-Content -Path $_.FullName | Add-Content -Path $Build.ClassScriptPath } } # # 1.4.b Add Init code to make the class modules available to the calling process when they use Import-Module # if( Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'class') -File -Filter '*.ps1' | Where-Object {[System.IO.Path]::GetExtension($_) -eq '.ps1' }) { # # "# using module $($Build.ModulePath)" | Add-Content -Path $Build.InitScriptPath # "# using module $($Build.Name)" | Add-Content -Path $Build.InitScriptPath # } # 1.4.b Copy "TypesData" file to the module's root directory Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'class') -File -Filter '*.types.ps1xml' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.ps1xml' ) { write-debug "Copying the TypesData file $($_)" $TypesToExport += $_.Name $_ | Copy-Item -Destination $Build.Destination } } # 1.4.b Copy "FormatData" file to the module's root directory Get-ChildItem -Path (Join-Path -Path $Build.Source -ChildPath 'class') -File -Filter '*.formats.ps1xml' | Foreach-Object { if([System.IO.Path]::GetExtension($_) -eq '.ps1xml' ) { write-debug "Copying the FormatData file $($_)" $FormatsToExport += $_.Name $_ | Copy-Item -Destination $Build.Destination } } # 1.5 Copy "resource" directory to the module's root directory if((Test-Path -Path (Join-Path -Path (Split-Path $Build.Source -Parent) -ChildPath 'resource') -PathType Container)) { Write-Debug "Copying the resource directory" Copy-Item -Recurse -LiteralPath (Join-Path -Path (Split-Path $Build.Source -Parent) -ChildPath 'resource') -Destination $Build.Destination } # 2. Build the Module's manifest Write-Verbose "Building the module's manifest" $ModuleManifest = $Build.ModuleSettings.Clone() $ModuleManifest.RootModule = "$($Build.Name).psm1" $ModuleManifest.ModuleVersion = $Build.Version $ModuleManifest.FunctionsToExport = $FunctionsToExport $ModuleManifest.ScriptsToProcess = @( "$($Build.InitScriptPath | Split-Path -Parent | Split-Path -Leaf)\$($Build.InitScriptPath | Split-Path -Leaf)" "$($Build.ClassScriptPath | Split-Path -Parent | Split-Path -Leaf)\$($Build.ClassScriptPath | Split-Path -Leaf)" ) if($TypesToExport.Length) { $ModuleManifest.TypesToProcess = $TypesToExport } if($FormatsToExport.Length) { $ModuleManifest.FormatsToProcess = $FormatsToExport } # Add the requirement to the Manifest $ModuleManifest.RequiredModules = $Build.ModuleDependencies # 3. Save the Module's manifest Write-Verbose "Saving the module's manifest $($Build.ManifestPath)" New-ModuleManifest @ModuleManifest -Path $Build.ManifestPath # Return the path where the module has been built Get-Item ($Build.ManifestPath | Split-Path) } |