Как записать данные в существующий процесс STDIN из внешнего процесса?
Я ищу способы записать данные в существующий процесс STDIN
из внешних процессов и нашел аналогичный вопрос Как вы передаете данные в STDIN программы из разных локальные/удаленные процессы в Python? в stackoverlow.
В этой теме @Michael говорит, что мы можем получить файловые дескрипторы существующего процесса на пути, как показано ниже, и разрешить записывать в них данные в Linux.
/proc/$PID/fd/
Итак, я создал простой script, указанный ниже, чтобы проверить запись данных в script STDIN
(и TTY
) из внешнего процесса.
#!/usr/bin/env python
import os, sys
def get_ttyname():
for f in sys.stdin, sys.stdout, sys.stderr:
if f.isatty():
return os.ttyname(f.fileno())
return None
if __name__ == "__main__":
print("Try commands below")
print("$ echo 'foobar' > {0}".format(get_ttyname()))
print("$ echo 'foobar' > /proc/{0}/fd/0".format(os.getpid()))
print("read :: [" + sys.stdin.readline() + "]")
В этом тесте script показаны пути STDIN
и TTY
, а затем подождите, пока он не напишет STDIN
.
Я запустил этот script и получил сообщения ниже.
Try commands below
$ echo 'foobar' > /dev/pts/6
$ echo 'foobar' > /proc/3308/fd/0
Итак, я выполнил команду echo 'foobar' > /dev/pts/6
и echo 'foobar' > /proc/3308/fd/0
с другого терминала. После выполнения обеих команд на терминале дважды отображается сообщение foobar
, на котором запущен тест script, но все. Строка print("read :: [" + sys.stdin.readline() + "]")
не была выполнена.
Есть ли способ записать данные из внешних процессов в существующий процесс STDIN
(или другие файловые дескрипторы), т.е. вызвать выполнение строки print("read :: [" + sys.stdin.readline() + "]")
из других процессов?
Ответы
Ответ 1
Ваш код не будет работать.
/proc/pid/fd/0
- это ссылка на файл /dev/pts/6
.
$echo 'foobar' > /dev/pts/6
$ echo 'foobar' > /proc/pid/fd/0
Так как обе команды записываются на терминал. Этот вход поступает на терминал, а не на процесс.
Он будет работать, если stdin - это труба.
Например, test.py
:
#!/usr/bin/python
import os, sys
if __name__ == "__main__":
print("Try commands below")
print("$ echo 'foobar' > /proc/{0}/fd/0".format(os.getpid()))
while True:
print("read :: [" + sys.stdin.readline() + "]")
pass
Запустите это как:
$ (while [ 1 ]; do sleep 1; done) | python test.py
Теперь с другого терминала напишите что-нибудь на /proc/pid/fd/0
, и оно придет к test.py
Ответ 2
Я хочу оставить здесь пример, который я нашел полезным. Это небольшое изменение в то время как истинный трюк выше, который прерывался с ошибкой на моей машине.
# pipe cat to your long running process
( cat ) | ./your_server &
server_pid=$!
# send an echo to your cat process that will close cat and in my hypothetical case the server too
echo "quit\n" > "/proc/$server_pid/fd/0"
Это было полезно для меня, потому что по определенным причинам я не мог использовать mkfifo
, что идеально подходит для этого сценария.