从闭包创建的 Pester Mock 中的空列表

Null List in Pester Mock created from a closure

为什么会出现下面的代码:

  function CreateExecutedCommandsLogger(){
     param(
        [System.Collections.Generic.List[System.String]]$cmdLog
     )
     "WWWWW=>{0}" -f $cmdLog | Write-Host
     return{
        param(
           [parameter(valuefrompipeline)]$command
        )
        "XXXXX=>{0}" -f $command | Write-Host
        "YYYYY=>{0}" -f $cmdLog | Write-Host
        $cmdLog.Add($command)
     }.GetNewClosure()
  }

  $executedCommands = New-Object System.Collections.Generic.List[System.String]
  Mock ExecuteSqlCommand (CreateExecutedCommandsLogger $executedCommands)

导致错误:

  RuntimeException: You cannot call a method on a null-valued expression.
  at <ScriptBlock>, <No file>: line 7
  at <ScriptBlock>, C:\Users\notme\Documents\WindowsPowerShell\Modules\Pester\Functions\Mock.ps1: line 1018
  at ExecuteBlock, C:\Users\notme\Documents\WindowsPowerShell\Modules\Pester\Functions\Mock.ps1: line 1022
  at Invoke-Mock, C:\Users\notme\Documents\WindowsPowerShell\Modules\Pester\Functions\Mock.ps1: line 868
  at <ScriptBlock><Process>, <No file>: line 53

作为参考,跟踪输出给出:

WWWWW=>System.Collections.Generic.List`1[System.String]
XXXXX=>C:\dummy\command.exe -S "blah" -d "blahblah" -i "something.sql"
YYYYY=>

当我使用以下内容时,它起作用了:

  $dummyMock = (CreateExecutedCommandsLogger $executedCommands)
  &$dummyMock "blah"

我假设这与模拟中脚本块的执行方式有关?

正如您从源代码 (Get-Command Mock).ScriptBlock 中看到的那样,只有一行使用了 $MockWith 参数:

$mockWithCopy = [scriptblock]::Create($MockWith.ToString())

因此,$MockWith 实际上是一个字符串,任何闭包都会被忽略。很可能它已经完成,因为在下一行新创建的脚本块 $mockWithCopy 绑定到某个会话状态,因此您无论如何都会松开闭包:

Set-ScriptBlockScope -ScriptBlock $mockWithCopy -SessionState $contextInfo.Session

Pester 首先执行脚本块的私有副本以不影响传递的实例。