使用其 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

编辑:

如何处理这个问题取决于一些细节,例如谁拥有后台进程,以及您可以更改它的启动方式。

情况 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/01 的长列表,您会看到它们是指向 /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 的 nmbdsmbd 进程做了类似的事情。