我有一个 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() 改变我的终端设置?有可能吗,或者这是一个错误?