I want to do something like this:
<statement> | <filter1> | <filter2> if <condition> | <filter3> | <filter4> | <filter5>
The results of <statement> run through <filter1>, then they run through <filter2> only if <condition> is met, then through the remaining filters regardless of whether <filter2> was applied. This is the equivalent of:
if (<condition>) {
<statement> | <filter1> | <filter2> | <filter3> | <filter4> | <filter5>
} else {
<statement> | <filter1> | <filter3> | <filter4> | <filter5>
}
This would be useful in functions where a given filter is applied to the result set only if a certain switch was invoked. If the conditional filter occurs early in a long pipeline, writing it with an outer if-block results in a lot of repetition of code, especially if there is more than one conditional filter.
Here's an example. The following function shows the permissions a given account has in a given directory subtree (e.g. Show-AccountPerms \\SERVERX\Marketing DOMAIN\jdoe
gives a report of permissions that the user DOMAIN\jdoe has in the directory tree under \SERVERX\Marketing).
function Show-AccountPerms {
param (
[parameter(mandatory = $true)]$rootdir,
[parameter(mandatory = $true)]$account,
[switch]$files,
[switch]$inherited
)
gci -r $rootdir `
|where {$_.psiscontainer} `
|foreach {
$dir = $_.fullname
(get-acl $_.pspath).access `
| where {$_.isinherited -eq 'False'} `
|foreach {
if ($_.identityreference -eq $account) {
"{0,-25}{1,-35}{2}" -f $_.identityreference, $_.filesystemrights, $dir
}
}
}
}
By default, it only shows explicit permissions (enforced by the | where {$_.isinherited -eq 'False'}
filter), and only on directories (enforced by the |where {$_.psiscontainer}
filter).
However, I want to ignore |where {$_.psiscontainer}
if the -files switch is invoked, and ignore | where {$_.isinherited -eq 'False'}
if the -inherited switch is invoked. Accomplishing this with outer if blocks would quadruple the code, and almost 75% of it would be repetition. Is there a way to keep these filters in-line but instruct powershell to only apply them of the corresponding switch is false?
Please note that this is just an example, so I'm not interested in any workarounds specific to this function. I'm looking for an answer to my general question regarding piping conditionally, not a solution for how to accomplish this particular task.
Sorry, I didn't mean to abandon this question. The answers that were posted weren't what I was driving at, but I figured out a way to do it shortly after posting, and didn't come back to the site for a long time. Since a solution hasn't been posted, here's what I came up with. It's not quite what I had in mind when I asked the question and it isn't too pretty, but apparently it's the only way to do it:
<statement> | <filter1> | foreach {if (<condition>) {$_ | <filter2>} else {$_} | <filter3> | <filter4> | <filter5>
So, in the example, the line
|where {$_.psiscontainer} `
would be changed to
|foreach {if (-not $files) {$_ | where {$_.psiscontainer}} else {$_}} `
and
|where {$_.isinherited -eq 'False'} `
would be changed to
|foreach {if (-not $inherited) {$_ | where {$_.isinherited -eq 'False'}} else {$_}} `
(Yes, normally I'd write that as |foreach {if ($files) {$_} else {$_ | where {$_.psiscontainer}}}
, and |foreach {if ($inherited) {$_} else {$_ | where {$_.isinherited -eq 'False'}}}
but I did it this way for clarity.)
I was hoping there might be something more elegant, that would evaluate a condition in front of the filter once to determine whether to execute or skip a stage of the pipeline. Something like this:
<statement> | <filter1> | if (<condition>) {<filter2>} | <filter3>
(a special case of if
, not the usual meaning; a different keyword could be used), or maybe
<statement> | <filter1> | (<condition>) ? <filter2> | <filter3>
$_
would be invalid in the condition, unless it's defined outside the current pipeline, for example if the pipeline is contained within a switch
statement, $_
in the <condition>
would refer the switch
statement's $_
.
I think I'll make a feature suggestion to Microsoft. This would not only make the code more elegant, it would be more efficient as well, because if it's a built-in feature, <condition>
could be evaluated once for the entire pipeline, rather then testing the same independent condition in each iteration.