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

2024-10-17 08:47:00
admin
原创
68
摘要:问题描述:我希望能够在启动交互式进程时注入一个初始命令,这样我就可以执行如下操作: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”之间)提供的内容将传递给命令。

相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

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

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

内置subversion和git源码管理

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

免费试用