我正在阅读这篇关于将函数传递到脚本块以用于作业的帖子:
Powershell start-job -scriptblock 无法识别同一文件中定义的函数?
我通过将函数作为变量传递来了解它的工作原理,它适用于简单的示例。那么现实世界的解决方案呢,有没有更优雅的方法来处理这个问题?
我有用于将更改部署到供应商软件的脚本。它读取一个 xml,告诉它如何导航环境并执行各种任务,即:映射驱动器、停止服务、调用 perl 安装脚本。我想为脚本提供一个参数以允许它同时运行,这样如果 perl 脚本需要 5 分钟(并不罕见)并且您正在推出 11 台服务器,您无需等待脚本运行一小时。
我只是要发布一些片段,因为完整的脚本有点冗长。日志功能:
function Log
{
Param(
[parameter(ValueFromPipeline=$true)]
$InputObject,
[parameter()]
[alias("v")]
$verbosity = $debug
)
$messageIndex = [array]::IndexOf($verbosityArray, $verbosity)
$verbosityIndex = [array]::IndexOf($verbosityArray, $loggingVerbosity)
if($messageIndex -ge $VerbosityIndex)
{
switch($verbosity)
{
$debug {Write-Host $verbosity ": " $InputObject}
$info {Write-Host $verbosity ": " $InputObject}
$warn {Write-Host $verbosity ": " $InputObject -ForegroundColor yellow}
$error {Write-Host $verbosity ": " $InputObject -ForegroundColor red}
}
}
}
这是另一个调用 log 函数的函数:
function ExecuteRollout
{
param(
[parameter(Mandatory=$true)]
[alias("ses")]
$session,
[parameter(Mandatory=$true)]
$command
)
#invoke command
Invoke-Command -session $session -ScriptBlock {$res = cmd /v /k `"$args[0]`"} -args $command
#get the return code from the remote session
$res = Invoke-Command -session $session {$res}
Log ("Command Output: "+$res)
$res = [string] $res
$exitCode = $res.substring($res.IndexOf("ExitCode:"), 10)
$exitCode = $exitCode.substring(9,1)
Log ("Exit code: "+$exitCode)
return $exitCode
}
最后是我主要的一个片段,这样你就可以了解发生了什么。$target.Destinations.Destination 将包含部署将转到的所有服务器和有关它们的相关信息。我删除了一些变量设置和日志记录以使其更紧凑,所以是的,您会看到从未定义过的引用变量:
#Execute concurrently
$target.Destinations.Destination | %{
$ScriptBlock = {
$destination = $args[0]
Log -v $info ("Starting remote session on: "+$destination.Server)
$session = New-PSSession -computerName $destination.Server
$InitializeRemote -session $session -destination $destination
#Gets a little tricky here, we need to keep the cmd session so it doesn't lose the sys vars set by env.bat
#String everything together with &'s
$cmdString = $destDrive + ": & call "+$lesDestDir+"data\env.bat & cd "+$rolloutDir+" & perl ..\JDH-rollout-2010.pl "+$rollout+" NC,r:\les & echo ExitCode:!errorlevel!"
Log ("cmdString: "+$cmdString)
Log -v $info ("Please wait, executing the rollout now...")
$exitCode = $ExecuteRollout -session $session -command $cmdString
Log ("ExitCode: "+$exitCode)
#respond to return code from rollout script
$HandleExitCode -session $session -destination $destination -exitCode $exitCode
$CleanUpRemote -session $session -destination $destination
}
Start-Job $ScriptBlock -Args $_
}
因此,如果我采用链接中的方法,我会将所有函数转换为变量并将它们传递给脚本块。目前,我的日志函数将默认登录 DEBUG,除非详细参数显式传递为不同的详细程度。如果我将我的函数转换为变量,但是 powershell 似乎不喜欢这种语法:
$Log ("Print this to the log")
所以我想我现在需要一直使用这个参数:
$Log ("Print this to the log" -v $debug
所以底线看起来我只需要将我的所有函数作为变量传递给脚本块,并在我调用它们时更改一些格式。这不是一个巨大的努力,但我想知道在我开始破解我的脚本之前是否有更好的方法。感谢您的输入和查看,我知道这是一篇很长的帖子。