about_Pipeline_Chain_Operators

Descrição breve

Descreve pipelines de encadeamento com os && operadores e || no PowerShell.

Descrição longa

A partir do PowerShell 7, o PowerShell implementa os && operadores e || para encadear condicionalmente pipelines. Esses operadores são conhecidos no PowerShell como operadores de cadeia de pipeline e são semelhantes às listas AND-OR em shells POSIX, como bash, zsh e sh, bem como símbolos de processamento condicional no Shell de Comando do Windows (cmd.exe).

O operador && executa o pipeline à direita, se o pipeline à esquerda foi bem-sucedido. Por outro lado, o operador || executa o pipeline à direita, se o pipeline à esquerda falhou.

Esses operadores usam as variáveis $? e $LASTEXITCODE para determinar se um pipeline falhou. Isso permite que você os utilize com comandos nativos, e não apenas com cmdlets ou funções. Por exemplo:

# Create an SSH key pair - if successful copy the public key to clipboard
ssh-keygen -t rsa -b 2048 && Get-Content -Raw ~\.ssh\id_rsa.pub | clip

Exemplos

Dois comandos bem-sucedidos

Write-Output 'First' && Write-Output 'Second'
First
Second

O primeiro comando falha, fazendo com que o segundo não seja executado

Write-Error 'Bad' && Write-Output 'Second'
Write-Error: Bad

O primeiro comando é bem-sucedido, portanto, o segundo comando não é executado

Write-Output 'First' || Write-Output 'Second'
First

O primeiro comando falha, então o segundo comando é executado

Write-Error 'Bad' || Write-Output 'Second'
Write-Error: Bad
Second

O sucesso do pipeline é definido pelo valor da variável, que o PowerShell define automaticamente após a $? execução de um pipeline com base em seu status de execução. Isso significa que os operadores da cadeia de dutos têm a seguinte equivalência:

Test-Command '1' && Test-Command '2'

funciona da mesma forma que

Test-Command '1'; if ($?) { Test-Command '2' }

e

Test-Command '1' || Test-Command '2'

funciona da mesma forma que

Test-Command '1'; if (-not $?) { Test-Command '2' }

Atribuição de cadeias de dutos

A atribuição de uma variável de uma cadeia de pipeline leva a concatenação de todos os pipelines na cadeia:

$result = Write-Output '1' && Write-Output '2'
$result
1
2

Se ocorrer um erro de encerramento de script durante a atribuição de uma cadeia de pipeline, a atribuição não terá êxito:

try
{
    $result = Write-Output 'Value' && $(throw 'Bad')
}
catch
{
    # Do nothing, just squash the error
}

"Result: $result"
Result:

Sintaxe e precedência do operador

Ao contrário de outros operadores, && e || operam em dutos, em vez de em expressões como + ou -and, por exemplo.

&& e || têm uma precedência menor do que tubulação (|) ou redirecionamento (>), mas uma precedência maior do que operadores de trabalho (&), atribuição (=) ou ponto-e-vírgula (;). Isso significa que os pipelines dentro de uma cadeia de pipeline podem ser redirecionados individualmente e que cadeias de pipeline inteiras podem ser colocadas em segundo plano, atribuídas a variáveis ou separadas como instruções.

Para usar a sintaxe de precedência mais baixa em uma cadeia de pipeline, considere o uso de parênteses (...). Da mesma forma, para incorporar uma instrução em uma cadeia de pipeline, uma subexpressão $(...) pode ser usada. Isso pode ser útil para combinar comandos nativos com fluxo de controle:

foreach ($file in 'file1','file2','file3')
{
    # When find succeeds, the loop breaks
    find $file && Write-Output "Found $file" && $(break)
}
find: file1: No such file or directory
file2
Found file2

A partir do PowerShell 7, o comportamento dessas sintaxes foi alterado para que $? seja definido conforme o esperado quando um comando é bem-sucedido ou falha entre parênteses ou uma subexpressão.

Como a maioria dos outros operadores no PowerShell, && e || também são associativos à esquerda, o que significa que eles se agrupam a partir da esquerda. Por exemplo:

Get-ChildItem -Path ./file.txt ||
    Write-Error "file.txt doesn't exist" &&
    Get-Content -Raw ./file.txt

agrupar-se-á como:

(Get-ChildItem -Path ./file.txt || Write-Error "file.txt doesn't exist") &&
    Get-Content -Raw ./file.txt

sendo equivalente a:

Get-ChildItem -Path ./file.txt

if (-not $?) { Write-Error "file.txt does not exist" }

if ($?) { Get-Content -Raw ./file.txt }

Interação de erro

Os operadores da cadeia de dutos não absorvem erros. Quando uma instrução em uma cadeia de pipeline lança um erro de encerramento de script, a cadeia de pipeline é encerrada.

Por exemplo:

$(throw 'Bad') || Write-Output '2'
Exception: Bad

Mesmo quando o erro é detectado, a cadeia de pipeline ainda é encerrada:

try
{
    $(throw 'Bad') || Write-Output '2'
}
catch
{
    Write-Output "Caught: $_"
}
Write-Output 'Done'
Caught: Bad
Done

Se um erro não terminar ou apenas encerrar um pipeline, a cadeia de pipeline continuará, respeitando o valor de $?:

function Test-NonTerminatingError
{
    [CmdletBinding()]
    param()

    $exception = [System.Exception]::new('BAD')
    $errorId = 'BAD'
    $errorCategory = 'NotSpecified'

    $errorRecord = [System.Management.Automation.ErrorRecord]::new(
        $exception, $errorId, $errorCategory, $null
    )

    $PSCmdlet.WriteError($errorRecord)
}

Test-NonTerminatingError || Write-Output 'Second'
Test-NonTerminatingError: BAD
Second

Encadeamento de pipelines em vez de comandos

Os operadores da cadeia de dutos, por seu nome, podem ser usados para encadear dutos, em vez de apenas comandos. Isso corresponde ao comportamento de outros shells, mas pode tornar o sucesso mais difícil de determinar:

function Test-NotTwo
{
    [CmdletBinding()]
    param(
      [Parameter(ValueFromPipeline)]
      $Input
    )

    process
    {
        if ($Input -ne 2)
        {
            return $Input
        }

        $exception = [System.Exception]::new('Input is 2')
        $errorId = 'InputTwo'
        $errorCategory = 'InvalidData'

        $errorRecord = [System.Management.Automation.ErrorRecord]::new(
            $exception, $errorId, $errorCategory, $null
        )

        $PSCmdlet.WriteError($errorRecord)
    }
}

1,2,3 | Test-NotTwo && Write-Output 'All done!'
1
Test-NotTwo : Input is 2
3

Observe que não é executado, uma vez Test-NotTwo que Write-Output 'All done!' é considerado como tendo falhado após gerar o erro de não encerramento.

Confira também