Tests/Module.Tests.ps1
Set-StrictMode -Version Latest $TestScriptRoot = Split-Path $MyInvocation.MyCommand.Path -Parent $ModuleRoot = Resolve-Path "$TestScriptRoot\.." $ModuleManifest = "$ModuleRoot\AtomicTestHarnesses.psd1" Remove-Module [A]tomicTestHarnesses $Module = Import-Module $ModuleManifest -Force -ErrorAction Stop -PassThru $ExportedFunctionList = $Module.ExportedCommands.Keys Describe 'Module-wide tests' -Tags 'Module' { foreach ($FunctionName in $ExportedFunctionList) { Context "Required exported command naming scheme for $FunctionName" { It 'should have an "ATH" noun prefix' { $FunctionInfo = Get-Item -Path Function:$FunctionName $FunctionInfo.Noun.Substring(0, 3) | Should -BeExactly 'ATH' } -TestCases @(@{ FunctionName = $FunctionName }) } Context "Pester test checks for $FunctionName" { It 'should have test code written for the exported function' { $FunctionInfo = Get-Item -Path Function:$FunctionName $FunctionFilePath = $FunctionInfo.ScriptBlock.File $FunctionFileInfo = Get-Item -Path $FunctionFilePath $FunctionDirectory = $FunctionFileInfo.DirectoryName $FunctionFileBaseName = $FunctionFileInfo.BaseName # The filename without the extension $TestFilePath = Join-Path -Path $FunctionDirectory -ChildPath "$FunctionFileBaseName.Tests.ps1" Test-Path -Path $TestFilePath -PathType Leaf | Should -BeTrue $TestFileInfo = Get-Item -Path $TestFilePath $TestFileInfo | Should -Not -BeNullOrEmpty # For now, just validate that it isn't an empty file. # Until Pester can better discover tests without executing them, we'll hold off on incorporating more thorough logic. $TestFileInfo.Length | Should -BeGreaterThan 0 } -TestCases @(@{ FunctionName = $FunctionName }) } Context "Comment-based help for: $FunctionName" { $Script:Help = Get-Help -Name $FunctionName -Full # Parse the function using AST $Script:AST = [Management.Automation.Language.Parser]::ParseInput((Get-Content Function:$FunctionName), [ref]$null, [ref]$null) It 'should contain a .SYNOPSIS block' { $Help = Get-Help -Name $FunctionName -Full $Help.Synopsis | Should -Not -BeNullOrEmpty } -TestCases @(@{ FunctionName = $FunctionName }) It 'should have a .SYNOPSIS block that ends with a well-formatted attack technique ID' { $Help = Get-Help -Name $FunctionName -Full $Help.Synopsis.Split("`r`n")[-1] -match '^(?-i:Technique ID: )(?<TechniqueID>\S+) (?<TechniqueDescription>\(.+\))$' | Should -BeTrue } -TestCases @(@{ FunctionName = $FunctionName }) It 'should contain a .DESCRIPTION block' { $Help = Get-Help -Name $FunctionName -Full $Help.Description | Should -Not -BeNullOrEmpty } -TestCases @(@{ FunctionName = $FunctionName }) # Examples It 'should contain at least one .EXAMPLE block' { $Help = Get-Help -Name $FunctionName -Full @($Help.Examples.Example.Code).Count | Should -BeGreaterThan 0 } -TestCases @(@{ FunctionName = $FunctionName }) It 'should contain a matching number of .PARAMETER blocks for all defined parameters and be documented accordingly' { $Help = Get-Help -Name $FunctionName -Full $HelpParameters = @($Help.Parameters.Parameter) $AST = [Management.Automation.Language.Parser]::ParseInput((Get-Content Function:$FunctionName), [ref]$null, [ref]$null) $ASTParameters = @($AST.ParamBlock.Parameters.Name.Variablepath.Userpath) $NamedArgs = try { $AST.ParamBlock.Attributes.NamedArguments } catch { $null } if ($NamedArgs -and $NamedArgs.ArgumentName -contains 'SupportsShouldProcess') { $Count = $ASTParameters.Count + 2 # Accounting for -WhatIf and -Confirm } else { $Count = $ASTParameters.Count } $HelpParameters.Count | Should -Be $Count # Each defined parameter in help should have a defined description $HelpParameters | ForEach-Object { if ($ASTParameters -contains $_.Name) { $_.Description | Should -Not -BeNullOrEmpty } } } -TestCases @(@{ FunctionName = $FunctionName }) } } } |