使用其 PID 捕获具有 Python 的 Linux 进程的标准输出
Capturing stdout of a Linux Process with Python using its PID
我在后台 Linux 运行ning 中有一个进程写入标准输出。
我不能 运行 它作为 python 脚本中的 subprocess
。
是否有任何方法可以使用其 PID 从 python 脚本中捕获其标准?在 bash.
中使用 strace -p<pid> -e write
编辑:
- 我正在寻找类似 python
Queues
或 Pipes
的内容。
- 我想避免改变调用后台进程的方式(例如将标准输出转发到 pipe/file)。
- 后台进程和 python 脚本将由同一用户启动。
如何处理这个问题取决于一些细节,例如谁拥有后台进程,以及您可以更改它的启动方式。
情况 1:不是您的后台进程
如果您不拥有后台进程,则无法读取其输出,除非该进程以某种方式使其输出可用。 (想象一下,如果一个进程可以捕获其他任何人的进程输出...其他人的脚本将能够读取您的电子邮件 reader 或文本编辑器的控制台输出。)
案例 2:您的后台进程
即使您的 Python 脚本将由 运行 后台进程的同一用户 运行 执行,调用者使用其文件描述符也已经太晚了在它开始之后。让后台进程首先将其输出发送到任何你想要的地方会更好——要么通过重定向它的标准输出,要么通过重写进程以记录到一个文件。
选项:使用 tee
拆分标准输出
如果其他内容依赖于该后台进程的标准输出,您可以通过 tee
传输输出,这会将其所有输入的副本保存在一个文件中,然后将该输入重复为标准输出。
$ cat out.txt
cat: out.txt: No such file or directory
$ echo "This is a test." | tee out.txt | wc
1 4 16
$ cat out.txt
This is a test.
使用 tee -a <filename>
附加到文件而不是覆盖它。
选项:/proc/
文件系统技巧
您可以尝试阅读/proc/${bg_process_pid}/fd/1
,如中所述。我对 /proc/
文件描述符符号链接的有限经验是它们很少做你想做的事,但我只从命令行尝试过。
充其量,我可以让字符显示在正确的终端 window,但它们不会被捕获为 output(也不会被视为某些输入子进程)。我不确定这是 bash
还是 /dev/pts/${some_number}
伪终端干扰了我正在尝试做的事情,Linux 中的某些安全功能,或者我只是没有做对。
案例 3:守护进程
如果后台进程是完全成熟的守护进程(值得怀疑,但有些人在他们说 "background process" 时包括守护进程),它会在启动后立即完全脱离其控制终端 --- 所以它根本不会有通常意义上的 stdout
。如果您有权获得其 /proc/${pid}/fd/0
和 1
的长列表,您会看到它们是指向 /dev/null
.
的符号链接
我发现一些服务器进程似乎将 stderr
重定向到一个文件,但它们的系统启动脚本不会这样做。我怀疑打开的文件描述符 2 实际上是由守护进程显式打开的,只是碰巧重用了数字 2。
这是一个在老式 Fedora Linux 服务器上的 Apache httpd
守护进程的例子:
# ls -l /proc/1408/fd/[012]
lr-x------. ... /proc/1408/fd/0 -> /dev/null
l-wx------. ... /proc/1408/fd/1 -> /dev/null
l-wx------. ... /proc/1408/fd/2 -> /var/log/httpd/error_log
Samba 的 nmbd
和 smbd
进程做了类似的事情。
我在后台 Linux 运行ning 中有一个进程写入标准输出。
我不能 运行 它作为 python 脚本中的 subprocess
。
是否有任何方法可以使用其 PID 从 python 脚本中捕获其标准?在 bash.
中使用strace -p<pid> -e write
编辑:
- 我正在寻找类似 python
Queues
或Pipes
的内容。 - 我想避免改变调用后台进程的方式(例如将标准输出转发到 pipe/file)。
- 后台进程和 python 脚本将由同一用户启动。
如何处理这个问题取决于一些细节,例如谁拥有后台进程,以及您可以更改它的启动方式。
情况 1:不是您的后台进程
如果您不拥有后台进程,则无法读取其输出,除非该进程以某种方式使其输出可用。 (想象一下,如果一个进程可以捕获其他任何人的进程输出...其他人的脚本将能够读取您的电子邮件 reader 或文本编辑器的控制台输出。)
案例 2:您的后台进程
即使您的 Python 脚本将由 运行 后台进程的同一用户 运行 执行,调用者使用其文件描述符也已经太晚了在它开始之后。让后台进程首先将其输出发送到任何你想要的地方会更好——要么通过重定向它的标准输出,要么通过重写进程以记录到一个文件。
选项:使用 tee
拆分标准输出
如果其他内容依赖于该后台进程的标准输出,您可以通过 tee
传输输出,这会将其所有输入的副本保存在一个文件中,然后将该输入重复为标准输出。
$ cat out.txt
cat: out.txt: No such file or directory
$ echo "This is a test." | tee out.txt | wc
1 4 16
$ cat out.txt
This is a test.
使用 tee -a <filename>
附加到文件而不是覆盖它。
选项:/proc/
文件系统技巧
您可以尝试阅读/proc/${bg_process_pid}/fd/1
,如/proc/
文件描述符符号链接的有限经验是它们很少做你想做的事,但我只从命令行尝试过。
充其量,我可以让字符显示在正确的终端 window,但它们不会被捕获为 output(也不会被视为某些输入子进程)。我不确定这是 bash
还是 /dev/pts/${some_number}
伪终端干扰了我正在尝试做的事情,Linux 中的某些安全功能,或者我只是没有做对。
案例 3:守护进程
如果后台进程是完全成熟的守护进程(值得怀疑,但有些人在他们说 "background process" 时包括守护进程),它会在启动后立即完全脱离其控制终端 --- 所以它根本不会有通常意义上的 stdout
。如果您有权获得其 /proc/${pid}/fd/0
和 1
的长列表,您会看到它们是指向 /dev/null
.
我发现一些服务器进程似乎将 stderr
重定向到一个文件,但它们的系统启动脚本不会这样做。我怀疑打开的文件描述符 2 实际上是由守护进程显式打开的,只是碰巧重用了数字 2。
这是一个在老式 Fedora Linux 服务器上的 Apache httpd
守护进程的例子:
# ls -l /proc/1408/fd/[012]
lr-x------. ... /proc/1408/fd/0 -> /dev/null
l-wx------. ... /proc/1408/fd/1 -> /dev/null
l-wx------. ... /proc/1408/fd/2 -> /var/log/httpd/error_log
Samba 的 nmbd
和 smbd
进程做了类似的事情。