【问题标题】:Python subprocess call cannot take grep [duplicate]Python子进程调用不能接受grep [重复]
【发布时间】:2016-06-14 10:44:00
【问题描述】:

Python 子进程调用应该按原样作为命令运行,但它会抱怨其中是否有管道。这是我的代码:

#!/usr/bin/python

import sys
import subprocess
import time
service_name= "mysrvc"
state ="STOPPED"
mycmd ="sc query " + service_name + " " + "|" + " findstr" + " " + state 
print(mycmd)
if subprocess.call(mycmd)==0:
  print("Service stopped successfully")

我得到的错误是:

ERROR: Invalid Option; Would you like to see help for the QUERY and QUERYEX commands? [ y | n ]:

如果我将命令更改为只是

mycmd = "sc query " + service_name 

我能够成功运行脚本。只是管道和它后面的参数是一个问题。如果我直接在命令行上运行sc query mysvrc | findstr STOPPED,它就可以正常工作。

我怎样才能让它工作?请注意,我使用 jython2.7 运行此 python 脚本。我没有成功使用 win32serviceutil,因为它找不到模块 win32serviceutil。

【问题讨论】:

  • 请将整个错误消息复制粘贴到您的问题中。请参阅minimal reproducible example 了解更多信息。
  • 我得到的错误是:错误:无效选项;您想查看有关 QUERY 和 QUERYEX 命令的帮助吗? [ 是 | n]:
  • edit您的问题包含错误消息 --- 它将被埋在 cmets 中。

标签: python jython


【解决方案1】:

我不确定jython,但subprocess 文档建议您的命令需要是一个列表,而不是字符串,除非您将shell 变量设置为True。如果您将调用更改为 subprocess.call(mycmd, shell=True),您的代码应该可以工作,但请务必阅读文档中有关将 shell 设置为 True 所固有的安全风险的警告。

如果您不想设置shell=True,您将无法直接在命令中使用管道,但文档中有一节replacing shell pipeline 介绍了如何模仿管道使用subprocess.Popen

【讨论】:

    【解决方案2】:

    如前所述,subprocess can't handle single str inputs and shell metacharacters like | unless shell=True。但在这种情况下,你真的不需要管道。您可以让 Python 进行过滤并完全避免使用到 findstr 的管道:

    # sc query command only, as list which gets better safety/performance
    mycmd = ["sc", "query", service_name]
    
    # Open command to run asynchronously, capturing output
    proc = subprocess.Popen(mycmd, stdout=subprocess.PIPE)
    
    # Wait for process to complete while slurping output
    stdout, _ = proc.communicate()
    
    # Check if expected output was seen and process exited successfully
    if state in stdout and proc.returncode == 0:
        print("Service stopped successfully")
    

    【讨论】:

    • 嗨!代码中的“状态”是什么?我收到以下错误:如果 stdout 和 proc.returncode == 0 中的状态:NameError: name 'state' is not defined
    • 对不起.. 我了解状态是什么。我在“if state in stdout and proc.returncode==0”中添加了一个 else 来打印错误。当我执行它时,它总是会转到 else 部分。我认为它没有评估标准输出中的状态。
    • 我打印了标准输出,发现状态为“STOP PENDING”。这就是它失败的原因。我将编写一个 while do 循环来继续查询,直到它变为 STOPPED。这应该会有所帮助
    • @user1164061:很高兴你知道了。 :-) 直到现在才在电脑前看到你的 cmets。
    • 非常感谢您对我的帮助。
    猜你喜欢
    • 2011-12-18
    • 2021-05-01
    • 2017-03-01
    • 1970-01-01
    • 2018-07-03
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多