From 7d1a4ee975d69d8725b00f1e8bb2f501ca86eebf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Jare=C5=A1?= Date: Wed, 26 Jun 2024 15:28:42 +0200 Subject: [PATCH] Fix collection assertions with Should --- src/functions/assert/Collection/Should-All.ps1 | 11 +++++++++-- src/functions/assert/Collection/Should-Any.ps1 | 14 +++++++++++--- .../assert/Collection/Should-Any.Tests.ps1 | 6 ++++++ 3 files changed, 26 insertions(+), 5 deletions(-) diff --git a/src/functions/assert/Collection/Should-All.ps1 b/src/functions/assert/Collection/Should-All.ps1 index 87b4661bc..1f8dfeab4 100644 --- a/src/functions/assert/Collection/Should-All.ps1 +++ b/src/functions/assert/Collection/Should-All.ps1 @@ -76,9 +76,16 @@ $appendMore = $true } - $pass = $false + $pass = @($false) + } + + # The API returns a collection and user can return anything from their script + # or there can be no output when assertion is used, so we are checking if the first item + # in the output is a boolean $false. The scriptblock should not fail in $null for example, + # hence the explicit type check + if (($pass.Count -ge 1) -and ($pass[0] -is [bool]) -and ($false -eq $pass[0])) { + $item } - if (-not $pass) { $item } } # Make sure are checking the count of the filtered items, not just truthiness of a single item. diff --git a/src/functions/assert/Collection/Should-Any.ps1 b/src/functions/assert/Collection/Should-Any.ps1 index 6aab975a4..bd2f970b0 100644 --- a/src/functions/assert/Collection/Should-Any.ps1 +++ b/src/functions/assert/Collection/Should-Any.ps1 @@ -55,7 +55,6 @@ $failReasons = $null $appendMore = $false - $pass = $false foreach ($item in $Actual) { $underscore = [PSVariable]::new('_', $item) try { @@ -72,9 +71,18 @@ $appendMore = $true } - $pass = $false + # InvokeWithContext returns collection. This makes it easier to check the value if we throw and don't assign the value. + $pass = @($false) + } + + # The API returns a collection and user can return anything from their script + # or there can be no output when assertion is used, so we are checking if the first item + # in the output is a boolean $false. The scriptblock should not fail in $null for example, + # hence the explicit type check + if (-not (($pass.Count -ge 1) -and ($pass[0] -is [bool]) -and ($false -eq $pass[0]))) { + $pass = $true + break } - if ($pass) { break } } if (-not $pass) { diff --git a/tst/functions/assert/Collection/Should-Any.Tests.ps1 b/tst/functions/assert/Collection/Should-Any.Tests.ps1 index 4b18301e0..03fe6fd9f 100644 --- a/tst/functions/assert/Collection/Should-Any.Tests.ps1 +++ b/tst/functions/assert/Collection/Should-Any.Tests.ps1 @@ -9,6 +9,12 @@ Describe "Should-Any" { $Actual | Should-Any -FilterScript { $_ -eq 1 } } + It "Passes when at least one item in the given collection passes the predicate with assertion" -TestCases @( + @{ Actual = @(1, 2, 3) } + ) { + $Actual | Should-Any -FilterScript { $_ | Should-Be 1 } + } + It "Fails when none of the items passes the predicate" -TestCases @( @{ Actual = @(1, 2, 3) } @{ Actual = @(1) }