- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想要实现的目标:
可以通过传递不同的 IO 管道来捕获 STDOUT/STDERR,但是子进程可以检测到它不在 tty
中。 。例如git log
不会打印影响文本颜色的字符,也不会使用它的寻呼机。
使用 pty
启动进程本质上是“欺骗”子进程,使其认为它是由用户启动的。据我所知,这正是我想要的,而且其结果基本上满足了所有要求。
用于测试解决方案是否满足我的需求的一般测试是:
ls -al
通常?vim
通常?irb
通常?以下 Ruby 代码能够检查以上所有内容:
to_execute = "vim"
output = ""
require 'pty'
require 'io/console'
master, slave = PTY.open
slave.raw!
pid = ::Process.spawn(to_execute, :in => STDIN, [:out, :err] => slave)
slave.close
master.winsize = $stdout.winsize
Signal.trap(:WINCH) { master.winsize = $stdout.winsize }
Signal.trap(:SIGINT) { ::Process.kill("INT", pid) }
master.each_char do |char|
STDOUT.print char
output.concat(char)
end
::Process.wait(pid)
master.close
这在大多数情况下都有效,但事实证明它并不完美。由于某种原因,某些应用程序似乎无法切换到 raw
状态。即使vim
工作得很好,但事实证明 neovim 没有。起初我以为这是 neovim 中的一个错误,但后来我能够使用 Rust 语言的 Termion 箱重现该问题。
通过在执行前手动设置为 raw ( IO.console.raw!
),像 Neovim 这样的应用程序会按预期运行,但是像 irb
这样的应用程序不。
奇怪的是,在 Python 中生成另一个 pty,在此 pty 内,允许应用程序按预期工作(使用 python -c 'import pty; pty.spawn("/usr/local/bin/nvim")'
)。这显然不是一个真正的解决方案,但仍然很有趣。
对于我的实际问题,我想我正在寻求任何帮助来解决奇怪的问题 raw
问题,或者说,如果我完全误解了 tty/pty,我应该在哪里/如何看待问题的任何不同方向。
最佳答案
[已编辑:请参阅底部的修订更新]
明白了:)
为了真正理解这个问题,我阅读了大量有关 PTY 如何工作的内容。在我把它画出来之前,我认为我并没有真正理解它。基本上 PTY 可用于终端模拟器,这是考虑其数据流的最简单方法:
keyboard -> OS -> terminal -> master pty -> termios -> slave pty -> shell
|
v
monitor <- OS <- terminal <- master pty <- termios
(注意:这可能不是100%正确,我绝对不是这个主题的专家,只是发布它以防其他人理解它)
因此,图中我没有真正意识到的重要一点是,当您键入时,您在屏幕上看到输入的唯一原因是因为它返回(向左)到大师。
首先,这个 ruby 脚本应该首先将 tty 设置为 raw (IO.console.raw!
),它可以在执行完成后恢复它 (IO.console.raw!)。煮熟了!
)。这将确保此父 Ruby 脚本不会打印键盘输入。
第二件事是从属本身不应该是原始的,因此删除了 slave.raw!
调用。为了解释这一点,我最初添加它是因为它从输出中删除了额外的回车符:运行 echo hello
会产生 “hello\r\n”
。我错过的是这个回车符是终端仿真器的关键指令(哎呀)。
第三件事,该进程应该只与从站对话。传递 STDIN
感觉很方便,但它打乱了图中所示的流程。
这带来了如何传递用户输入的新问题,所以我尝试了这个。所以我们基本上将 STDIN
传递给 master
:
input_thread = Thread.new do
STDIN.each_char do |char|
master.putc(char) rescue nil
end
end
这种有点有效,但它有自己的问题,因为某些交互进程有时没有收到 key 。时间会证明一切,但使用 IO.copy_stream 似乎可以解决这个问题(当然读起来更好)。
input_thread = Thread.new { IO.copy_stream(STDIN, master) }
8 月 21 日更新:
因此,上面的示例大部分都有效,但由于某些原因,像 CTRL+c 这样的键仍然无法正常运行。我什至抬头看other people's approach看看我可能做错了什么,实际上,这似乎是相同的方法 - 因为 IO.copy_stream(STDIN, master) 已成功将 3 发送到主机。以下似乎没有任何帮助:
master.putc 3
master.putc "\x03"
master.putc "\003"
在我深入研究尝试用较低级别的语言实现这一目标之前,我又尝试了一种东西 - block 语法。显然, block 语法神奇地解决了这个问题。
为了防止这个答案变得过于冗长,以下方法似乎有效:
require 'pty'
require 'io/console'
def run
output = ""
IO.console.raw!
input_thread = nil
PTY.spawn('bash') do |read, write, pid|
Signal.trap(:WINCH) { write.winsize = STDOUT.winsize }
input_thread = Thread.new { IO.copy_stream(STDIN, write) }
read.each_char do |char|
STDOUT.print char
output.concat(char)
end
Process.wait(pid)
end
input_thread.kill if input_thread
IO.console.cooked!
end
Bundler.send(:with_env, Bundler.clean_env) do
run
end
关于Ruby 生成过程,捕获 STDOUT/STDERR,同时表现得就像定期生成一样,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57451744/
我有以下代码来捕获和处理运行命令输出。 如何修改它以便运行命令窗口显示输出并同时记录输出?更换 @SW_HIDE与 @SW_SHOW (或等效的)只显示一个空白的命令窗口。 类似于linux的东西te
[已编辑] 用于处理 subprocess.Popen 标准输出的 2 个选项是 stdout="a_file_name" 和 stdout=subprocess.PIPE。 stderr 可以通过
c99 的 7.19.3/7 节指出: At program start-up, three text streams are predefined and need not be opened ex
我正在运行以下 proc_open 函数。加载页面时,出现错误: Use of undefined constant STDOUT - assumed 'STDOUT'` 如何正确设置STDOUT和S
我有一个运行多个进程的开发堆栈:网络服务器、自动测试、后台编译等。所有这些都是基本的命令行命令,例如 node app.js 或 lein midje :autotest. 是否可以使用一个脚本在“后
我正在使用 SLURM 在 super 计算机上调度作业。我已设置 --output=log.out 选项,将作业标准输出中的内容放入文件 (log.out)。我发现该文件每 30-60 分钟更新一次
ansible/ansible-playbook 版本:2.1.2.0/2.2.0.0 我正在尝试使用 yum/apt 安装软件包,但由于安装软件包的存储库位于 packagecloud.io 中,有
bala@hp:~$ echo "Hello World" > stdout bala@hp:~$ cat stdout Hello World bala@hp:~$ echo "Hello Worl
在从 Fortran 编写的外部库中调用嘈杂的函数之前,我正在使用如下代码重定向标准输出: // copy standard output out = dup(STDOUT_FILENO); // c
这个问题在这里已经有了答案: How can I pipe stderr, and not stdout? (11 个答案) 关闭 6 年前。 我有一个程序,我想检查其 STDERR 输出并在其上运
我正在从 perl 运行一个 java 应用程序 这是脚本 #!/usr/bin/perl use strict; use warnings; $| = 1; my $args = join (" "
我正在尝试将 python 脚本的 STDOUT 重定向到一个文件。 如果 STDOUT 是从 sys 导入的,脚本的输出不会被重定向到一个文件: from sys import stdout std
我正在尝试使用 PHP 和 Apache(在 Docker 的前台运行)写入 stdout(或 stderr)。 这些作品: file_put_contents( "php://stderr","wo
我正在尝试重定向标准输出,以便 Windows 应用程序中的 printf 将转到我选择的文件。 我这样做: outFile = fopen("log.txt", "w"); *stdout = *o
在 Ruby 中,$stdout(前面有一个美元符号)和 STDOUT(全部大写)有什么区别?在进行输出重定向时,应该使用哪个,为什么? $stderr 和 STDERR 也是如此。 编辑: 刚找到一
让我们看看这个Hello World程序 #include int main(int argc, char ** argv) { printf("Hello, World!"); c
我在 64 位机器上运行 Ubuntu 20.04。 我想将 stdout 重定向到从 memfd_create 获得的描述符。似乎使用 memfd_create 创建的匿名文件只有在 stdout
我想在 Python 脚本末尾捕获控制台输出。也就是说,我既想正常打印到控制台,又想在执行结束时将控制台输出保存到文件中。 我看过各种相关的 SO 问题 1 , 2 , 3尽管他们要么简单地重定向输出
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我是一名优秀的程序员,十分优秀!