2016-01-23 75 views
1

我写了一个期望脚本的工作原理如下:而在循环问题expect脚本

  1. SSH到server1
  2. 在Server1 SSH到另一台服务器服务器2
  3. 从服务器2到server(服务器),然后须藤成一个用户并运行这些命令。

在脚本中,我从两个名为hostnames.out和commands.out的文件中读取要执行的主机名和命令。我使用了一个while循环遍历hostnames.out中的每个条目并从commands.out文件运行命令。

我在hostnames.out中使用单个条目测试了我的脚本,但它工作正常,但是当我添加多行时,它并未从第二行开始在主机名上运行命令。

的commands.out文件的格式是(每行一个命令):

 
ls -lrt 
hostname 
whoami 

的hostnames.out文件的格式是:

 
server1 user password server2 user password server3 user password 

我已附加的脚本参考。请让我知道问题出在哪里。

#!/usr/bin/expect 
##################################################### 
# script to automate manual works - remote 2  # 
# Gopinath           # 
##################################################### 
#Variable declaration: 

#Setting variable "prompt" for multiple prompts: 
set prompt {[]$#%]\s*$} 

#Reading commands list from file: 

set fp1 [open "commands_list_2.out" "r"] 
set file_data [read $fp1] 
close $fp1 


# read the hosts file one line at a time 
# There should be no new line at the end of the hostnames.out file 

    set fp [open "hostnames_2.out" "r"] 

    while { [gets $fp data] >= 0 } { 

      set ssh1 [lindex $data 0] 
      set ssh1_usr [lindex $data 1] 
      set ssh1_pwd [lindex $data 2] 
     set ods [lindex $data 3] 
     set ods_usr [lindex $data 4] 
      set ods_pwd [lindex $data 5] 
     set serv1 [lindex $data 6] 
     set serv1_usr [lindex $data 7] 
     set serv1_pwd [lindex $data 8] 

      puts $ssh1 
     puts $ssh1_usr 
      puts $ssh1_pwd 
     puts $ods 
     puts $ods_usr 
     puts $ods_pwd 
     puts $serv1 
     puts $serv1_usr 
      puts $serv1_pwd 

    spawn -noecho ssh [email protected]$ssh1 
    expect { 

     "*password:" { send "$ssh1_pwd\r"} 
     "*route*" { puts "login failed"; exit 1 } 
     "timed out" { puts "login failed timed out"; exit 1 } 
     } 

     expect { 
       -re $prompt { send "whoami\r"} 
      }    


        expect -re $prompt { 

          send "ssh [email protected]$ods\r" }   
        expect { 
        "password:" { send "$ods_pwd\r" } 
         } 
       } 


      expect { 
     -re $prompt { send "whoami\r"} 
      } 


        expect -re $prompt { 

          send "ssh [email protected]$serv1\r" } 
        expect { 
        "password:" { send "$serv1_pwd\r" } 
         } 

        expect -re $prompt 


     foreach a [list $file_data] { 
     send "$a" 
     expect -re prompt 
     } 
      expect -re prompt { 
       send "exit\r" 
       } 
    expect eof 
    close $fp 
` 

+0

您是否收到任何错误讯息? – Dinesh

+0

不,我没有得到任何错误信息,当我添加第二行到hostnames.out文件。 – smgopinath

回答

1

到目前为止,我所看到的,主要的问题是,你有子进程的这个复杂链条上的一个没有被清理妥善不同的主机。这是由你的代码难以阅读的不一致的缩进复合而成的。

的关键是要在Server1和Server2你的SSH会话只是 ssh来在链中的下一个服务器,和server(服务器)上的SSH会话只是做了须藤。最简单的解决方法是使用exec ssh而不是sshexec sudo而不是sudo。然后,在最内层执行exit将导致所有这些中间程序也终止,并且您的expect eof将触发。

spawn -noecho ssh [email protected]$ssh1 
expect { 
    "*route*" { puts "login failed"; exit 1 } 
    "timed out" { puts "login failed timed out"; exit 1 } 
    "*password:" { send "$ssh1_pwd\r"; exp_continue } 
    -re $prompt { send "whoami\r" } 
} 
expect -re $prompt { 
    send "exec ssh [email protected]$ods\r" 
} 
expect { 
    "password:" { send "$ods_pwd\r"; exp_continue } 
    -re $prompt { send "whoami\r" } 
} 
expect -re $prompt { 
    send "exec ssh [email protected]$serv1\r" 
} 
expect { 
    "password:" { send "$serv1_pwd\r"; exp_continue } 
    -re $prompt { send "whoami\r" } 
} 

# You seem to be missing the sudo here? 

foreach a [split $file_data "\n"] { 
    expect -re $prompt 
    send "$a" 
} 

expect -re $prompt 
send "exit\r" 
expect eof 
close;   ### TERMINATE THE SPAWN. IMPORTANT! 

您也有您正在使用[list $file_data]时,你应该有[split $file_data "\n"]打破它由线的问题。


上面的代码需要循环遍历每组证书。这部分是最容易被写成这样的事情(因为你真的不会有足够的数据线需要流处理):

set fp [open "hostnames_2.out" "r"] 
set hostnamedata [split [read $fp] "\n"] 
close $fp 

foreach line $hostnamedata { 
    lassign $line ssh1 ssh1_usr ssh1_pwd ods ods_usr ods_pwd serv1 serv1_usr serv1_pwd 

    # The code from above goes here 
} 

分别建议考虑设置在所有中间服务器上使用SSH公钥,并使用ssh的-A选项。这可以大大简化认证/授权部分而不会影响安全性。

+0

也注意到你可以传递一个命令来执行到ssh,比如'ssh user1 @ serv1 ssh user2 @ serv2 sudo yourcommand'。现在你只需要处理密码的东西。 –