-
Notifications
You must be signed in to change notification settings - Fork 126
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
RFC Trap handle non terminating errors #182
Open
TylerLeonhardt
wants to merge
3
commits into
PowerShell:master
Choose a base branch
from
TylerLeonhardt:RFC-Trap-Handle-Non-Terminating-Errors
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from all commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,126 @@ | ||
--- | ||
RFC: RFC0XXX | ||
Author: Tyler Leonhardt | ||
Status: Draft | ||
Area: Traps | ||
Comments Due: | ||
--- | ||
|
||
# Trap should handle non-terminating errors | ||
|
||
This RFC proposes that we change the behavior of the Trap statement to also trap non-terminating exceptions. | ||
|
||
To do this, | ||
we will also add a property on ErrorRecord to determine if it was a terminating or non-terminating exception. | ||
|
||
With this change, the following code: | ||
|
||
```pwsh | ||
trap { | ||
"hello" | ||
continue | ||
} | ||
Get-Process IDontExist | ||
``` | ||
|
||
will output the same as the following code that works today: | ||
|
||
```pwsh | ||
trap { | ||
"hello" | ||
continue | ||
} | ||
Get-Process IDontExist -ErrorAction Stop | ||
``` | ||
|
||
If you wanted to handle both terminating and non-terminating errors in your | ||
`trap` | ||
you can use the | ||
`ErrorType` | ||
property: | ||
|
||
```pwsh | ||
trap { | ||
if($_.ErrorType -eq "Terminating") { | ||
"oh no!" | ||
} else { | ||
"this is ok" | ||
continue | ||
} | ||
} | ||
``` | ||
|
||
## Motivation | ||
|
||
The Trap statement has fallen out of favor in recent years to its error handling cousin - | ||
the try/catch block. | ||
|
||
This gives us an opportunity to revamp the Trap statement so that it can serve a purpose that the try/catch could not serve. | ||
|
||
The purpose would be to offer a | ||
*single solution* | ||
to handle both terminating and non-terminating issues. | ||
|
||
## Specification | ||
|
||
`ErrorRecord` | ||
will get an additional property called | ||
`ErrorType` | ||
this will come from an | ||
`enum` | ||
called | ||
`ErrorRecordType` | ||
which has 2 values: | ||
|
||
* `Terminating` | ||
* `NonTerminating` | ||
|
||
Traps will then also be passed non-terminating errors to be handled. | ||
|
||
## Alternate Proposals and Considerations | ||
|
||
### Introduce a `trapall` | ||
|
||
Instead of changing the behavior of | ||
`trap`, | ||
we would add a | ||
`trapall` | ||
that does the above behavior. | ||
This might get tricky if a | ||
`trap` | ||
and a | ||
`trapall` | ||
were defined. | ||
This also means that code written using | ||
`trapall` | ||
would not run at all in older versions of PowerShell. | ||
|
||
### Add a -OnError parameter to every cmdlet | ||
|
||
The motivation of this RFC comes from the issue | ||
[PowerShell#6010](https://github.com/PowerShell/PowerShell/issues/6010). | ||
|
||
The proposal there was a parameter that you pass a ScriptBlock to: | ||
|
||
```pwsh | ||
Stop-Process foo* -OnError { | ||
Write-Host "Had an error." | ||
} | ||
``` | ||
|
||
This would then: | ||
|
||
* run for every non-terminating error | ||
* run once for the terminating error | ||
|
||
However, | ||
it's a bit confusing how this plays well with the pipeline and is suggesting changing the meaning of | ||
`continue` | ||
and | ||
`break` | ||
statements by adding them to a new context. | ||
|
||
The | ||
`OnError` | ||
proposal reminds me of a JavaScript callback which isn't really a well adopted pattern in the PowerShell world, | ||
which is why I chose against it. |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My personal preference is
trapall
, since:trap
trapall
will still parse and execute, but will just throw when the statement invokingtrapall
is hit in older PowerShell verisons. Compare this totrap -NonTerminating
which will not parse in older PowerShell versions. This means you can wrap the former in a version condition and it will still work cross-version.We might even able to carefully define
trapall
so thatICustomAstVisitor
isn't broken, by reusing theTrapStatementAst
type with a new parameter in the constructor detailing whether it's atrap
or atrapall
statement.