Ответ 1
Я изучаю эту проблему, которая возникла в проекте, над которым я работаю, и обнаружила три вещи:
- Проблема связана с модулями.
- Если код, вызывающий
scriptBlock
физически расположен в любом месте файла .psm1, мы видим поведение. - Мы также видим поведение, если код, вызывающий
scriptBlock
, находится в отдельном файле script (.ps1), еслиscriptBlock
был передан из модуля. - Мы не видим поведения, если код, вызывающий
scriptBlock
, находится где угодно в файле script (.ps1), еслиscriptBlock
не был передан из модуля.
- Если код, вызывающий
-
scriptBlock
не обязательно будет выполняться в глобальной области. Скорее, он всегда появляется в любой области, из которой вызывается функция модуля. - Проблема не ограничивается "." оператор (dotsource). Я проверил три разных способа вызова
scriptBlock
: "." оператора, "&" оператора и методаscriptBlock
objectinvoke()
. В последних двух случаяхscriptBlock
выполняется с неправильной родительской областью. Это можно исследовать, пытаясь вызвать, например{set-variable -name "message" -scope 1 -value "From scriptBlock"}
Я надеюсь, что это проливает еще немного света на проблему, хотя я недостаточно далеко продвинулся, чтобы предложить обходное решение.
У кого-нибудь еще установлен PowerShell 1? Если это так, было бы полезно, если бы вы могли проверить, отображает ли оно то же поведение.
Вот файлы для моих тестовых случаев. Чтобы запустить их, создайте все четыре файла в одном каталоге и затем выполните "./all_tests.ps1" в командной строке PowerShell ISE
script_toplevel.ps1
param($script_block)
set-alias "wh" write-host
$message = "Script message"
wh " Script message before: '$message'"
. $script_block
wh " Script message after: '$message'"
script_infunction.ps1
param($script_block)
set-alias "wh" write-host
function f {
param($script_block)
$message = "Function message"
wh " Function message before: '$message'"
. $script_block
wh " Function message after: '$message'"
}
$message = "Script message"
wh " Script message before: '$message'"
f -script_block $script_block
wh " Script message after: '$message'"
module.psm1
set-alias "wh" write-host
function simple_test_fun {
param($script_block)
$message = "ModFunction message"
wh " ModFunction message before: '$message'"
. $script_block
wh " ModFunction message after: '$message'"
}
function ampersand_test_fun {
param($script_block)
$message = "ModFunction message"
wh " ModFunction message before: '$message'"
& $script_block
wh " ModFunction message after: '$message'"
}
function method_test_fun {
param($script_block)
$message = "ModFunction message"
wh " ModFunction message before: '$message'"
$script_block.invoke()
wh " ModFunction message after: '$message'"
}
function test_mod_to_script_toplevel {
param($script_block)
$message = "ModFunction message"
wh " ModFunction message before: '$message'"
& .\script_toplevel.ps1 -script_block $script_block
wh " ModFunction message after: '$message'"
}
function test_mod_to_script_function {
param($script_block)
$message = "ModFunction message"
wh " ModFunction message before: '$message'"
& .\script_infunction.ps1 -script_block $script_block
wh " ModFunction message after: '$message'"
}
export-modulemember -function "simple_test_fun", "test_mod_to_script_toplevel", "test_mod_to_script_function", "ampersand_test_fun", "method_test_fun"
all_tests.ps1
remove-module module
import-module .\module.psm1
set-alias "wh" write-host
wh "Test 1:"
wh " No problem with . at script top level"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -amp-calls-> Script -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: Script message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
& .\script_toplevel.ps1 -script_block {$message = "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 1 showed expected behavior"
wh
wh
wh "Test 2:"
wh " No problem with . inside function in script"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -amp-calls-> Script -calls-> Function -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: Function message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
& .\script_infunction.ps1 -script_block {$message = "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 2 showed expected behavior"
wh
wh
wh "Test 3:"
wh " Problem with with . with function in module"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -calls-> ModFunction -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
simple_test_fun -script_block {$message = "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 3 showed problem behavior"
wh
wh
wh "Test 4:"
wh " Confirm that problem scope is always scope where ScriptBlock is created"
wh " ScriptBlock created at 'f1' scope"
wh " TopScript -calls-> f1 -calls-> f2 -amp-calls-> ModFunction -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: f1 message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
function f1 {
$message = "f1 message"
wh " f1 message before: '$message'"
f2 -script_block {$message = "Script block message"}
wh " f1 message after: '$message'"
}
function f2 {
param($script_block)
$message = "f2 message"
wh " f2 message before: '$message'"
simple_test_fun -script_block $script_block
wh " f2 message after: '$message'"
}
f1
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 4 showed problem behavior"
wh
wh
wh "Test 4:"
wh " Confirm that problem scope is always scope where ScriptBlock is created"
wh " ScriptBlock created at 'f1' scope"
wh " TopScript -calls-> f1 -calls-> f2 -amp-calls-> ModFunction -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: f1 message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
function f1 {
$message = "f1 message"
wh " f1 message before: '$message'"
f2 -script_block {$message = "Script block message"}
wh " f1 message after: '$message'"
}
function f2 {
param($script_block)
$message = "f2 message"
wh " f2 message before: '$message'"
simple_test_fun -script_block $script_block
wh " f2 message after: '$message'"
}
f1
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 4 showed problem behavior"
wh
wh
wh "Test 5:"
wh " Problem with with . when module function invokes script (toplevel)"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -calls-> ModFunction -amp-calls-> Script -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
test_mod_to_script_toplevel -script_block {$message = "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 5 showed problem behavior"
wh
wh
wh "Test 6:"
wh " Problem with with . when module function invokes script (function)"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -calls-> ModFunction -amp-calls-> Script -calls-> function -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
test_mod_to_script_function -script_block {$message = "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 6 showed problem behavior"
wh
wh
wh "Test 7:"
wh " Problem with with & with function in module"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -calls-> ModFunction -amp-calls-> Script -calls-> function -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
ampersand_test_fun -script_block {set-variable -scope 1 -name "message" -value "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 7 showed problem behavior"
wh
wh
wh "Test 8:"
wh " Problem with with invoke() method with function in module"
wh " ScriptBlock created at 'TopScript' scope"
wh " TopScript -calls-> ModFunction -amp-calls-> Script -calls-> function -dot-calls-> ScriptBlock:"
wh
wh " Expected behavior: ModFunction message after: 'Script block message'"
wh " Problem behavior: TopScript message after: 'Script block message'"
wh
wh "Results:"
$global:message = "Global message"
$message = "Top script message"
wh " Global message before: '$global:message'"
wh " TopScript message before: '$message'"
method_test_fun -script_block {set-variable -scope 1 -name "message" -value "Script block message"}
wh " TopScript message after: '$message'"
wh " Global message after: '$global:message'"
wh
wh "Test 8 showed problem behavior"