我可以发送一些文本到在屏幕会话中运行的活动进程的STDIN吗?


70

我的Linux服务器上的屏幕会话中有一个长时间运行的服务器进程。这有点不稳定(可惜不是我的软件,所以我无法解决这个问题!),所以我想编写一个每晚重启进程的脚本,以帮助稳定。使其正常关机的唯一方法是进入屏幕进程,切换至正在运行的窗口,然后在其控制台上输入字符串“ stop”。

我是否可以做任何智能重定向操作,以使cronjob每天在固定时间发送该stop命令?

Answers:


85

这个答案不能解决问题,但是留在这里是因为30多个人发现它很有用,否则我很久以前就将其删除了。

写入/proc/*pid of the program*/fd/0。在fd子目录中包含的所有打开的文件的描述符和文件描述符0是标准输入(1是stdout和2是标准错误)。

您可以使用它在运行程序的tty上输出消息,尽管它不允许您写入程序本身。

1号航站楼:

[ciupicri@hermes ~]$ cat
shows on the tty but bypasses cat

2号航站楼:

[ciupicri@hermes ~]$ pidof cat
7417
[ciupicri@hermes ~]$ echo "shows on the tty but bypasses cat" > /proc/7417/fd/0

3
@James Lawrie:然后看看proc(5)proc.txt
Cristian Ciupitu

2
+2,无论您认为了解多少,总会有更多要学习的知识:)流畅。
troyengel 2010年

3
请注意,尽管proc fd仅重定向到用作stdin的源。在您的示例中,如果您在终端1中输入内容,它将再次打印出该内容(将其发送到cat stdin并由cat打印),从而导致您两次看到该内容。另一方面,如果您将某些内容发送到fd / 0,它将被发送到控制台,但不会发送到cat,因此仅显示一次。由于在这个示例中cat仅简单地再次打印了输入,因此您无法真正看到是否正在打印输入或输出,因此产生了这种误解。/ fd / 0指向控制台/ pts;见ls -l /proc/7417/fd/0
Kissaki

4
实际示例:我已经启动了gphoto2 --get-all-files,它要求进行100次确认。当我回显“ y”> / proc / PID / fd / 0时,gphoto2无法继续,但是,在终端中打印了“ y”。
Thorsten Staerk

2
@ThorstenStaerk,我知道,这就是我添加该注释的原因。您只在写与运行gphoto2的终端相对应的设备文件(例如/dev/pts/19),y字符无法到达应用程序本身。这类似于使用write(1)命令时发生的情况。无论如何,请尝试我的其他答案或图形自动化工具(如xdotool)
Cristian Ciupitu 2015年

36

基于屏幕的解决方案

像这样启动服务器:

# screen -d -m -S ServerFault tr a-z A-Z # replace with your server

屏幕将以分离模式启动,因此,如果要查看发生了什么,请运行:

# screen -r ServerFault

像这样控制服务器:

# screen -S ServerFault -p 0 -X stuff "stop^M"
# screen -S ServerFault -p 0 -X stuff "start^M"
# screen -S ServerFault -p 0 -X stuff "^D" # send EOF

(此答案基于将文本输入Unix&Linux兄弟站点发送到分离的屏幕上的信息)

参数说明

-d -m
   Start screen in "detached" mode. This creates a new session but doesn't
   attach to it.  This is useful for system startup scripts.
-S sessionname
   When creating a new session, this option can be used to specify a meaningful
   name for the session.
-r [pid.tty.host]
-r sessionowner/[pid.tty.host]
   resumes a detached screen session.
-p number_or_name|-|=|+
   Preselect a window. This is useful when you want to reattach to a specific
   window or you want to send a command via the "-X" option to a specific
   window.
-X
   Send the specified command to a running screen session e.g. stuff.

东西 [字符串]

   Stuff the string string in the input  buffer of the current window.
   This is like the "paste" command but with much less overhead.  Without
   a parameter, screen will prompt for a string to stuff.

基于tmux的解决方案

像这样启动服务器:

# tmux new-session -d -s ServerFault 'tr a-z A-Z' # replace with your server

tmux将以分离模式启动,因此,如果要查看发生了什么,请运行:

# tmux attach-session -t ServerFault

像这样控制服务器:

# tmux send-keys -t ServerFault -l stop
# tmux send-keys -t ServerFault Enter
# tmux send-keys -t ServerFault -l start
# tmux send-keys -t ServerFault Enter
# tmux send-keys -t ServerFault C-d # send EOF

参数说明

 new-session [-AdDP] [-c start-directory] [-F format] [-n window-name] [-s
         session-name] [-t target-session] [-x width] [-y height]
         [shell-command]
         Create a new session with name session-name.

         The new session is attached to the current terminal unless -d is
         given.  window-name and shell-command are the name of and shell
         command to execute in the initial window.  If -d is used, -x and
         -y specify the size of the initial window (80 by 24 if not
         given).

 send-keys [-lR] [-t target-pane] key ...
               (alias: send)
         Send a key or keys to a window.  Each argument key is the name of
         the key (such as `C-a' or `npage' ) to send; if the string is not
         recognised as a key, it is sent as a series of characters.  The
         -l flag disables key name lookup and sends the keys literally.

4

试试这个开始:

# screen
# cd /path/to/wd
# mkfifo cmd
# my_cmd <cmd
C-A d

并杀死:

# cd /path/to/wd
# echo "stop" > cmd
# rm cmd

3
这样做很好,但是它可能具有程序运行时无法发送其他命令的缺点。如果程序在stdin上达到EOF时停止,则第echo "xxx" > cmd一个程序将停止(因为管道将关闭)。尽管某些程序足够聪明,rewind(3)当遇到EOF时可以重新打开它们的stdin。
Cristian Ciupitu

2

可以在不运行screen实用程序或任何其他实用程序的情况下将输入文本发送到正在运行的进程。可以通过将输入文本发送到流程的标准输入“文件”来完成/proc/PID#/fd/0

但是,需要以一种特殊的方式发送输入文本,以供过程读取。通过常规文件write方法发送输入文本将不会导致进程接收文本。这是因为这样做只会附加到该“文件”,而不会触发进程读取字节。

要触发该过程以读取字节,必须对要发送的每个单个字节IOCTL执行类型的操作TIOCSTI。这会将字节放入进程的标准输入队列中。

这里用C,Perl和Python中的一些示例进行讨论:

https://unix.stackexchange.com/questions/48103/construct-a-command-by-putting-a-string-into-a-tty/48221

-

因此,要回答将近9年前提出的原始问题,cron作业将需要运行一些小型实用程序脚本/程序,类似于人们为另一个问题编写的示例,它将向该服务器进程发送字符串“ stop \ n”在问题中,通过IOCTLtype 的操作发送5个字节中的每个字节TIOCSTI

当然,这仅适用于支持TIOCSTI IOCTL操作类型的系统(例如Linux),并且仅适用于root用户帐户,因为这些“文件”由/proc/拥有root


1

万一它对任何人
都有帮助:我有一个类似的问题,并且由于我使用的过程不在screenor 之下tmux,所以我不得不采用其他方法。

我附加gdbxterm进程在其中运行的,并用于call write(5, "stop\n", 5)从其gdb写入主pty文件描述符。
我通过查找/proc/<pid>/fd/dev/ptmx两个选项之间的链接,然后通过反复试验(将我的字符串发送到两个匹配的文件描述符似乎并没有造成伤害)来确定将数据发送至哪个文件描述符。

编辑

原来,xterm我附加到的进程是spawn-new-terminal() xterm通过键绑定中的操作生成的,第二个ptmx打开的文件描述符只是未关闭ptmx的父xterm进程的文件。
因此,尝试和错误调用已将输出发送到该另一个终端。
大多数xterm进程没有两个ptmx文件描述符。

结束编辑

这样可以有效地将字符串输入终端,然后将其发送到在其下运行的进程。

注意,您可能需要允许通过类似以下内容附加到正在运行的进程
sudo bash -c "echo 0 > /proc/sys/kernel/yama/ptrace_scope"


0

由于我无法评论“克里斯蒂安·丘皮图(Cristian Ciupitu)(2010年)”这一最广为接受的答案,因此我不得不将其单独回答:

这个问题已经在以下线程中解决:https : //stackoverflow.com/questions/5374255/how-to-write-data-to-existing-processs-stdin-from-external-process

简而言之:

您必须使用stdin的管道启动进程,该管道在写入当前输入时不会阻塞也不关闭。这可以通过一个简单的无穷循环来实现,该循环将通过管道传递给相关进程:

$ (while [ 1 ]; do sleep 1; done) | yourProgramToStart

我可以确认,这与krissi打开在我的情况下不起作用的管道的方式不同。所示解决方案确实起作用了。

然后,您可以写入该进程的... / fd / 0文件以向其发送指令。唯一的缺点是,您还需要终止bash进程,该进程在服务器关闭后正在执行无限循环。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.