Как получить вывод из внешнего командного комбайна с Pipe

StackOverflow https://stackoverflow.com/questions/1404693

  •  05-07-2019
  •  | 
  •  

Вопрос

У меня есть такая команда.

wmctrl -lp | awk '/gedit/ { print $1 }'

И я хочу выводить его в скрипте Python, я попробовал этот код

>>> import subprocess
>>> proc =  subprocess.Popen(["wmctrl -lp", "|","awk '/gedit/ {print $1}"], shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
>>> proc.stdout.readline()
'0x0160001b -1 6504   beer-laptop x-nautilus-desktop\n'
>>> proc.stdout.readline()
'0x0352f117  0 6963   beer-laptop How to get output from external command combine with Pipe - Stack Overflow - Chromium\n'
>>> proc.stdout.readline()
'0x01400003 -1 6503   beer-laptop Bottom Expanded Edge Panel\n'
>>> 

Кажется, мой код неверен, только wmctrl -lp был выполнен и | awk '{print $ 1}' опущено Мой ожидаемый вывод хотел бы 0x03800081

$ wmctrl -lp | awk '/gedit/ {print $1}'
0x03800081

Помогите ли вы, пожалуйста.

Это было полезно?

Решение

При shell = True вы должны использовать одну строку вместо массива, в противном случае ваши дополнительные аргументы интерпретируются как аргументы оболочки. Из подпроцесса документации :

  

В Unix с shell = True: если args является строкой, она указывает командную строку для выполнения через оболочку. Если args - последовательность, первый элемент задает командную строку, и любые дополнительные элементы будут рассматриваться как дополнительные аргументы оболочки.

Итак, ваш звонок должен быть:

subprocess.Popen("wmctrl -lp | sed /gedit/ '{print $1}'", shell=True, ...

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

Другие советы

Поскольку вы передаете последовательность для программы, она думает, что канал является аргументом для wmcrtrl , например, если вы это сделали

wmctrl -lp "|"

и, следовательно, фактическая операция канала потеряна.

Создание одной строки действительно должно дать вам правильный результат:

>>> import subprocess as s
>>> proc = s.Popen("echo hello | grep e", shell=True, stdout=s.PIPE, stderr=s.PIPE)
>>> proc.stdout.readline()
'hello\n'
>>> proc.stdout.readline()
''

После некоторых исследований у меня есть следующий код, который очень хорошо работает для меня. Он в основном печатает как stdout, так и stderr в режиме реального времени. Надеюсь, что это поможет кому-то еще, кто нуждается в этом.

stdout_result = 1
stderr_result = 1


def stdout_thread(pipe):
    global stdout_result
    while True:
        out = pipe.stdout.read(1)
        stdout_result = pipe.poll()
        if out == '' and stdout_result is not None:
            break

        if out != '':
            sys.stdout.write(out)
            sys.stdout.flush()


def stderr_thread(pipe):
    global stderr_result
    while True:
        err = pipe.stderr.read(1)
        stderr_result = pipe.poll()
        if err == '' and stderr_result is not None:
            break

        if err != '':
            sys.stdout.write(err)
            sys.stdout.flush()


def exec_command(command, cwd=None):
    if cwd is not None:
        print '[' + ' '.join(command) + '] in ' + cwd
    else:
        print '[' + ' '.join(command) + ']'

    p = subprocess.Popen(
        command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=cwd
    )

    out_thread = threading.Thread(name='stdout_thread', target=stdout_thread, args=(p,))
    err_thread = threading.Thread(name='stderr_thread', target=stderr_thread, args=(p,))

    err_thread.start()
    out_thread.start()

    out_thread.join()
    err_thread.join()

    return stdout_result + stderr_result

При необходимости, я думаю, что легко собрать вывод или ошибку в строку и вернуть.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top