我如何将初始输入传送到可交互的进程中?

2024-10-17 08:47:00
admin
原创
207
摘要:问题描述:我希望能够在启动交互式进程时注入一个初始命令,这样我就可以执行如下操作:echo "initial command" | INSERT_MAGIC_HERE some_tool tool> initial command [result of init...

问题描述:

我希望能够在启动交互式进程时注入一个初始命令,这样我就可以执行如下操作:

echo "initial command" | INSERT_MAGIC_HERE some_tool

tool> initial command 

[result of initial command] 

tool> [now I type an interactive command]

无效的方法:

  • 仅通过管道传输初始命令不起作用,因为这会导致 stdin 无法连接到终端

  • 写入 /dev/pts/[number] 会将输出发送到终端,而不是像来自终端一样输入到进程

但有缺点的是:

  • 创建一个命令,该命令分叉一个子进程,写入其标准输入,然后从其自己的标准输入转发所有内容。缺点 - 终端控制(如行模式与字符模式)不起作用。也许我可以用伪终端代理做些什么?

  • 制作一个 xterm 的修改版本(无论如何我都会为这个任务启动一个),并在遇到所需的提示字符串后使用命令行选项注入其他命令。太丑了。

  • 制作我正在尝试运行的工具的修改版本,以便它接受命令行上的初始命令。破坏标准安装。

(顺便说一下,目前感兴趣的工具是 android 的 adb shell - 我想在手机上打开一个交互式 shell,自动运行一个命令,然后进行交互式会话)


解决方案 1:

您不需要编写新工具来转发stdin- 已经编写了一个(cat):

(echo "initial command" && cat) | some_tool

这确实存在将管道连接到而some_tool不是终端的缺点。

解决方案 2:

接受的答案很简单,而且大多是好的。

但它有一个缺点:程序将管道作为输入,而不是终端。这意味着自动完成将不起作用。在很多情况下,这还会禁用漂亮的输出,而且我听说如果 stdin 不是终端,某些程序就会拒绝工作。

以下程序解决了这个问题。它创建一个伪终端,生成一个连接到该伪终端的程序。它首先提供通过命令行传递的额外输入,然后提供用户通过 stdin 提供的输入。

例如,ptypipe "import this" python3让 Python 首先执行“import this”,然后将您带到交互式命令提示符,其中包含工作完成和其他内容。

同样,ptypipe "date" bash运行 Bash,它会执行date并返回一个 shell。同样,具有工作完成、彩色提示符等。

#!/usr/bin/env python3

import sys
import os
import pty
import tty
import select
import subprocess

STDIN_FILENO = 0
STDOUT_FILENO = 1
STDERR_FILENO = 2

def _writen(fd, data):
    while data:
        n = os.write(fd, data)
        data = data[n:]

def main_loop(master_fd, extra_input):
    fds = [master_fd, STDIN_FILENO]

    _writen(master_fd, extra_input)

    while True:
        rfds, _, _ = select.select(fds, [], [])
        if master_fd in rfds:
            data = os.read(master_fd, 1024)
            if not data:
                fds.remove(master_fd)
            else:
                os.write(STDOUT_FILENO, data)
        if STDIN_FILENO in rfds:
            data = os.read(STDIN_FILENO, 1024)
            if not data:
                fds.remove(STDIN_FILENO)
            else:
                _writen(master_fd, data)

def main():
    extra_input = sys.argv[1]
    interactive_command = sys.argv[2]

    if hasattr(os, "fsencode"):
        # convert them back to bytes
        # http://bugs.python.org/issue8776
        interactive_command = os.fsencode(interactive_command)
        extra_input = os.fsencode(extra_input)

    # add implicit newline
    if extra_input and extra_input[-1] != b'
':
        extra_input += b'
'

    # replace LF with CR (shells like CR for some reason)
    extra_input = extra_input.replace(b'
', b'
')

    pid, master_fd = pty.fork()

    if pid == 0:
        os.execlp("sh", "/bin/sh", "-c", interactive_command)

    try:
        mode = tty.tcgetattr(STDIN_FILENO)
        tty.setraw(STDIN_FILENO)
        restore = True
    except tty.error:    # This is the same as termios.error
        restore = False

    try:
        main_loop(master_fd, extra_input)
    except OSError:
        if restore:
            tty.tcsetattr(0, tty.TCSAFLUSH, mode)

    os.close(master_fd)
    return os.waitpid(pid, 0)[1]

if __name__ == "__main__":
    main()

(注意:我担心这个解决方案可能存在死锁。您可能需要以小块的形式提供 extra_input 以避免这种情况)

解决方案 3:

使用“expect”程序可以轻松实现这一点,该程序旨在让您编写脚本来与程序进行交互。

我通过编写一个预期脚本 bc.exp 来测试这一点,以启动计算器“bc”并向其发送命令“obase=16”以使其进入十六进制输出模式,然后将控制权交给我。

该脚本(在名为 bc.exp 的文件中)是

spawn bc
send "obase=16
"
interact {
  exit
}

一个人运行

expect bc.exp

解决方案 4:

@caf 的回答很可靠。

我想我会在这里用一个对我来说很有效的相关选项来扩展它。

我的<initial command>实际上是命令列表,位于文本文件中。因此,我想将其导入<some command>并连接到stdin

cat处理这个问题很好,它接受-读取stdin

cat init_file - | some_command

这与 cafs 答案中讨论的有相同的限制。

解决方案 5:

您还可以(在某些情况下,从终端运行时)使用tee它直接写入终端输出。名称tee指的是像 T 一样,将 stdin 传递到 stdout,同时写入文件(/dev/tty,在本例中为终端)。

echo "initial command" | tee /dev/tty | some_tool

https://unix.stackexchange.com/a/178754/89546

解决方案 6:

如果您在 tmux 中运行,您可以告诉它发送按键。

例如

$ stty -echo; tmux send-keys test; stty echo

将输入test到您的终端输入。(stty 阻止在终端上看到按键)

或者如果你使用 vi 模式:

stty -echo
tmux send-keys escape 0
stty echo
read -p "rename: " -e -i 'old name' new_name

这会将 escape 和 0 放入输入缓冲区,以将以下 readline 置于命令模式,并将光标移动到 0。

解决方案 7:

也许您可以使用此处的文档将您的输入传递给abd。例如像这样(使用bc进行简单计算作为示例)。

[axe@gromp ~]$ bc <<END
> 3 + 4
> END
7

会话bc随后保持打开状态,因此开始和结束标记之间(“<<END”和“END”之间)提供的内容将传递给命令。

相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1136  
  IPD(Integrated Product Development,集成产品开发)流程是一种广泛应用于高科技和制造业的产品开发方法论。它通过跨职能团队的紧密协作,将产品开发周期缩短,同时提高产品质量和市场成功率。在IPD流程中,CDCP(Concept Decision Checkpoint,概念决策检查点)是一个关...
IPD培训课程   90  
  研发IPD(集成产品开发)流程作为一种系统化的产品开发方法,已经在许多行业中得到广泛应用。它不仅能够提升产品开发的效率和质量,还能够通过优化流程和资源分配,显著提高客户满意度。客户满意度是企业长期成功的关键因素之一,而IPD流程通过其独特的结构和机制,能够确保产品从概念到市场交付的每个环节都围绕客户需求展开。本文将深入...
IPD流程   84  
  IPD(Integrated Product Development,集成产品开发)流程是一种以跨职能团队协作为核心的产品开发方法,旨在通过优化资源分配、提高沟通效率以及减少返工,从而缩短项目周期并提升产品质量。随着企业对产品上市速度的要求越来越高,IPD流程的应用价值愈发凸显。通过整合产品开发过程中的各个环节,IPD...
IPD项目管理咨询   92  
  跨部门沟通是企业运营中不可或缺的一环,尤其在复杂的产品开发过程中,不同部门之间的协作效率直接影响项目的成败。集成产品开发(IPD)作为一种系统化的项目管理方法,旨在通过优化流程和增强团队协作来提升产品开发的效率和质量。然而,跨部门沟通的复杂性往往成为IPD实施中的一大挑战。部门之间的目标差异、信息不对称以及沟通渠道不畅...
IPD是什么意思   89  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用