Private/New-HaloError.ps1
using namespace System.Collections.Generic using namespace System.Management.Automation function New-HaloError { [CmdletBinding()] [Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSUseShouldProcessForStateChangingFunctions', '', Justification = 'Private function - no need to support.')] param ( [Parameter(Mandatory = $true)] [errorrecord]$ErrorRecord, [Parameter()] [switch]$HasResponse ) Write-Verbose 'Generating Halo error output.' $ExceptionMessage = [Hashset[String]]::New() $APIResultMatchString = '*The Halo API said*' $HTTPResponseMatchString = '*The API returned the following HTTP*' if ($ErrorRecord.ErrorDetails) { Write-Verbose 'ErrorDetails contained in error record.' Write-Debug "Raw ErrorDetails: $($ErrorDetails | Out-String)" $ErrorDetailsIsJson = Test-Json -Json $ErrorRecord.ErrorDetails -ErrorAction SilentlyContinue if ($ErrorDetailsIsJson) { Write-Verbose 'ErrorDetails is JSON.' $ErrorDetails = $ErrorRecord.ErrorDetails | ConvertFrom-Json Write-Debug "Raw error details: $($ErrorDetails | Out-String)" if ($null -ne $ErrorDetails) { if (($null -ne $ErrorDetails.resultCode) -and ($null -ne $ErrorDetails.errorMessage)) { Write-Verbose 'ErrorDetails contains resultCode and errorMessage.' $ExceptionMessage.Add("The Halo API said $($ErrorDetails.resultCode): $($ErrorDetails.errorMessage).") | Out-Null } elseif (($null -ne $ErrorDetails.ClassName) -and ($null -ne $ErrorDetails.Message)) { Write-Verbose 'ErrorDetails contains ClassName and Message.' $ExceptionMessage.Add("The Halo API said $($ErrorDetails.ClassName): $($ErrorDetails.Message)") | Out-Null } elseif ($null -ne $ErrorDetails.Message) { Write-Verbose 'ErrorDetails contains Message.' $ExceptionMessage.Add("The Halo API said $($ErrorDetails.Message)") | Out-Null } elseif ($null -ne $ErrorDetails.error) { Write-Verbose 'ErrorDetails contains error.' $ExceptionMessage.Add("The Halo API said $($ErrorDetails.error).") | Out-Null } elseif ($null -ne $ErrorDetails) { Write-Verbose 'ErrorDetails is not null.' $ExceptionMessage.Add("The Halo API said $($ErrorRecord.ErrorDetails).") | Out-Null } else { Write-Verbose 'ErrorDetails is null.' $ExceptionMessage.Add('The Halo API returned an error.') | Out-Null } } else { Write-Verbose 'No ErrorDetails - or could not parse.' } } elseif ($ErrorRecord.ErrorDetails -Like $APIResultMatchString -and $ErrorRecord.ErrorDetails -Like $HTTPResponseMatchString) { $Errors = $ErrorRecord.ErrorDetails -Split "`r`n" if ($Errors -is [array]) { ForEach-Object -InputObject $Errors { $ExceptionMessage.Add($_) | Out-Null } } elseif ($Errors -is [string]) { $ExceptionMessage.Add($_) } } else { Write-Verbose 'ErrorDetails was not JSON or a parseable string or array.' } } else { $ExceptionMessage.Add('The Halo API returned an error but did not provide a result code or error message.') | Out-Null } if (($ErrorRecord.Exception.Response -and $HasResponse) -or $ExceptionMessage -NotLike $HTTPResponseMatchString) { $Response = $ErrorRecord.Exception.Response Write-Debug "Raw HTTP response: $($Response | Out-String)" if ($Response.StatusCode.value__ -and $Response.ReasonPhrase) { $ExceptionMessage.Add("The API returned the following HTTP error response: $($Response.StatusCode.value__) $($Response.ReasonPhrase)") | Out-Null } else { $ExceptionMessage.Add('The API returned an HTTP error response but did not provide a status code or reason phrase.') } } $Exception = [System.Exception]::New( $ExceptionMessage, $ErrorRecord.Exception ) $HaloError = [ErrorRecord]::New( $ErrorRecord, $Exception ) $UniqueExceptions = $ExceptionMessage | Get-Unique if ($UniqueExceptions.Count -gt 1) { $HaloError.ErrorDetails = [String]::Join("`r`n", $UniqueExceptions) } else { $HaloError.ErrorDetails = $UniqueExceptions } $PSCmdlet.ThrowTerminatingError($HaloError) } |