3

Я пытаюсь найти способ проверить существование команды оболочки перед ее выполнением.

Например, я выполню команду ack-grep. Итак, я пытаюсь сделать:

подпроцесс импорта
из подпроцесса импорт ТРУБА

cmd_grep = подпроцесс.Popen(["ack-grep", "--no-color", "--max-count = 1", "--no-group", "def run_main", "../cgedit/"], stdout = ТРУБА, stderr = ТРУБА)

Чем, если я выполню

cmd_grep.stderr.read()

Я получаю '', как на выходе. Но у меня нет команды ack-grep на моем пути. Итак, почему Popen не помещает сообщение об ошибке в мою переменную .stderr?

Кроме того, есть ли более простой способ сделать то, что я пытаюсь сделать?

5 ответов5

3

Вы можете использовать модуль подпроцесса под Python 3 или модуль команд для Python 2 следующим образом:

status, result = subprocess.getstatusoutput("ls -al")

status, result = commands.getstatusoutput("ls -al")

Затем проверьте значение status .

Примеры с сайта:

>>> import subprocess
>>> subprocess.getstatusoutput('ls /bin/ls')
(0, '/bin/ls')
>>> subprocess.getstatusoutput('cat /bin/junk')
(256, 'cat: /bin/junk: No such file or directory')
>>> subprocess.getstatusoutput('/bin/junk')
(256, 'sh: /bin/junk: not found')
2

не могли бы вы использовать команду "который" как-то? команда which автоматически выполняет поиск приложения в путях. Я думаю, что вам просто нужно будет вызвать эту команду и передать имя команды, которую вы хотите просмотреть, а затем проанализировать результаты.

1

Для подобных ситуаций я использую:

def find_program(prog_filename, error_on_missing=False):
    bdirs = ['$HOME/Environment/local/bin/',
             '$HOME/bin/',
             '/share/apps/bin/',
             '/usr/local/bin/',
             '/usr/bin/']
    paths_tried = []
    for d in bdirs:
        p = os.path.expandvars(os.path.join(d, prog_filename))
        paths_tried.append(p)
        if os.path.exists(p):
            return p
    if error_on_missing:
        raise Exception("*** ERROR: '%s' not found on:\n  %s\n" % (prog_filename, "\n  ".join(paths_tried)))
    else:
        return None

Тогда вы можете сделать что-то вроде:

grep_path = find_program('ack_grep', False)
if grep_path is None:
    # default to standard system grep
    grep_path = 'grep'
0

Я, наконец, оставил это работать так:

пытаться:

    cmd_grep = ["ack-grep", "--no-color", "--max-count=1", "--no-group", function_definition, file_path]

    first_exec = Popen(cmd_grep,stdout=PIPE)

    execution = Popen(cmd_sed, shell=True, stdin=first_exec.stdout, stdout=PIPE)

except:

    #use grep instead

    cmd_grep = cmd_grep = r'grep -R -n "' + function_definition + '" ' + file_path

    execution = Popen(cmd_grep + '|' + cmd_sed,shell=True,stdout=PIPE)

output = execution.stdout.read()
-1

Это должно сработать для вас:

import subprocess 
from subprocess import PIPE
import shutil 

cmd = ["ack-grep", "--no-color", "--max-count=1", "--no-group", "def run_main", "../cgedit/"]
path_to_cmd = shutil.which(cmd[0])    
if path_to_cmd is None:
     print(f"There is no {cmd[0]} on the path!")
else:
    cmd_grep = subprocess.Popen(cmd,  stdout=PIPE, stderr=PIPE) 

Мыслительный процесс ......... Я использовал это решение, чтобы начать:

import subprocess
status, response = subprocess.getstatusoutput('which ack-grep')
print(f'Exitcode: {status}, Output: {response}')

Но это привело к определенному ответу только от систем Linux.

import shutil
shutil.which(cmd[0])

Возвращает путь к исполняемому файлу, когда вызывается «ack-grep», или None, если ничего не вызывается.

Всё ещё ищете ответ? Посмотрите другие вопросы с метками .