使用脚本中的参数调用第二个脚本

Posted

技术标签:

【中文标题】使用脚本中的参数调用第二个脚本【英文标题】:Invoke a second script with arguments from a script 【发布时间】:2012-10-02 18:18:12 【问题描述】:

我有一个脚本,它读取一个配置文件,该文件生成一组名称值对,我想将它们作为参数传递给第二个 PowerShell 脚本中的函数。

我不知道在设计时会在这个配置文件中放置什么参数,所以在我需要调用第二个 PowerShell 脚本的时候,我基本上只有一个变量,它具有第二个脚本的路径,第二个变量是一个参数数组,传递给路径变量中标识的脚本。

因此,包含第二个脚本 ($scriptPath) 路径的变量可能具有如下值:

"c:\the\path\to\the\second\script.ps1"

包含参数的变量 ($argumentList) 可能类似于:

-ConfigFilename "doohickey.txt" -RootDirectory "c:\some\kind\of\path" -Max 11

如何使用 $argumentList 中的所有参数执行 script.ps1?

我希望第二个脚本中的任何写入主机命令对调用第一个脚本的控制台可见。

我尝试过点源、Invoke-Command、Invoke-Expression 和 Start-Job,但我没有找到不会产生错误的方法。

例如,我认为最简单的第一条路线是尝试如下调用的 Start-Job:

Start-Job -FilePath $scriptPath -ArgumentList $argumentList

...但是失败并出现以下错误:

System.Management.Automation.ValidationMetadataException:
Attribute cannot be added because it would cause the variable
ConfigFilename with value -ConfigFilename to become invalid.

...在这种情况下,“ConfigFilename”是第二个脚本定义的参数列表中的第一个参数,我的调用显然试图将其值设置为“-ConfigFilename”,这显然是为了识别按名称参数,而不是设置其值。

我错过了什么?

编辑:

好的,这是一个被调用脚本的模型,在一个名为 invokee.ps1 的文件中

Param(
[parameter(Mandatory=$true)]
[alias("rc")]
[string]
[ValidateScript( Test-Path $_ -PathType Leaf )]
$ConfigurationFilename,
[alias("e")]
[switch]
$Evaluate,
[array]
[Parameter(ValueFromRemainingArguments=$true)]
$remaining)

function sayHelloWorld()

    Write-Host "Hello, everybody, the config file is <$ConfigurationFilename>."
    if ($ExitOnErrors)
    
        Write-Host "I should mention that I was told to evaluate things."
    
    Write-Host "I currently live here: $gScriptDirectory"
    Write-Host "My remaining arguments are: $remaining"
    Set-Content .\hello.world.txt "It worked"


$gScriptPath = $MyInvocation.MyCommand.Path
$gScriptDirectory = (Split-Path $gScriptPath -Parent)
sayHelloWorld

...这里是调用脚本的模型,在一个名为 invoker.ps1 的文件中:

function pokeTheInvokee()

    $scriptPath = (Join-Path -Path "." -ChildPath "invokee.ps1")
    $scriptPath = $ExecutionContext.SessionState.Path.GetUnresolvedProviderPathFromPSPath($scriptPath)
    
    $configPath = (Join-Path -Path "." -ChildPath "invoker.ps1")
    $configPath = $ExecutionContext.SessionState.Path.GetUnresolvedProviderPathFromPSPath($configPath)
    
    $argumentList = @()
    $argumentList += ("-ConfigurationFilename", "`"$configPath`"")
    $argumentList += , "-Evaluate"

    Write-Host "Attempting to invoke-expression with: `"$scriptPath`" $argumentList"
    Invoke-Expression "`"$scriptPath`" $argumentList"
    Invoke-Expression ".\invokee.ps1 -ConfigurationFilename `".\invoker.ps1`" -Evaluate
    Write-Host "Invokee invoked."


pokeTheInvokee

当我运行invoker.ps1 时,这是我在第一次调用Invoke-Expression 时遇到的错误:

Invoke-Expression : You must provide a value expression on
the right-hand side of the '-' operator.

第二个调用工作得很好,但一个显着的区别是第一个版本使用的参数的路径中有空格,而第二个没有。我是否错误地处理了这些路径中存在的空格?

【问题讨论】:

如果您为每个脚本提供源代码也会有所帮助。例如,创建 POC(概念证明)项目来说明您的问题。然后我们可以测试并使用它,也许有人会找到解决方案或解决方法。 是的 - 我正在尝试制作一个迷你版来说明。很快就会发布... 【参考方案1】:

啊哈。原来这是一个简单的问题,脚本路径中有空格。

将 Invoke-Expression 行更改为:

Invoke-Expression "& `"$scriptPath`" $argumentList"

...足以让它开始。感谢 Neolisk 的帮助和反馈!

【讨论】:

添加到您的答案中。路径中有空格意味着您必须使用引号。 Invoke-Expression 将引号视为字符串,而不是命令。因此,如果要执行字符串,则需要使用“调用运算符”,即示例开头的 & 符号。 Here 是呼叫操作员的定义。 当我以这种方式尝试我的脚本时,它有一个缺少结束引号的语法错误。下面的方法 2 对我有用 我的脚本参数列表中的一个字符串参数中的分号字符有问题。 Invoke-Expression 一直将其解释为语句分隔符,并且仅将字符串的开头作为参数传递给脚本。当我在我的代码中应用这个答案时,它工作得很好。【参考方案2】:

Invoke-Expression 应该可以正常工作,只要确保您正确使用它即可。对于您的情况,它应该如下所示:

Invoke-Expression "$scriptPath $argumentList"

我使用 Get-Service 测试了这种方法,并且似乎按预期工作。

【讨论】:

Invoke-Expression "$scriptPath $argumentList" 立即返回并且似乎没有运行脚本。如果我将其更改为 Invoke-Expression "powershell $scriptPath $argumentList",PowerGui 似乎会挂起。 @Hoobajoob:Invoke-Expression 不会立即返回,它会耐心等待调用的脚本结束——我刚刚在 POC 项目上进行了测试。您定位的脚本一定有问题。 Hm - 我已经从批处理文件、Visual Studio IDE、DOS 命令行和 powershell 控制台会话中以多种不同的方式运行第二个脚本,但这是我第一次尝试运行它来自另一个powershell脚本。脚本本身可能有哪些内容会阻止它通过 Invoke-Expression 成功启动? 这只有在参数列表中的对象都是字符串类型时才有效,对吧? @Backwards_Dave: $scriptPath$argumentList 这里是字符串。一个字符串可以包含一个非字符串参数(写成一个字符串)。希望它能回答你的问题。【参考方案3】:

其实很多simpler:

方法一:

Invoke-Expression $scriptPath $argumentList

方法二:

& $scriptPath $argumentList

方法三:

$scriptPath $argumentList

如果您的scriptPath 中有空格,请不要忘记将它们转义`"$scriptPath`"

【讨论】:

感谢您的选择,我真的很喜欢这个答案在另一个问题上提供的附加选项。 ***.com/a/33206405/3794873【参考方案4】:

这是一个涵盖从 PS 脚本调用另一个 PS 脚本的更一般性问题的答案,如果您正在编写由许多小型、用途狭窄的脚本组成的脚本,您可能会这样做。

我发现这只是使用点源的一个案例。也就是说,您只需这样做:

# This is Script-A.ps1

. ./Script-B.ps1 -SomeObject $variableFromScriptA -SomeOtherParam 1234;

我发现所有的 Q/A 都非常混乱和复杂,最终找到了上面的简单方法,这真的就像调用另一个脚本一样,好像它是原始脚本中的一个函数,我觉得这更直观。

点源可以“导入”整个其他脚本,使用:

. ./Script-B.ps1

现在好像两个文件合并了。

最终,我真正缺少的是我应该构建一个可重用函数模块的概念。

【讨论】:

&amp; "./Script-B.ps1" -SomeObject $variableFromScriptA -SomeOtherParam 1234 一个模块不需要导出到其他系统有什么好处?【参考方案5】:

我们可以为此使用splatting:

& $command @args

@args (automatic variable $args) 被放入参数数组中。

在 PS 下,5.1

【讨论】:

【参考方案6】:

我尝试了使用 Invoke-Expression cmdlet 的公认解决方案,但它对我不起作用,因为我的参数上有空格。我试图解析参数并转义空格,但我无法使其正常工作,而且在我看来这确实是一项肮脏的工作。 所以经过一些实验,我对这个问题的看法是:

function Invoke-Script

    param
    (
        [Parameter(Mandatory = $true)]
        [string]
        $Script,

        [Parameter(Mandatory = $false)]
        [object[]]
        $ArgumentList
    )

    $ScriptBlock = [Scriptblock]::Create((Get-Content $Script -Raw))
    Invoke-Command -NoNewScope -ArgumentList $ArgumentList -ScriptBlock $ScriptBlock -Verbose


# example usage
Invoke-Script $scriptPath $argumentList

此解决方案的唯一缺点是您需要确保您的脚本没有“Script”或“ArgumentList”参数。

【讨论】:

【参考方案7】:

您可以像 SQL 查询一样执行它。 首先,构建您的命令/表达式并存储在变量中并执行/调用。

$command =  ".\yourExternalScriptFile.ps1" + " -param1 '$paramValue'"

它非常前卫,我认为不需要解释。 所以现在都准备好执行你的命令了,

Invoke-Expression $command

我建议在这里捕获异常

【讨论】:

【参考方案8】:

我假设您想从另一个 .ps1 文件运行 .ps1 文件 [此处为 $scriptPath 以及存储在 $argumentList 中的多个参数]

Invoke-Expression "& $scriptPath $argumentList"

这段代码可以正常工作

【讨论】:

以上是关于使用脚本中的参数调用第二个脚本的主要内容,如果未能解决你的问题,请参考以下文章

java调用shell脚本并传参

python 如何调用带参数的shell脚本

Zabbix调用外部脚本发送邮件:python编写脚本

为啥“使用严格”导致函数未定义

当第一个脚本运行第二个脚本方法时,第二个脚本无法识别第一个脚本的变量

如何使用绑定变量执行 SQL 脚本