Skip to content

Commit

Permalink
Clean up mocks in the session state where we defined them (pester#1725)
Browse files Browse the repository at this point in the history
  • Loading branch information
nohwnd authored Oct 23, 2020
1 parent 8d55526 commit a412842
Show file tree
Hide file tree
Showing 2 changed files with 74 additions and 2 deletions.
27 changes: 25 additions & 2 deletions src/functions/Mock.ps1
Original file line number Diff line number Diff line change
Expand Up @@ -457,26 +457,49 @@ function Remove-MockHook {
$removeMockStub = {
param (
[string] $CommandName,
[string[]] $Aliases
[string[]] $Aliases,
[bool] $Write_Debug_Enabled,
$Write_Debug
)

if ($ExecutionContext.InvokeProvider.Item.Exists("Function:\$CommandName", $true, $true)) {
$ExecutionContext.InvokeProvider.Item.Remove("Function:\$CommandName", $false, $true, $true)
if ($Write_Debug_Enabled) {
& $Write_Debug -Scope Mock -Message "Removed function $($CommandName)$(if ($ExecutionContext.SessionState.Module) { " from module $($ExecutionContext.SessionState.Module) session state"} else { " from script session state"})."
}
}
else {
# # this runs from OnContainerRunEnd in the mock plugin, it might be running unnecessarilly
# if ($Write_Debug_Enabled) {
# & $Write_Debug -Scope Mock -Message "ERROR: Function $($CommandName) was not found$(if ($ExecutionContext.SessionState.Module) { " in module $($ExecutionContext.SessionState.Module) session state"} else { " in script session state"})."
# }
}

foreach ($alias in $Aliases) {
if ($ExecutionContext.InvokeProvider.Item.Exists("Alias:$alias", $true, $true)) {
$ExecutionContext.InvokeProvider.Item.Remove("Alias:$alias", $false, $true, $true)
if ($Write_Debug_Enabled) {
& $Write_Debug -Scope Mock -Message "Removed alias $($alias)$(if ($ExecutionContext.SessionState.Module) { " from module $($ExecutionContext.SessionState.Module) session state"} else { " from script session state"})."
}
}
else {
# # this runs from OnContainerRunEnd in the mock plugin, it might be running unnecessarilly
# if ($Write_Debug_Enabled) {
# & $Write_Debug -Scope Mock -Message "ERROR: Alias $($alias) was not found$(if ($ExecutionContext.SessionState.Module) { " in module $($ExecutionContext.SessionState.Module) session state"} else { " in script session state"})."
# }
}
}
}

$Write_Debug_Enabled = $PesterPreference.Debug.WriteDebugMessages.Value
$Write_Debug = $(if ($PesterPreference.Debug.WriteDebugMessages.Value) { $SafeCommands["Write-PesterDebugMessage"] } else { $null })

foreach ($h in $Hooks) {
if ($PesterPreference.Debug.WriteDebugMessages.Value) {
Write-PesterDebugMessage -Scope Mock -Message "Removing function $($h.BootstrapFunctionName)$(if($h.Aliases) { " and aliases $($h.Aliases -join ", ")" }) for$(if($h.ModuleName) { " $($h.ModuleName) -" }) $($h.CommandName)."
}

$null = Invoke-InMockScope -SessionState $h.CallerSessionState -ScriptBlock $removeMockStub -Arguments $h.BootstrapFunctionName, $h.Aliases
$null = Invoke-InMockScope -SessionState $h.SessionState -ScriptBlock $removeMockStub -Arguments $h.BootstrapFunctionName, $h.Aliases, $Write_Debug_Enabled, $Write_Debug
}
}

Expand Down
49 changes: 49 additions & 0 deletions tst/Pester.Mock.RSpec.ts.ps1
Original file line number Diff line number Diff line change
Expand Up @@ -668,4 +668,53 @@ i -PassThru:$PassThru {
$t.Result | Verify-Equal "Passed"
}
}

b "clean up mocks when -ModuleName is used" {
t "cleans up Mock in the module where it was defined" {
# when module name is used we should clean up in the module in which we defined the mock
# command and aliased, and not in the caller scope
# https://github.com/pester/Pester/issues/1693

Get-Module m | Remove-Module
$m = New-Module -Name m {
# calling 'a' which calls 'f' so we run the mock that is effective
# inside of this module
function a () { f }
function f () { "real" }

Export-ModuleMember -Function a
} -PassThru

$m | Import-Module

$sb = {
BeforeAll {
Mock f -ModuleName m { "mock" }
}

Describe "d1" {
It "i1" {
# this should mock and then on the end the mock should be teared down
# correctly so it is no longer effective in the next test run, when this is
# done incorrectly, the alias and mock function remain defined in the module
# and the mock hook remains in place. This breaks mock counting in subsequent tests.
a | Should -Be "mock"
}
}
}


$r = Invoke-Pester -Configuration ([PesterConfiguration]@{
Run = @{ ScriptBlock = $sb; PassThru = $true }
})

$r.Containers[0].Blocks[0].Tests[0].Result | Verify-Equal "Passed"
$command = & ($m) { Get-Command -Name f }

# this should be function. It should not be alias, because aliases are used for
# mocking
$command.CommandType | Verify-Equal "Function"
# $command.DisplayName
}
}
}

0 comments on commit a412842

Please sign in to comment.