MetaNull.ModuleMaker.psm1
#Requires -Module Microsoft.PowerShell.PSResourceGet # Module Constants Set-Variable INSIDE_MODULEMAKER_MODULE -option Constant -value $true Function Get-ResourceDirectory { <# .SYNOPSIS Get the location of the Module's ResourceDirectory .EXAMPLE $Item = Get-ResourceDirectory #> [CmdletBinding()] [OutputType([PSCustomObject])] param() Process { if((Get-Variable INSIDE_MODULEMAKER_MODULE -ErrorAction SilentlyContinue)) { #INSIDE_MODULEMAKER_MODULE is a constant defined in the module #If it is set, then the script is run from a loaded module, PSScriptRoot = Directory of the psm1 Get-Item (Join-Path $PSScriptRoot resource) } else { #Otherwise, the script was probably called from the command line, PSScriptRoot = Directory /source/private Get-Item (Join-Path (Split-Path (Split-Path $PSScriptRoot)) resource) } } } Function Test-JoinPath { <# .SYNOPSIS Test if the output of Join-Path exists .EXAMPLE Test-JoinPath -Path $env:TEMP -Name toto .EXAMPLE JoinPath -Path $env:TEMP -Name toto | Test-JoinPath #> [CmdletBinding(DefaultParameterSetName = 'LiteralPathAny')] [OutputType([PSCustomObject])] param( [Parameter(Mandatory,Position=0,ValueFromPipeline,ParameterSetName = 'LiteralPathAny')] [Parameter(Mandatory,Position=0,ValueFromPipeline,ParameterSetName = 'LiteralPathDirectory')] [Parameter(Mandatory,Position=0,ValueFromPipeline,ParameterSetName = 'LiteralPathFile')] [string] $LiteralPath, [Parameter(Mandatory,Position=0,ParameterSetName = 'PathAny')] [Parameter(Mandatory,Position=0,ParameterSetName = 'PathDirectory')] [Parameter(Mandatory,Position=0,ParameterSetName = 'PathFile')] [string] $Path, [Parameter(Mandatory,Position=0,ParameterSetName = 'PathAny')] [Parameter(Mandatory,Position=0,ParameterSetName = 'PathDirectory')] [Parameter(Mandatory,Position=0,ParameterSetName = 'PathFile')] [string] $Name, [Parameter(Mandatory,Position=1,ParameterSetName = 'PathDirectory')] [Parameter(Mandatory,Position=1,ParameterSetName = 'LiteralPathDirectory')] [switch] $Directory, [Parameter(Mandatory,Position=1,ParameterSetName = 'PathFile')] [Parameter(Mandatory,Position=1,ParameterSetName = 'LiteralPathFile')] [switch] $File ) Process { if($Directory.IsPresent -and $Directory) { $PathType = 'Container' } elseif($File.IsPresent -and $File) { $PathType = 'Container' } else { $PathType = 'Any' } if($PsCmdlet.ParameterSetName -in 'LiteralPathAny','LiteralPathDirectory','LiteralPathFile') { return Test-Path -LiteralPath $LiteralPath -PathType $PathType } else { return Test-Path -LiteralPath (Join-Path -Path $Path -ChildPath $Name) -PathType $PathType } } } Function Test-ModuleDefinition { [CmdletBinding()] param( [Parameter(Mandatory = $false)] [AllowEmptyString()] [AllowNull()] [Alias('Path','DataFile')] [string] $ModuleDefinitionPath ) End { $EAB = $ErrorActionPreference try { $ErrorActionPreference = 'Stop' if(-not $ModuleDefinitionPath) { Write-Debug "Module definition path is null, empty or not provided" return $false } if(-not (Test-Path -Path $ModuleDefinitionPath -PathType Leaf)) { Write-Debug "Module definition file does not exist or is not a file, for path: $ModuleDefinitionPath" return $false } Write-Debug "Importing Module definition from: $ModuleDefinitionPath" $ModuleDefinition = Import-PowerShellDataFile -Path $ModuleDefinitionPath if($null -eq $ModuleDefinition.Name -or $null -eq $ModuleDefinition.ModuleSettings -or $null -eq $ModuleDefinition.ModuleSettings.GUID) { Write-Debug "Module definition is invalid: `$_.Name or `$_.GUID are missing" return $false } if($ModuleDefinition.Name -eq '%%MODULE_NAME%%' -or $ModuleDefinition.ModuleSettings.GUID -eq '%%MODULE_GUID%%') { Write-Debug "Module definition is invalid: `$_.Name or `$_.GUID are not initialized" return $false } if($ModuleDefinition.Name -eq [string]::empty -or $ModuleDefinition.ModuleSettings.GUID -eq [string]::empty) { Write-Debug "Module definition is invalid: `$_.Name or `$_.GUID are empty" return $false } Write-Debug "Module definition is valid. Module: $($ModuleDefinition.Name)" $ModulePath = $ModuleDefinitionPath | Split-Path -Parent Write-Debug "Testing Module structure from: $ModulePath" if(-not (Test-Path -Path (Join-Path $ModulePath source) -PathType Container)) { Write-Debug "Module source directory does not exist" return $false } if(-not (Test-Path -Path (Join-Path $ModulePath test) -PathType Container)) { Write-Debug "Module test directory does not exist" return $false } if(-not (Test-Path -Path (Join-Path $ModulePath Build.ps1) -PathType Leaf)) { Write-Debug "Module's Build script does not exist" return $false } if(-not (Test-Path -Path (Join-Path $ModulePath Publish.ps1) -PathType Leaf)) { Write-Debug "Module's Publish script does not exist" return $false } Write-Debug "Module structure is valid. Path: $($ModulePath)" return $true } catch { Write-Verbose "$($_.Exception.Message)" # Swallow exception, to permit returninbg $false instead of throwing } finally { $ErrorActionPreference = $EAB } return $false } } Function Invoke-BuildModule { [CmdletBinding(DefaultParameterSetName = 'IncrementBuild')] param( [Parameter(Mandatory,ValueFromPipeline)] [ValidateScript({ Test-ModuleDefinition -ModuleDefinitionPath $_ })] [Alias('Path','DataFile')] [string] $ModuleDefinitionPath, [switch] $IncrementMajor, [switch] $IncrementMinor, [switch] $IncrementRevision ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { $ModulePath = $ModuleDefinitionPath | Split-Path -Parent $ScriptPath = Join-Path $ModulePath Build.ps1 -Resolve Push-Location (Split-Path $ScriptPath -Parent) $ScriptArguments = $args | Where-Object { $_ -ne $ModuleDefinitionPath } if($ScriptArguments -eq $null) { $ScriptArguments = @() } . $ScriptPath @ScriptArguments $ModuleDefinitionPath | Write-Output } finally { Pop-Location $ErrorActionPreference = $BackupErrorActionPreference } } } Function Invoke-PublishModule { [CmdletBinding(DefaultParameterSetName = 'psgallery')] param( [Parameter(Mandatory,ValueFromPipeline)] [ValidateScript({ Test-ModuleDefinition -ModuleDefinitionPath $_ })] [Alias('Path','DataFile')] [string] $ModuleDefinitionPath, [Parameter(Mandatory = $true, ParameterSetName = 'custom')] [string] $RepositoryUri, [Parameter(Mandatory = $true, ParameterSetName = 'custom')] [string] $RepositoryName, [Parameter(Mandatory = $false)] [string] $VaultName = 'MySecretVault', [Parameter(Mandatory = $false)] [string] $SecretName = 'PSGalleryCredential', [Parameter(Mandatory = $false)] [Switch] $PromptSecret, [Parameter(Mandatory = $false)] [Switch] $PersonalAccessTokenAsString ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { $ModulePath = $ModuleDefinitionPath | Split-Path -Parent $ScriptPath = Join-Path $ModulePath Publish.ps1 -Resolve Push-Location (Split-Path $ScriptPath -Parent) $ScriptArguments = $args | Where-Object { $_ -ne $ModuleDefinitionPath } if($ScriptArguments -eq $null) { $ScriptArguments = @() } . $ScriptPath @ScriptArguments $ModuleDefinitionPath | Write-Output } finally { Pop-Location $ErrorActionPreference = $BackupErrorActionPreference } } } Function Invoke-TestModule { [CmdletBinding()] param( [Parameter(Mandatory,ValueFromPipeline)] [ValidateScript({ Test-ModuleDefinition -ModuleDefinitionPath $_ })] [Alias('Path','DataFile')] [string] $ModuleDefinitionPath ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { $ModulePath = $ModuleDefinitionPath | Split-Path -Parent $ScriptPath = Join-Path $ModulePath Build.ps1 -Resolve Push-Location (Split-Path $ScriptPath -Parent) Invoke-Pester -Path . -OutputFile .\testresults.xml -OutputFormat NUnitXml -CodeCoverageOutputFile .\coverage.xml -PassThru $ModuleDefinitionPath | Write-Output } finally { Pop-Location $ErrorActionPreference = $BackupErrorActionPreference } } } Function New-Module { <# .SYNOPSIS Create an empty Powershell Module .DESCRIPTION Create an empty Powershell Module with the following structure: - ModuleName - Build.psd1 # Contains the module's configuration (automatically generated by New-Module) - Version.psd1 # Contains the module's version (automatically generated during Build) - Build.ps1 # A script that builds the module - Publish.ps1 # A script that publishes the built module to a repository such as PSGallery - source # Contains the module's source code - public # Contains public/exposed functions - Verb-Name.ps1 # Defines the public function "Verb-Name" (e.g.: "Get-Something") - private # Contains private functions (accessible only from within the module) - Verb-Name.ps1 # Defines the private function "Verb-Name" (e.g.: "Invoke-Something") - init # Contains module initialization code - Init.ps1 # Contains module initialization code, such as definition of Constants, etc. - class # Contains .net classes exposed by the module - ClassName.cs # Defines the class "ClassName" - test # Contains the module's tests - public # Contains tests for public/exposed functions - Verb-Name.Tests.ps1 # Contains tests for the public function "Verb-Name" (e.g.: "Get-Something") - private # Contains tests for private functions - Verb-Name.Tests.ps1 # Contains tests for the private function "Verb-Name" (e.g.: "Invoke-Something") - resource # Contains resources contained in the module (e.g. data files that are accessible to the module) .PARAMETER LiteralPath The path to the directory where the module will be created. E.g: $env:TEMP .PARAMETER Name The name of the module. E.g.: MyModule .PARAMETER Description The description of the module. E.g.: "A module that does something" .PARAMETER Uri The URI of the module's documentation or repository. E.g.: "https://www.test.com/mymodule" .PARAMETER Author The author of the module. E.g.: "Pascal Havelange" .PARAMETER Vendor The vendor/company of the authors. E.g.: "MetaNull" .PARAMETER Copyright The copyright statement of the module. E.g.: "(c) 2025" .PARAMETER ModuleDependencies An array of module dependencies. E.g.: @("PackageManagement", "PowerShellGet") The module automatically adds the dependencies to the module's configuration file. .PARAMETER AssemblyDependencies An array of assembly dependencies. E.g.: @("System.Net.WebUtility", "System.Management.Automation.PSCredential") The module automatically adds the dependencies to the module's configuration file. .PARAMETER Force If the module directory already exists, overwrite it. .OUTPUTS Returns a [System.IO.FileInfo] object representing the module's configuration file (Build.psd1) .EXAMPLE # Create a new 'MyModule' module in the $env:TEMP directory $Module = New-Module -Path $env:TEMP -Name MyModule # Create a new public function 'Get-Something' in the module, and a test for it $Module | New-Function -Public -Verb Get -Name Something $Module | Invoke-Build $Module | Invoke-Publish #> [CmdletBinding()] [OutputType([System.IO.FileInfo])] param( [Parameter(Mandatory)] [ValidateScript({ Test-Path -Path $_ -PathType Container })] [Alias('Path')] [string] $LiteralPath, [Parameter(Mandatory)] [ValidateScript({ $_ -match '^[a-zA-Z][a-zA-Z0-9\._-]*$' })] [string] $Name, [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyString()] [string] $Description = $null, [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyString()] [ValidateScript({ try { if($null -eq $_ -or [string]::empty -eq $_ -or [System.Uri]::new($_)) { return $true } } catch { # Swallow exception, to permit returninbg $false instead of throwing } return $false })] [string] $Uri = 'https://www.test.com/mymodule', [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyString()] [string] $Author = ([System.Security.Principal.WindowsIdentity]::GetCurrent().Name), [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyString()] [string] $Vendor = 'Unknown', [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyString()] [string] $Copyright = "© $((Get-Date).Year). All rights reserved", [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyCollection()] [string[]] $ModuleDependencies, [Parameter(Mandatory = $false)] [AllowNull()] [AllowEmptyCollection()] [string[]] $AssemblyDependencies, [Parameter(Mandatory = $false)] [switch] $Force ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { $NewItemForce = $false $DirectoryPath = Join-Path $LiteralPath $Name if(Test-Path $DirectoryPath -PathType Any) { if($Force.IsPresent -and $Force) { "Target directory $DirectoryPath exists, overwriting." | Write-Warning $NewItemForce = $true } else { throw "Target directory $DirectoryPath exists" } } # Create the directories $RootDirectory = New-Item -Path $LiteralPath -Name $Name -ItemType Directory -Force:$NewItemForce $SourceDirectory = New-Item (Join-Path $RootDirectory source) -ItemType Directory -Force:$NewItemForce $TestDirectory = New-Item (Join-Path $RootDirectory test) -ItemType Directory -Force:$NewItemForce $SourcePublicDirectory = New-Item (Join-Path $SourceDirectory public) -ItemType Directory -Force:$NewItemForce $SourcePrivateDirectory = New-Item (Join-Path $SourceDirectory private) -ItemType Directory -Force:$NewItemForce $SourceInitDirectory = New-Item (Join-Path $SourceDirectory init) -ItemType Directory -Force:$NewItemForce $TestPublicDirectory = New-Item (Join-Path $TestDirectory public) -ItemType Directory -Force:$NewItemForce $TestPrivateDirectory = New-Item (Join-Path $TestDirectory private) -ItemType Directory -Force:$NewItemForce # Create the module's configuration file, script files, and module's sample source and test files if((Get-Variable INSIDE_MODULEMAKER_MODULE -ErrorAction SilentlyContinue)) { #INSIDE_MODULEMAKER_MODULE is a constant defined in the module #If it is set, then the script is run from a loaded module, PSScriptRoot = Directory of the psm1 $ResourceDirectory = Get-Item (Join-Path $PSScriptRoot resource) } else { #Otherwise, the script was probably called from the command line, PSScriptRoot = Directory /source/private $ResourceDirectory = Get-Item (Join-Path (Split-Path (Split-Path $PSScriptRoot)) resource) } Copy-Item $ResourceDirectory\script\*.ps1 $RootDirectory -Force:$NewItemForce Copy-Item $ResourceDirectory\data\*.psd1 $RootDirectory -Force:$NewItemForce Copy-Item -Path $ResourceDirectory\dummy\* -Destination $RootDirectory -Recurse -Force:$NewItemForce # Update Module's configuration $ManifestFile = Get-Item (Join-Path $RootDirectory Build.psd1) $ManifestFileContent = Get-Content -LiteralPath $ManifestFile $Replace = @{ '%%MODULE_GUID%%'= (New-Guid) '%%MODULE_NAME%%' = "$Name" '%%MODULE_DESCRIPTION%%' = "$Description" '%%MODULE_URI%%' = "$Uri" '%%MODULE_AUTHOR%%' = "$Author" '%%MODULE_VENDOR%%'= "$Vendor" '%%MODULE_COPYRIGHT%%' = "$Copyright" "%%ASSEMBLY_DEPENDENCIES%%" = $null "%%MODULE_DEPENDENCIES%%" = $null } if($AssemblyDependencies) { $Replace.'%%ASSEMBLY_DEPENDENCIES%%' = "'$($AssemblyDependencies -join "','")'" } if($ModuleDependencies) { $Replace.'"%%MODULE_DEPENDENCIES%%"' = "'$($ModuleDependencies -join "','")'" } $Replace.GetEnumerator() | Foreach-Object { if($_.Value) { $ManifestFileContent = $ManifestFileContent -replace "$($_.Key)","$($_.Value)" } else { $ManifestFileContent = $ManifestFileContent -replace "$($_.Key)" } } $ManifestFileContent | Set-Content -LiteralPath $ManifestFile -Force:$NewItemForce $ManifestFile | Write-Output } finally { $ErrorActionPreference = $BackupErrorActionPreference } } } Function New-ModuleFunction { <# .SYNOPSIS Create a new function in a module created by ModuleMaker .DESCRIPTION This function creates a new function in a module created by ModuleMaker. It creates a new function in the source directory and a new test file in the test directory. .PARAMETER ModuleDefinitionPath The path to the module definition file (Build.psd1). This file is created by New-Module and contains the module's metadata. .PARAMETER Name The name of the new function. This name must be a valid Powershell function name. .PARAMETER Private If this switch is present, the function will be created in the private directory. Otherwise, it will be created in the public directory. .OUTPUTS The path to the module definition file (Build.psd1) is returned. #> [CmdletBinding()] param( [Parameter(Mandatory,ValueFromPipeline)] [ValidateScript({ Test-ModuleDefinition -ModuleDefinitionPath $_ })] [Alias('Path','DataFile')] [string] $ModuleDefinitionPath, [Parameter(Mandatory)] [ValidateScript({ $_ -match '^[a-zA-Z][a-zA-Z0-9\._-]*$' })] [string] $Name, [Parameter(Mandatory=$false)] [switch] $Private ) Process { $BackupErrorActionPreference = $ErrorActionPreference $ErrorActionPreference = 'Stop' try { $DummyName = "Get-Dummy" $ModulePath = $ModuleDefinitionPath | Split-Path -Parent $ResourcePath = Get-ResourceDirectory $Visibility = if($Private) { 'private' } else { 'public' } $TargetSourceDirectory = Join-Path (Join-Path $ModulePath source) $Visibility -Resolve $TargetTestDirectory = Join-Path (Join-Path $ModulePath test) $Visibility -Resolve $TemplateFunction = Join-Path $ResourcePath "dummy\source\public\$DummyName.ps1" -Resolve $TargetFunction = Join-Path $TargetSourceDirectory "$Name.ps1" Copy-Item -Path $TemplateFunction -Destination $TargetFunction | Out-Null $Content = Get-Content -LiteralPath $TemplateFunction -Raw $Content -replace $DummyName, $Name | Set-Content -LiteralPath $TargetFunction $TemplateTest = Join-Path $ResourcePath "dummy\test\public\$DummyName.Tests.ps1" -Resolve $TargetTest = Join-Path $TargetTestDirectory "$Name.Tests.ps1" Copy-Item -Path $TemplateTest -Destination $TargetTest | Out-Null $Content = Get-Content -LiteralPath $TemplateTest -Raw $Content -replace $DummyName, $Name | Set-Content -LiteralPath $TargetTest $ModuleDefinitionPath | Write-Output } finally { $ErrorActionPreference = $BackupErrorActionPreference } } } |