我试图理解ssh
的-t
选项:
-t
强制伪终端分配。这可用于在远程机器上执行任意基于屏幕的程序,这非常有用,例如在实现菜单服务时。多个 -t 选项强制分配 tty,即使 ssh 没有本地 tty。
所以,TTY 是一个设备。引用 TTY 的一种方法是通过描述符(通过打开 TTY 设备获得)。STDIN、STDOUT 和 STDERR 是描述符。但它们不一定指 TTY 设备。-t
选项强制他们引用 TTY 设备。为了理解这个选项的作用,这是正确的推理方式吗?
而 TTY 有什么特别之处,用普通的 STDIN、STDOUT 和 STDERR 可能无法实现?
-t
欢迎提供选项用例的示例。
通过哪种机制ssh
分配该 TTY?ssh
是在服务器上还是在客户端上创建新的 TTY ?如何检查这个?(/dev/
必须出现一个新节点或其他东西......)这个新的 TTY 如何与现有的 STDIN、STDOUT 和 STDERR 相关联?
否。该
-t
选项将在远程机器上运行命令,其 stdin/out/err 连接到伪 tty 从站而不是一对管道。如果(a)没有给出显式命令并且(b) ssh客户端的标准输入本身就是一个 tty ,则在 pty 中运行它是默认设置。
-t
即使不满足(a)条件,您也需要一个强制 tty 分配,而当(b)不满足时,其中两个( ) 。-tt
通过一些系统依赖的机制。如今,它主要是[1]
master_fd = posix_openpt()
遵循的标准slave_fd = open(ptsname(master_fd))
ssh
总是在服务器上创建新的伪终端,即在远程机器上。如果在服务器上分配了一个伪 tty,并且本地(客户端的)标准输入是一个 tty,则本地 tty 将被设置为原始模式。不必要。
/dev/pts
但是在 2019 年的普通 linux 机器上,每个新的伪 tty下都会出现一个新文件。就像任何其他文件描述符一样,使用
dup2(slave_fd, 0)
,dup2(slave_fd, 1)
等dup2(newfd, oldfd)
将关闭oldfd
所指的任何文件。如果分配了一个 pty,它也将成为远程会话的控制 tty。ssh -t /bin/bash
在您的登录 shell 所在的系统上csh
。如果省略-t
,/bin/bash
将在没有提示、作业控制、行编辑功能等的情况下运行。看上面 ;-)
与 tty 或其他类型的文件相比,管道不再是“普通”标准输入。
[1] 它本身有很多问题(有多个 devpts 挂载和挂载命名空间),这就是为什么将
TIOCGPTPEER
ioctl 添加到 Linux 的原因,它返回一个引用从属设备的 fd 而无需通过文件系统。