SCOrchDev-Exception.psm1

<#
.SYNOPSIS
    Logs the details of an exception to the specified stream.
 
.PARAMETER Exception
    The exception whose details should be logged.
 
.PARAMETER Stream
    The stream that the exception details should be logged to.
#>

Function Write-Exception
{
    Param(
        [Parameter(Mandatory = $True)]
        $Exception,

        [Parameter(Mandatory = $False)]
        [ValidateSet('Debug', 'Error', 'Verbose', 'Warning')]
        $Stream = 'Error'
    )
    $LogCommand = (Get-Command -Name "Write-$Stream")
    $ExceptionString = Convert-ExceptionToString -Exception $Exception
    & $LogCommand -Message "Exception information`r`n------`r`n$ExceptionString" -ErrorAction Continue -WarningAction Continue
}


<#
.SYNOPSIS
    Returns a PSObject representing a custom exception.
 
.PARAMETER Type
    The "type" of the exception.
 
    This isn't actually the type. It's a string you can use to figure out what the type would be
    if PowerShell actually supported custom exceptions.
 
.PARAMETER Message
    A message describing the failure.
 
.PARAMETER Property
    A hashtable containing other properties to propogate with the exception. All values in the
    hash table will be converted to strings.
 
.EXAMPLE
    $Exception = New-Exception -Type 'GenericFailure' -Message 'Failing, just because'
    throw $Exception
#>

Function New-Exception
{
    param(
        [Parameter(Mandatory = $True, ParameterSetName = 'Values')]
        [String]
        $Type,
        
        [Parameter(Mandatory = $True, ParameterSetName = 'Values')]
        [String]
        $Message,
        
        [Parameter(Mandatory = $False, ParameterSetName = 'Values')]
        [Hashtable]
        $Property = @{},

        [Parameter(Mandatory = $True, ParameterSetName = 'ExceptionInfo')]
        [Object]
        $ExceptionInfo
    )
    if($ExceptionInfo)
    {
        $Property = @{}
        ($ExceptionInfo |
            Get-Member |
            Where-Object -FilterScript {
                $_.MemberType -eq 'NoteProperty' 
        }).Name | ForEach-Object -Process `
        {
            $Property[$_] = $ExceptionInfo."$_"
        }
    }
    else
    {
        $Property = $Property.Clone()
        $Property['__CUSTOM_EXCEPTION__'] = $True
        $Property['Type'] = $Type
        $Property['Message'] = $Message
        $Property['InnerException'] = $null
    }
    return ($Property | ConvertTo-Json -Compress)
}

<#
.SYNOPSIS
    Converts an exception into a string suitable for reading, including
    as much detail as possible that is useful for troubleshooting.
 
.PARAMETER Exception
    The exception that should be converted to a string.
#>

Function Convert-ExceptionToString
{
    Param(
        [Parameter(Mandatory = $True)]
        [Object]
        $Exception
    )
    
    $ExceptionString = New-Object -TypeName 'System.Text.StringBuilder'

    $ExceptionInfo = Get-ExceptionInfo -Exception $Exception
    while($ExceptionInfo -ne $null)
    {
        # NoteProperty properties contain all the properties from the exception that
        # we care about, so filter on those.
        #
        # We also need to filter out the InnerException property, since nested exceptions
        # will be handled by the outer loop.
        $ExceptionInfo |
        Get-Member |
        Where-Object -FilterScript {
            $_.MemberType -eq 'NoteProperty' 
        } |
        Where-Object -FilterScript {
            $_.Name -ne 'InnerException' 
        } |
        ForEach-Object -Process {
            $PropertyName = $_.Name
            if(-not [String]::IsNullOrEmpty($ExceptionInfo."$PropertyName"))
            {
                $null = $ExceptionString.AppendLine("$PropertyName = $($ExceptionInfo."$PropertyName")")
            }
        }
        if($ExceptionInfo.InnerException)
        {
            $null = $ExceptionString.AppendLine('')
            $null = $ExceptionString.AppendLine('Which was raised from:')
            $null = $ExceptionString.AppendLine('')
        }
        $ExceptionInfo = $ExceptionInfo.InnerException
    }
    return $ExceptionString.ToString()
}

<#
.SYNOPSIS
    Returns the exception information for an exception.
 
.PARAMETER Exception
    The exception whose information should be returned.
#>

Function Get-ExceptionInfo
{
    Param(
        [Parameter(
            Mandatory = $True,
            ValueFromPipeline = $True
        )]
        [AllowNull()]
        [Object]
        $Exception
    )

    $Property = @{}
    $TopLevelExceptionInfo = $null
    $PreviousExceptionInfo = $null
    while($Exception -ne $null)
    {
        $CustomException = Select-CustomException -Exception $Exception 
        if($CustomException)
        {
            $ExceptionInfo = ConvertFrom-Json -InputObject $CustomException
        }
        else
        {
            # Properties which should be included in the human-readable exception string.
            # The hash key is the identifier that will be used in the output string for the
            # hash value.
            #
            # Values will only be included if they are not null.
            $Property = @{
                'Type' = $Exception.GetType().FullName -replace '^Deserialized\.', ''
                'Message' = Select-FirstValid @($Exception.Message,$Exception.Exception.Message,$Exception.InnerException.Message,$Exception.SerializedRemoteException.Message)
                'FullyQualifiedErrorId' = $Exception.FullyQualifiedErrorId
                'HResult' = $Exception.HResult
                'ScriptBlock' = Select-FirstValid @($Exception.SerializedRemoteInvocationInfo.MyCommand.ScriptBlock,$Exception.InvocationInfo.MyCommand.ScriptBlock)
                'PositionMessage' = Select-FirstValid @($Exception.SerializedRemoteInvocationInfo.PositionMessage,$Exception.InvocationInfo.PositionMessage)
                'ScriptStackTrace' = $Exception.ScriptStackTrace
                'StackTrace' = $Exception.StackTrace
                'InnerException' = $null
            }
            $ExceptionInfo = New-Object -TypeName 'PSObject' -Property $Property
        }

        if(-not $TopLevelExceptionInfo)
        {
            $TopLevelExceptionInfo = $ExceptionInfo
        }
        elseif($PreviousExceptionInfo)
        {
            $PreviousExceptionInfo.InnerException = $ExceptionInfo
        }

        $PreviousExceptionInfo = $ExceptionInfo
        $Exception = Select-FirstValid -Value $Exception.Exception, 
                                              $Exception.InnerException
    }
    try
    {
        Add-Member -InputObject $TopLevelExceptionInfo `
                   -MemberType NoteProperty `
                   -Name 'PSCallStack' `
                   -Value ((Get-PSCallStack) | Where-Object -FilterScript { $_.ScriptName -notlike '*-Exception.psm1' } | ConvertTo-JSON)
    }
    catch
    { }
    return $TopLevelExceptionInfo
}

<#
.SYNOPSIS
    Tests whether or not the exception is a custom exception as returned
    by New-Exception or thrown by Throw-Exception. If it is, returns the
    custom exception string. Otherwise, returns $False.
 
.PARAMETER Exception
    The exception which should be tested.
#>

Function Select-CustomException
{
    param(
        [Parameter(Mandatory = $True)]
        [Object]
        $Exception
    )

    $TestString = @(($Exception -as [String]), ($Exception.Message -as [String]))
    
    Foreach($_TestString in $TestString)
    {
        try
        {
            $ExceptionObject = ConvertFrom-Json -InputObject $_TestString
            if($ExceptionObject.'__CUSTOM_EXCEPTION__')
            {
                return $_TestString
            }
        }
        catch [System.ArgumentException]
        {
            # It's not valid JSON, do nothing.
        }
    }
}

<#
.SYNOPSIS
    Attepts to select the relevant exception that has been thrown inside an invoked
    script block.
 
.DESCRIPTION
    Invoking a script block that throws an exception will result in an object where
    the exception we actually care about is hidden in an InnerException property.
    The exception of interest may not always be nested at the same depth (i.e. you may
    need to do $Exception.InnerException or $Exception.InnerException. InnerException
    to get what you want).
 
    This cmdlet attempts to remove the crud by filtering out select exception types.
 
.PARAMETER RootException
    The exception that was thrown by an invoked script block.
#>

Function Select-RelevantException
{
    Param(
        [Parameter(Mandatory = $True)]
        $RootException
    )
    $InnerException = $RootException.Exception
    $BlacklistedExceptions = [System.Management.Automation.ErrorRecord],
                             [System.Management.Automation.MethodInvocationException],
                             [System.Management.Automation.CmdletInvocationException]
    While(($InnerException.GetType() -in $BlacklistedExceptions) -and $InnerException.InnerException)
    {
        $InnerException = $InnerException.InnerException
    }
    Return $InnerException
}
<#
.SYNOPSIS
    Throws a custom exception.
 
.PARAMETER Type
    The "type" of the exception.
 
    This isn't actually the type. It's a string you can use to figure out what the type would be
    if PowerShell actually supported custom exceptions. Damnit, PowerShell. Use a switch block
    or similar in your catch statement to work with this.
 
.PARAMETER Message
    A message describing the failure.
 
.EXAMPLE
    try
    {
        Throw-Exception -Type 'GenericFailure' -Message 'Failing, just because'
    }
    catch
    {
        switch -CaseSensitive ((Get-ExceptionInfo -Exception $_).Type)
        {
            'GenericFailure'
            {
                Write-Warning -Message 'Look, a generic failure'
            }
            'SomeOtherFailure'
            {
                Write-Warning -Message 'Look, a failure we will never see'
            }
            default
            {
                Write-Warning -Message 'I am handling ALL the failures!'
            }
        }
    }
#>

Function Throw-Exception
{
    param(
        [Parameter(Mandatory = $True, ParameterSetName = 'Values')]
        [String]
        $Type,

        [Parameter(Mandatory = $True, ParameterSetName = 'Values')]
        [String]
        $Message,

        [Parameter(Mandatory = $False, ParameterSetName = 'Values')]
        [Hashtable]
        $Property = @{},

        [Parameter(Mandatory = $True, ParameterSetName = 'ExceptionInfo')]
        [Object]
        $ExceptionInfo
    )
    if($ExceptionInfo)
    {
        throw New-Exception -ExceptionInfo $ExceptionInfo
    }
    else
    {
        throw New-Exception -Type $Type -Message $Message -Property $Property
    }
}

Export-ModuleMember -Function * -Verbose:$False