Powershell: Preventing setting 'Ignore' as the $ErrorActionPreference value is poorly enforced on the command line, and not at all in scripts

Created on 26 Jul 2017  路  15Comments  路  Source: PowerShell/PowerShell

Related: #1759

The Ignore error-action value is meant to be used only with the -ErrorAction _common parameter_, not with the $ErrorActionPreference _preference variable_.

An explicit check to prevent the latter was clearly implemented (as evidenced by error message The value Ignore is not supported for an ActionPreference variable. The provided value should be used only as a value for a preference parameter, and has been replaced by the default value.), but it doesn't / doesn't properly take effect.

  • On the command line, $ErrorActionPreference = 'Ignore' is accepted _at first_. When the next error occurs, it is the error message about the invalid $ErrorActionPreference value that surfaces instead of the actual error's, and _only at that point_ is $ErrorActionPreference reset to Continue.

  • In a script, $ErrorActionPreference = 'Ignore' is accepted _and_ takes effect. A contributing factor is that preference variables in child scopes are not type-constrained, so I presume that no validation takes place at assignment time - see #3483

Steps to reproduce

  • On the command line:
# On the command line: set to invalid value, reflect it, then trigger an error, then reflect it again.
> $ErrorActionPreference = 'Ignore'; $ErrorActionPreference; 1 / 0; $ErrorActionPreference
  • In a script (place code in a *.ps1 file):
$ErrorActionPreference = 'Ignore'; $ErrorActionPreference; 1/0; $ErrorActionPreference

Expected behavior

  • Both on the command line and in a script:
The value Ignore is not supported for an ActionPreference variable. The provided value should be used only as a 
value for a preference parameter, and has been replaced by the default value.
...
Continue
Attempted to divide by zero.
...
Continue

Actual behavior

  • On the command line:
Ignore
$ErrorActionPreference = 'Ignore'; $ErrorActionPreference; 1 / 0: The value Ignore is not supported for an ActionPreference variable. The provided value should be used 
only as a value for a preference parameter, and has been replaced by the default value. For more information, see the Help topic, "about_Preference_Variables."
    + CategoryInfo          : NotSpecified: (:) [], NotSupportedException
    + FullyQualifiedErrorId : System.NotSupportedException
Continue

The value was accepted at first and retained until the next error occurred. The next error's message is mistakenly replaced with the invalid-$ErrorActionPreference-value error message, and the value is reset to Continue at that point.

  • In a script:
Ignore
Ignore

The value was accepted, stayed in effect, and suppressed the statement-terminating error.

Environment data

PowerShell Core v6.0.0-beta.4 on macOS 10.12.5
PowerShell Core v6.0.0-beta.4 on Ubuntu 16.04.2 LTS
PowerShell Core v6.0.0-beta.4 on Microsoft Windows 10 Pro (64-bit; v10.0.15063)
Windows PowerShell v5.1.15063.413 on Microsoft Windows 10 Pro (64-bit; v10.0.15063)
Committee-Reviewed Issue-Bug Resolution-Fixed WG-Engine

Most helpful comment

@PowerShell/powershell-committee reviewed this and agrees that the check for Ignore should be removed and allow the user to set what they want. Separately, we should have a PSScriptAnalyzer rule to detect and recommend against setting EAP.

All 15 comments

Related to this, this issue drives me crazy:

function Test-EAPreference {
    [CmdletBinding()]
    param()
    1/0
}

Test-EAPreference -ErrorAction Ignore

The results of this command are as follows:

Test-EAPreference : The value Ignore is not supported for an ActionPreference variable. The provided value should be used only as a value for a preference parameter, and has been replaced by the default value. For more information, see the Help topic, "about_Preference_Variables."
At line:1 char:1
+ Test-EAPreference -ErrorAction Ignore
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ CategoryInfo          : NotSpecified: (:) [Test-EAPreference], NotSupportedException
+ FullyQualifiedErrorId : System.NotSupportedException,Test-EAPreference

According to docs, that's a bug, correct? Because technically the function is setting ErrorActionPreference, but from an end user's perspective, they're just using Ignore with -ErrorAction which should work just fine. I felt it was worth mentioning with this bug because this is a scenario where the prevention logic is being too strict, causing a bug.

Indeed, @KirkMunro. What you're describing is the subject of #1759, which also shows workarounds.

Just to highlight it here as well for visibility, the best solution to this issue seems to be:

  1. Remove all logic that tries to prevent any action preference variable from being used as is when it has a value of Ignore.
  2. Add a new PSScriptAnalyzer rule that flags the direct assignment of any action preference variable to Ignore (via the equals operator or Set-Variable) as red, with appropriate documentation for that rule explaining why that's dangerous and what should be done instead.

Also shared over here.

There's more to this. The fact that it works in a script is simply because the logic internally was written incorrectly, only preventing $ErrorActionPreference from having a value of [System.Management.Automation.ActionPreference]::Ignore (e.g. an enumerated value). Having a value that is of type string (which is what happens whenever you create a locally scoped $ErrorActionPreference value -- it doesn't have to be in a script at all) allows Ignore to work just fine. You can also delete the top level $ErrorActionPreference variable, which is strongly typed as an ActionPreference, and then recreate it as a string and again it works.

For example:

Remove-Variable -Name ErrorActionPreference
[string]$ErrorActionPreference = 'Ignore'
function Test-Ignore {
    [CmdletBinding()]
    param()
    Get-Process -Id 12345678
}
Test-Ignore # Ignores the error because it uses the ErrorActionPreference variable of type string
Test-Ignore -ErrorAction Ignore # Raises the error about using Ignore because internally it creates a strongly typed ErrorActionPreference variable of type ActionPreference, sets the value as Ignore, and then when an error comes up that would otherwise be ignored, generates the `NotSupportedException`.

That, combined with the example below, demonstrates how terrible a job PowerShell is doing in this case, when really this is a linting problem, not something that the parser/compiler/interpreter should care about at all.

function Test-Ignore2 {
    [CmdletBinding()]
    param()
    'No errors to see here, move along'
}
Test-Ignore2 -ErrorAction Ignore # I can use this for functions some of the time, as long as they don't have errors to ignore? *sigh*

@SteveL-MSFT: Here's one for a committee meeting. Please give the green light to just remove the nonsensical code blocking Ignore from $ErrorActionPreference (but only some of the time, as this illustrates), in favor of a PSSA rule to warn folks who don't know better instead.

To add an additional thought for the committee: ActionPreference.Suspend is not supported in PowerShell Core (it's only used for workflows), and today it just adds confusion to the code with checks to ensure that value isn't used where there doesn't need to be any (plus some of those checks suffer the same issues identified for ActionPreference.Ignore, since they use the same logic). Do we need to keep ActionPreference.Suspend around at this point or can we just chuck it and the half-dozen checks related to it?

Sorry for the ignorance here (and mostly this is a note to myself for a Committee discussion later today), but why is setting EAP to Ignore dangerous?

Mainly because it effectively disables all possibility to handle non-terminating errors if you're not aware it's been set. For example, say you don't quite understand the gravity of it, and you set it in your profile.

Many commands will now simply cease to provide output, with no apparent reason as to why. No errors are generated or stored in $error, nothing is displayed. It's possible that for some poorly-designed commands or functions that this could be actually harmful, but in my mind it would mostly hurt the user experience for those less familiar with the language.

That said, it probably isn't that huge of a concern, really.

@PowerShell/powershell-committee reviewed this and agrees that the check for Ignore should be removed and allow the user to set what they want. Separately, we should have a PSScriptAnalyzer rule to detect and recommend against setting EAP.

@SteveL-MSFT What about the question about ActionPreference.Suspend? What was the decision there?

@KirkMunro sorry we missed that. Can you open a new issue on that and cc me to mark for @PowerShell/powershell-committee review? Thanks.

FYI, I already included a fix for this as part of PR #8205.

The fix for this was to allow ActionPreference.Ignore to be assigned to any action preference variable, which allows it to be used in functions as well. This fix was merged and included in PowerShell 7 preview 4. I believe this issue can be closed.

/cc @iSazonov 馃檪

Fixed by #10317

@KirkMunro Thanks for fixing this!

Was this page helpful?
0 / 5 - 0 ratings