如何转义schtasks / tr参数

时间:2019-03-18 03:11:34

标签: windows powershell cmd scheduled-tasks schtasks

我需要安排一个PowerShell任务,如下所示:

powershell.exe -file ..\Execute\execute.ps1

我尝试将其分配给参数$Argument,然后将其传递给schtasks,如下所示:

$Argument = "powershell.exe -file '..\Execute\execute.ps1'"
schtasks /create /tn "SOE_Checks" /tr $Argument /sc DAILY /st 05:00 /ru "System" /rl HIGHEST /f

但是在运行上述代码后,什么也没发生-成功创建任务后,它似乎没有运行。

我也曾尝试将其分配给$Argument,但不带引号,但仍然有效,但出现以下警告:

ERROR: Invalid syntax. Value expected for '/tr'.
Type "SCHTASKS /CREATE /?" for usage.

有人可以让我知道我在这里做错了什么吗? (我知道我可以使用PowerShell的New-ScheduledTaskAction完成此操作,但我希望它能以这种方式工作)

只需添加一下,如果我将文件路径更改为$Argument中的特定位置,如
$Argument = "powershell.exe -file 'C:\SOE\Execute\execute.ps1'",它可以正常工作而不会发出任何警告,但这并不理想。 / p>

我已经读过this,但对我不起作用

1 个答案:

答案 0 :(得分:2)

使用schtasks.exe创建的计划任务在工作目录设置为$env:windir\system32 [1] 的情况下执行,因此除非您的脚本恰好位于相对于那里..\Execute\execute.ps1,您的命令将无法正常工作。

如果您不想将脚本路径直接硬编码到命令中,请通过解析 absolute 动态构造命令 。 >当您分配给$Argument 时:

$Argument = 'powershell.exe -file \"{0}\"' -f (Convert-Path ..\Execute\execute.ps1)

请注意-不幸的是-需要将嵌入式"转换为\",这是一个长期存在的错误,为了向后兼容,尚未解决-请参见{{3} }作为背景。

Convert-Path将相对路径解析为绝对路径。
请注意,相对路径必须引用现有文件(或目录)。

类似地,相对路径您的脚本也将相对于$env:windir\system32 ;要使其相对于 script 的目录,请通过在开始时执行 Set-Location $PSScriptRoot ,首先更改为脚本的目录脚本。


可选阅读:如何引用计划任务中运行的命令:

注意:实际上,相同的规则适用于从Windows“运行”对话框运行命令(按 WinKey + R )时,< strong>您可以使用 test-drive 命令(该命令传递给schtasks /tr,而不用外部引号,而不是整个schtasks命令行)-但是请注意,工作目录将是用户的主目录,并且您将无法在PowerShell CLI的'...'参数周围使用-File-引号-参见下文):

    在执行期间不涉及
  • cmd.exe ,这意味着:

    • 例如,您不必担心cmd.exe元字符的非双引号使用,例如&,因此即使在单行中也可以使用这些字符引用的字符串作为powershell.exe参数(的一部分)传递到PowerShell CLI -Command

    • 相反,不直接支持输出重定向(例如> c:\path\to\log.txt)。

    • 在调用PowerShell CLI的上下文中,这意味着:

      • 使用-File,您不能在命令行上使用它们,而必须在脚本的 内执行。

      • 使用-Command,您可以使用它们,因为 PowerShell 才应用它们(但请注意,Windows PowerShell的{{ 1}}运算符会创建UTF-16LE文件。

  • (即使不涉及> 环境变量的引用的引用形式与{{1 }} 展开(例如cmd.exe

    • 注意事项:您 无法逃避此类参考
    • cmd.exe不起作用-附加的%USERNAME%被简单地视为一个文字,并且扩展仍然发生;例如%%会产生%
    • %%OS%%(偶然)阻止扩展,但保留%Windows_NT% -^%不会 ;相反,它“破坏”了变量名,在这种情况下,令牌保持原样;例如,^会产生^,即按原样保留。

以上内容适用于必须在计划任务中定义的命令,它们必须在任务计划程序(^%OS^%)中交互查看或定义。

另外,用于从命令行/ PowerShell脚本/批处理文件创建计划任务

  • 您必须引用命令作为整体

  • 符合调用外壳程序有关转义和预先字符串内插的语法规则。

(如果命令仅由不需要转义的单个单词组成,例如不包含空格或空格的可执行文件的路径,则只能引用 。特殊字符,并且不传递任何参数。)

调用^%OS^% [2] 时,taskschd.msc参数作为一个整体引用如下

  • PowerShell ,如果需要扩展(字符串插值)命令字符串 upfront ,请使用schtasks.exe;否则,请使用/tr
    重要:在两种情况下都需要将{em>嵌套
    "..."转换为'...',这在外部"引用的情况下意味着嵌套的\"必须转义为"..."(原文如此)。

    • 令人惊讶的是, "会识别嵌入 \`"的引号并将其自动转换为schtasks.exe的引号-这就是您原始命令的原因,即使在直接调用中,PowerShell CLI也不 支持将'...'"..."结合使用,"powershell.exe -file '..\Execute\execute.ps1'"仍然有效。
  • 来自'...' (无论是直接还是来自批处理文件),您必须使用-File

    < / li>

PowerShell示例

  • 以下PowerShell命令创建并执行两次 计划的任务,名为cmd.exe"...",在下一个日历分钟开始时在主叫用户的上下文中可见地运行。 (之后,您必须手动删除这些任务。)

  • 您可能需要等待多达1分钟才能看到调用启动,这时将为每个任务弹出一个新的控制台窗口。

test1

[1]请注意,任务计划程序GUI允许您配置工作目录,但无法通过test2实用程序使用此功能。

[2]传递给# Create sample script test.ps1 in the current dir. that # echoes its arguments and then waits for a keypress. '"Hi, $Args."; Read-Host "Press ENTER to exit"' > test.ps1 # Find the start of the next calendar minute. $nextFullMinute = ([datetime]::Now.AddMinutes(1).TimeOfDay.ToString('hh\:mm')) # -File example: # Invoke test.ps1 and pass it 'foo' as an argument. # Note the escaped embedded "..." quoting around the script path # and that with -File you can only pass literal arguments at # invocation time). schtasks.exe /create /f /tn test1 /sc once /st $nextFullMinute ` /tr "powershell -File \`"$PWD/test.ps1\`" foo" #`# (dummy comment to fix broken syntax highlighting) # -Command example: # Invoke test.ps1 and pass it $env:USERNAME as an argument. # Note the '...' around the script path and the need to invoke it with # &, as well as the ` before $env:USERNAME to prevent its premature expansion. schtasks.exe /create /f /tn test2 /sc once /st $nextFullMinute ` /tr "powershell -Command & '$PWD/test.ps1' `$env:USERNAME" "Tasks will execute at ${nextFullMinute}:00" PowerShell cmdlet的schtasks.exe参数的值也是如此,尽管请注意,可执行文件名/路径是通过{{ 1}}参数。
相比之下,用于创建计划的PowerShell jobs -Argument cmdlet接受 script块作为要运行的命令,从而消除了引人头痛的问题。
< / p>