Functions/GenXdev.Coding.PowerShell.Modules/Invoke-GenXdevScriptAnalyzer.ps1

<##############################################################################
Part of PowerShell module : GenXdev.Coding.PowerShell.Modules
Original cmdlet filename : Invoke-GenXdevScriptAnalyzer.ps1
Original author : René Vaessen / GenXdev
Version : 1.3066.2025
################################################################################
Copyright (c) René Vaessen / GenXdev
 
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
 
    http://www.apache.org/licenses/LICENSE-2.0
 
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
################################################################################>

<#
.SYNOPSIS
Invokes PowerShell Script Analyzer to analyze PowerShell scripts for compliance
and best practices.
 
.DESCRIPTION
This function provides a wrapper around PSScriptAnalyzer to analyze PowerShell
scripts for compliance issues, best practices violations, and potential bugs.
It supports both file-based analysis and string-based script analysis with
customizable rules and settings.
 
.PARAMETER ScriptFilePath
Specifies the path to the script file to analyze. This parameter is mandatory
when using the Path parameter set.
 
.PARAMETER ScriptDefinition
Specifies the script definition as a string to analyze. This parameter is
mandatory when using the Script parameter set.
 
.PARAMETER EnableExit
Specifies that the tool should exit on error during analysis.
 
.PARAMETER Fix
Enables automatic fixing of violations where possible.
 
.PARAMETER Recurse
Recursively processes files in subdirectories when analyzing a directory path.
 
.PARAMETER ReportSummary
Reports a summary after analysis showing the total number of issues found.
 
.EXAMPLE
Invoke-GenXdevScriptAnalyzer -Path "C:\Scripts\MyScript.ps1"
 
.EXAMPLE
Invoke-GenXdevScriptAnalyzer -ScriptDefinition "Get-Process | Where-Object {$_.Name -eq 'notepad'}"
 
.EXAMPLE
Invoke-GenXdevScriptAnalyzer -Path "C:\Scripts\" -Recurse -Fix
#>

function Invoke-GenXdevScriptAnalyzer {

    [CmdletBinding(DefaultParameterSetName = 'Path')]
    param (
        ###############################################################################
        [Parameter(
            Position = 0,
            Mandatory = $true,
            ParameterSetName = 'Path',
            HelpMessage = 'Specifies the path to the script file.'
        )]
        [Alias('Path', 'FullName')]
        [string] $ScriptFilePath,
        ###############################################################################
        [Parameter(
            Mandatory = $true,
            ParameterSetName = 'Script',
            HelpMessage = 'Specifies the script definition as a string.'
        )]
        [string] $ScriptDefinition,
        ###############################################################################
        [Parameter(
            Mandatory = $false,
            HelpMessage = 'Specifies that the tool should exit on error.'
        )]
        [switch] $EnableExit,
        ###############################################################################
        [Parameter(
            Mandatory = $false,
            HelpMessage = 'Enables automatic fixing of violations.'
        )]
        [switch] $Fix,
        ###############################################################################
        [Parameter(
            Mandatory = $false,
            HelpMessage = 'Recursively process files.'
        )]
        [switch] $Recurse,
        ###############################################################################
        [Parameter(
            Mandatory = $false,
            HelpMessage = 'Reports a summary after analysis.'
        )]
        [switch] $ReportSummary
        #
    )

    begin {

        # check if psscriptanalyzer module is loaded
        $loaded = Microsoft.PowerShell.Core\Get-Module -Name PSScriptAnalyzer -ErrorAction SilentlyContinue

        if (-not $loaded) {

            # define the folder path for psscriptanalyzer installation
            $folder = GenXdev.FileSystem\Expand-Path "$PSScriptRoot\..\..\..\..\PSScriptAnalyzer\1.24.0\"

            # define the path to the script analyzer dll
            $path = GenXdev.FileSystem\Expand-Path "$folder\PSv7\GenXdev.Coding.PowerShell.Modules.ScriptAnalyzer.dll"

            if (-not (Microsoft.PowerShell.Management\Test-Path -LiteralPath $folder)) {

                # install psscriptanalyzer module if folder does not exist
                $null = PowerShellGet\Install-Module -Name PSScriptAnalyzer -RequiredVersion 1.24.0 -Scope CurrentUser -Force
            }

            if (-not (Microsoft.PowerShell.Management\Test-Path -LiteralPath $path)) {

                # copy assets if dll does not exist
                $source = GenXdev.FileSystem\Expand-Path "$PSScriptRoot\..\..\assets\Modules\PSScriptAnalyzer\1.24.0\*"

                $null = GenXdev.FileSystem\Start-RoboCopy $source "$folder\"
            }

            # import the psscriptanalyzer module
            $null = Microsoft.PowerShell.Core\Import-Module PSScriptAnalyzer
        }

        # load script analyzer settings from configuration file
        $settingsPath = "$PSScriptRoot\PSScriptAnalyzerSettings.psd1"

        $settings = Microsoft.PowerShell.Utility\Invoke-Expression `
        ([System.IO.File]::ReadAllText($settingsPath))

        # output verbose information about loaded settings
        Microsoft.PowerShell.Utility\Write-Verbose `
            "Loaded PSScriptAnalyzer settings from: $settingsPath"
    }

    process {

        try {

            # initialize invocation parameters variable
            $invocationParams = $null

            # check which parameter set is being used
            if ($PSCmdlet.ParameterSetName -eq 'Path') {

                # configure parameters for path-based analysis
                $invocationParams = @{
                    Path                  = $ScriptFilePath
                    IncludeRule           = $settings.IncludeRules
                    ExcludeRule           = $settings.ExcludeRules
                    CustomRulePath        = $settings.CustomRulePath
                    IncludeDefaultRules   = $settings.IncludeDefaultRules
                    RecurseCustomRulePath = $settings.RecurseCustomRulePath
                    Recurse               = [bool]$Recurse
                    Fix                   = [bool]$Fix
                    EnableExit            = [bool]$EnableExit
                    ReportSummary         = [bool]$ReportSummary
                    Severity              = 'Warning'
                }

                # output verbose information about path analysis
                Microsoft.PowerShell.Utility\Write-Verbose `
                    "Analyzing script file: $ScriptFilePath"
            }
            else {

                # configure parameters for script definition analysis
                $invocationParams = @{
                    ScriptDefinition      = $ScriptDefinition
                    IncludeRule           = $settings.IncludeRules
                    ExcludeRule           = $settings.ExcludeRules
                    CustomRulePath        = $settings.CustomRulePath
                    IncludeDefaultRules   = $settings.IncludeDefaultRules
                    RecurseCustomRulePath = $settings.RecurseCustomRulePath
                    Recurse               = [bool]$Recurse
                    Fix                   = [bool]$Fix
                    EnableExit            = [bool]$EnableExit
                    ReportSummary         = [bool]$ReportSummary
                    Severity              = 'Warning'
                }

                # output verbose information about script definition analysis
                Microsoft.PowerShell.Utility\Write-Verbose `
                    'Analyzing script definition content'
            }

            # set retry count for handling transient errors
            [int] $retries = 3;

            while ($retries-- -gt 0) {

                try {

                    # invoke script analyzer with configured parameters
                    $results = @(PSScriptAnalyzer\Invoke-ScriptAnalyzer @invocationParams -Verbose:$false)

                    # output verbose information about analysis results
                    Microsoft.PowerShell.Utility\Write-Verbose `
                        "Analysis completed with $($results.Count) results"

                    break;
                }
                catch {

                    # output warning for script analyzer errors
                    Microsoft.PowerShell.Utility\Write-Warning `
                        "ScriptAnalyzer error: $($_.Exception.Message)"

                    if ($retries -eq 1) {

                        return
                    }

                    Microsoft.PowerShell.Utility\Start-Sleep 1
                }
            }

            # check if results were returned
            if ($results -and ($results.Length -gt 0)) {

                # process each result to enhance with additional information
                $results |
                    Microsoft.PowerShell.Core\ForEach-Object {

                        # convert result to hashtable for easier manipulation
                        $resultHashTable = $_ |
                            GenXdev.Helpers\ConvertTo-HashTable

                            # get the rule details safely
                            $ruleDetails = PSScriptAnalyzer\Get-ScriptAnalyzerRule `
                                -Name $resultHashTable.RuleName

                            # add rule description with null check
                            $resultHashTable['Description'] = `
                                if ($ruleDetails -and $ruleDetails.Description) {

                                $ruleDetails.Description
                            }
                            else {

                                'No description available'
                            }

                            # output the enhanced result
                            Microsoft.PowerShell.Utility\Write-Output $resultHashTable
                        }
            }
        }
        catch {

            # create error output for general exceptions
            $errorResult = @{
                Message     = $_.Exception.Message
                RuleName    = ($null -eq $ScriptFilePath ? '?' : `
                        [System.IO.Path]::GetFileNameWithoutExtension($ScriptFilePath)) + `
                    ' @ Invocation in Invoke-GenXdevScriptAnalyzer'
                Description = 'An error occurred while invoking the Script Analyzer.'
            }

            # output the error result
            Microsoft.PowerShell.Utility\Write-Output $errorResult
        }
    }

    end {
    }
}
#################################################################################