9

我有一个 Python 脚本,用于使用 Python 子进程模块跨多个主机并行执行命令。它包装了 SSH,基本上是这样调用的:

output = subprocess.Popen(["/bin/env", env, "/usr/bin/ssh", "-t", "%s@%s" % (user, host), "--", command], stdout=subprocess.PIPE, stderr=subprocess.PIPE).communicate()

有效的命令是这样执行的:

/bin/env TERM=$TERM:password /usr/bin/ssh -t "%s@%s" % (user, host), "--", command

它工作正常,除了在运行脚本后我的终端出现混乱(丢失换行符)时出现间歇性错误。命令行中的“重置”可以修复它,但我不确定这是如何发生的。我注意到有时元组输出中第一项的末尾有一个“\r\n” ,有时它不存在。请参阅以下内容,特别是“ Permission denied\r\n”

**** Okay output ****
[user@/home/user]# ./command.py hosts.lists "grep root /etc/shadow"
Running command "grep root /etc/shadow" on hosts in file "hosts.test"
('grep: /etc/shadow: Permission denied\r\n', 'Connection to server1.example.com closed.\r\n')
('grep: /etc/shadow: Permission denied\r\n', 'Connection to server2.example.com closed.\r\n')
[user@/home/user]#


**** Output causes terminal to not display newlines ****
[user@/home/user]# ./command.py hosts.list "grep root /etc/shadow"
('grep: /etc/shadow: Permission denied\r\n', 'Connection to server1.example.com closed.\r\n')
('grep: /etc/shadow: Permission denied\n', 'Connection to server2.example.com closed.\r\n')
                                 [user@/home/user]# [user@/home/user]# [user@/home/user]

第二个输出稍作修改,但显示了丢失的"\r",以及运行脚本后我的提示如何“变坏”。

我认为这与在我的子进程命令中使用“-t”选项有关。不知何故,我失去了 \r。如果我删除“-t”选项,这个问题就会消失,但长话短说,我需要它来传递环境变量以在远程机器上使用(我正在使用 TERM 变量来传递用户的密码sudo 的目的,因为我不能假设 AcceptEnv 允许在远程 sshd 服务器上传递任意变量;我这样做是为了避免在命令行上传递密码,这将显示在远程机器上的进程列表中)。

只是想知道是否有人知道在不删除“-t”选项的情况下解决此问题的方法?

更新:在我的脚本中运行subprocess.Popen(...).communicate()命令 后,我的 tty 设置似乎发生了变化,无论我是否实际将输出打印到屏幕上。我觉得这真的很奇怪。以下是我的 tty 配置中的前后差异(来自 stty -a):

-ignbrk brkint ignpar -parmrk -inpck -istrip -inlcr -igncr icrnl ixon -ixoff
-ignbrk brkint ignpar -parmrk -inpck -istrip -inlcr -igncr -icrnl -ixon -ixoff


opost -olcuc -ocrnl onlcr -onocr -onlret -ofill -ofdel nl0 cr0 tab0 bs0 vt0 ff0
-opost -olcuc -ocrnl onlcr -onocr -onlret -ofill -ofdel nl0 cr0 tab0 bs0 vt0 ff0

isig icanon -iexten echo echoe echok -echonl -noflsh -xcase -tostop -echoprt
-isig -icanon -iexten -echo -echoe -echok -echonl -noflsh -xcase -tostop -echoprt

我想知道如何阻止communicate() 改变我的终端设置?有可能吗,或者这是一个错误?

4

2 回答 2

9

我找到

stty sane

将控制台恢复到以前的状态。我真的不明白这里的其他答案,所以我这对某人有帮助。

在这里找到了答案。

于 2018-02-01T11:33:18.710 回答
5

我在 Perl 脚本中遇到了同样的问题。为了解决这个问题,我必须保存本地终端的当前设置(以便在脚本末尾恢复它)并在执行远程命令之前添加“stty -raw”。

所以在 Perl 中:

#保存当前终端设置(您可以在文件名中添加PID)

`stty -g > ~/tmp/.currentTtySettings`;

#执行以“stty -raw”为前缀的远程命令

我的@out=`ssh -t -q user@server1.example.com "stty -raw ; grep root /etc/shadow"`;

#恢复终端设置

`stty \`cat ~/tmp/.currentTtySettings\``;

希望对你有帮助!

其他非常有用的链接:

- ssh和tty的详细解释(-t选项)https://unix.stackexchange.com/questions/151916/why-is-this-binary-file-being-changed

-一些 Perl 和 ssh 的灵感http://search.cpan.org/~bnegrao/Net-SSH-Expect-1.09/lib/Net/SSH/Expect.pod

- 如何避免 sudo 的“-t” https://unix.stackexchange.com/questions/122616/why-do-i-need-a-tty-to-run-sudo-if-i-can-sudo-无密码

于 2014-12-12T21:12:32.880 回答