Как использовать подпроцесс, когда несколько аргументов содержат пробелы? - PullRequest
19 голосов
/ 30 апреля 2009

Я работаю над сценарием-оберткой, который будет работать с исполняемым файлом vmware, позволяющим автоматизировать действия запуска / завершения работы / регистрации / отмены регистрации виртуальной машины. Я пытаюсь использовать подпроцесс для обработки вызова исполняемого файла, но пробелы в пути к исполняемым файлам и в параметрах исполняемого файла неправильно обрабатываются подпроцессом. Ниже приведен фрагмент кода:

vmrun_cmd = r"c:/Program Files/VMware/VMware Server/vmware-cmd.bat"
def vm_start(target_vm):
    list_arg = "start"
    list_arg2 = "hard"
    if vm_list(target_vm):
            p = Popen([vmrun_cmd, target_vm, list_arg, list_arg2],   stdout=PIPE).communicate()[0]
            print p
    else:
            vm_register(target_vm)
            vm_start(target_vm)
def vm_list2(target_vm):
    list_arg = "-l"
    p = Popen([vmrun_cmd, list_arg], stdout=PIPE).communicate()[0]
    for line in p.split('\n'):
            print line

Если я вызываю функцию vm_list2, я получаю следующий вывод:

$ ./vmware_control.py --list                                                
C:\Virtual Machines\QAW2K3Server\Windows Server 2003 Standard Edition.vmx
C:\Virtual Machines\ubunturouter\Ubuntu.vmx
C:\Virtual Machines\vacc\vacc.vmx
C:\Virtual Machines\EdgeAS-4.4.x\Other Linux 2.4.x kernel.vmx
C:\Virtual Machines\UbuntuServer1\Ubuntu.vmx
C:\Virtual Machines\Other Linux 2.4.x kernel\Other Linux 2.4.x kernel.vmx
C:\Virtual Machines\QAClient\Windows XP Professional.vmx

Если я вызываю функцию vm_start, для которой требуется параметр path-to-vm, я получаю следующий вывод:

$ ./vmware_control.py --start "C:\Virtual Machines\ubunturouter\Ubuntu.vmx"
'c:\Program' is not recognized as an internal or external command,
operable program or batch file.

Очевидно, что наличие второго параметра со встроенными пробелами изменяет способ, которым подпроцесс интерпретирует первый параметр. Любые предложения о том, как решить эту проблему?

python2.5.2 / Cygwin / WinXP

Ответы [ 10 ]

5 голосов
/ 30 апреля 2009

Если у вас есть пробелы в пути, я нашел самый простой способ правильно их интерпретировать:

subprocess.call('""' + path + '""')

Я не знаю, для чего именно нужны двойные двойные кавычки, но это то, что работает.

5 голосов
/ 30 апреля 2009
'c:\Program' is not recognized as an internal or external command,
operable program or batch file.

Чтобы получить это сообщение, вы:

  1. Использование shell=True:

    vmrun_cmd = r"c:\Program Files\VMware\VMware Server\vmware-cmd.bat"
    subprocess.Popen(vmrun_cmd, shell=True)
    
  2. Изменение vmrun_cmd на другую часть вашего кода

  3. Получение этой ошибки из чего-то внутри vmware-cmd.bat

Что попробовать:

  • Откройте приглашение Python и выполните следующую команду:

    subprocess.Popen([r"c:\Program Files\VMware\VMware Server\vmware-cmd.bat"])
    

Если это работает, то о проблемах цитирования не может быть и речи. Если нет, то вы изолировали проблему.

2 голосов
/ 30 апреля 2009

В Python в MS Windows класс subprocess.Popen использует API CreateProcess для запуска процесса. CreateProcess принимает строку, а не что-то вроде массива аргументов. Python использует subprocess.list2cmdline для преобразования списка аргументов в строку для CreateProcess.

На вашем месте я бы посмотрел, что возвращает subprocess.list2cmdline (args) (где args - первый аргумент Popen). Было бы интересно посмотреть, ставит ли оно кавычки вокруг первого аргумента.

Конечно, это объяснение может не применяться в среде Cygwin.

Сказав все это, у меня нет MS Windows.

1 голос
/ 30 апреля 2009

Я полагаю, что list2cmdline (), который выполняет обработку ваших аргументов списка, разбивает любые строковые аргументы на пробельные символы, если строка не содержит двойных кавычек. Так что я бы ожидал

vmrun_cmd = r'"c:/Program Files/VMware/VMware Server/vmware-cmd.bat"'

быть тем, кем ты хочешь.

Вы также, вероятно, захотите заключить другие аргументы (например, target_vm) в двойные кавычки при условии, что каждый из них также представляет отдельный аргумент для представления в командной строке. Что-то вроде

r'"%s"' % target_vm

(например) должно подойти.

См. список документов listcmcm

D'A

0 голосов
/ 08 апреля 2016

Это было довольно сложной проблемой для последних трех наших ... пока ничего не сказано, ничего не сработало, ни используя r "" или Popen со списком и так далее. В конце концов сработало сочетание строки формата и r "". Поэтому мое решение таково:

subprocess.Popen("{0} -f {1}".format(pathToExe, r'"%s"' % pathToVideoFileOrDir))

где обе переменные pathToExe и pathToVideoFileOrDir имеют пробелы на своем пути. Использование \ "в отформатированной строке не сработало и привело к той же ошибке, что первый путь больше не определяется правильно.

0 голосов
/ 14 июня 2009

Одна проблема заключается в том, что если команда заключена в кавычки и не имеет пробелов, это также может сбить с толку оболочку.

Итак, я делаю это:

if ' ' in raw_cmd:
    fmt = '"%s"'
else:
    fmt = '%s'

cmd = fmt % raw_cmd
0 голосов
/ 30 апреля 2009

Вот что мне не нравится

vmrun_cmd = r"c:/Program Files/VMware/VMware Server/vmware-cmd.bat"

В названии самой команды есть пробелы, что ставит в тупик вашу оболочку. Следовательно, «c: \ Program» не распознается как внутренняя или внешняя команда, работающая программа или командный файл. "

Вариант 1 - поместить файл .BAT в другое место. Действительно, поместите все свои VMWare куда-нибудь еще. Вот правило: Не используйте каталог «Program Files» для чего-либо. Это просто неправильно.

Вариант 2 - указать vmrun_cmd значение

vmrun_cmd = r'"c:/Program Files/VMware/VMware Server/vmware-cmd.bat"'
0 голосов
/ 30 апреля 2009

Возможно, глупое предложение, но, возможно, попробуйте следующее, чтобы удалить подпроцесс + пробелы из уравнения:

import os
from subprocess Popen, PIPE

os.chdir(
    os.path.join("C:", "Program Files", "VMware", "VMware Server")
)

p = Popen(
    ["vmware-cmd.bat", target_vm, list_arg, list_arg2],
    stdout=PIPE
).communicate()[0]

Возможно, стоит попробовать ..

p = Popen(
    [os.path.join("C:", "Program Files", "VMware", "VMware Server", "vmware-cmd.bat"), ...
0 голосов
/ 30 апреля 2009

2 вещи

1)

Вы, вероятно, не хотите использовать трубу Если выходные данные подпрограммы превышают 64 КБ, скорее всего, ваш процесс завершится сбоем. http://thraxil.org/users/anders/posts/2008/03/13/Subprocess-Hanging-PIPE-is-your-enemy/

2) Subprocess.Popen имеет ключевое слово аргумент shell, делая так, как если бы оболочка анализировала ваши аргументы, установка shell = True должна делать то, что вы хотите.

0 голосов
/ 30 апреля 2009

Почему вы используете r ""? Я считаю, что если вы удалите «r» с самого начала, он будет рассматриваться как стандартная строка, которая может содержать пробелы. Затем Python должен правильно заключать строку в кавычки при отправке в оболочку.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...