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

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

相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   1565  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1354  
  信创国产芯片作为信息技术创新的核心领域,对于推动国家自主可控生态建设具有至关重要的意义。在全球科技竞争日益激烈的背景下,实现信息技术的自主可控,摆脱对国外技术的依赖,已成为保障国家信息安全和产业可持续发展的关键。国产芯片作为信创产业的基石,其发展水平直接影响着整个信创生态的构建与完善。通过不断提升国产芯片的技术实力、产...
国产信创系统   21  
  信创生态建设旨在实现信息技术领域的自主创新和安全可控,涵盖了从硬件到软件的全产业链。随着数字化转型的加速,信创生态建设的重要性日益凸显,它不仅关乎国家的信息安全,更是推动产业升级和经济高质量发展的关键力量。然而,在推进信创生态建设的过程中,面临着诸多复杂且严峻的挑战,需要深入剖析并寻找切实可行的解决方案。技术创新难题技...
信创操作系统   27  
  信创产业作为国家信息技术创新发展的重要领域,对于保障国家信息安全、推动产业升级具有关键意义。而国产芯片作为信创产业的核心基石,其研发进展备受关注。在信创国产芯片的研发征程中,面临着诸多复杂且艰巨的难点,这些难点犹如一道道关卡,阻碍着国产芯片的快速发展。然而,科研人员和相关企业并未退缩,积极探索并提出了一系列切实可行的解...
国产化替代产品目录   28  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

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

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

内置subversion和git源码管理

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

免费试用