ICode9

精准搜索请尝试: 精确搜索
首页 > 系统相关> 文章详细

Linux expect错误提示:bash: spawn: command not found;couldn‘t read file|expect安装|expect详解|expect精美案例

2021-10-01 21:00:39  阅读:667  来源: 互联网

标签:spawn couldn send tcl expect exp usr


[root@RHEL7 expect]# . ssh.exp 
bash: spawn: command not found...
couldn't read file "{": no such file or directory
bash: yes/no: No such file or directory

重点:先说问题的根源。

1、务必保证已安装了expect的依赖tcl

2、务必保证安装了expect

问题根源:在执行脚本时,用的是#. file.exp 而不是 #./file.exp。问题就出在这个斜杠上。

一、shell回顾

shell的执行方式的种类:

第一种使用绝对路径执行

第二种使用相对路径执行,如./的方式

第三种使用 sh命令来执行 格式 sh 脚本名 不需要执行权限 -x参数(显示执行过程)

第四种使用 . (空格)脚本名称的方式执行 不需要执行权限 . a.sh

第五种使用 source 脚本名称 不需要执行权限(主要用于生效配置文件)

案例中的expect用第四种方法 . (空格)来执行就会报错: 

[root@RHEL7 expect]# . ssh.exp 
bash: spawn: command not found...
couldn't read file "{": no such file or directory
bash: yes/no: No such file or directory

 第二种,相对路径的方式执行就可以了。

注:普通的bash shell,以上5种执行方法均可,expect就没这么好使了,优先选第二种方式!


 特别说明:至此问题已解决,你还有宝贵的时间,可以继续,如果比较忙,下面内容可以略过!


 二、expect安装+问题还原

1.安装expect依赖tcl

1.下载依赖包

http://www.tcl.tk/software/tcltk/downloadnow84.tml

选择: tcl8.4.20-src.tar.gz

 

2.安装依赖

#tar zxf tcl8.4.20-src.tar.gz -C /usr/src/

#cd /usr/src/tcl8.4.20

#ls 可以看到解压后,该工具支持多种系统

#cd unix/ 进入unix系统

#./configure --prefix=/usr/local/tcl --enable-shared 开启共享功能

#make -j 4

#make install

 

 2.安装expect 

1.下载安装包

下载地址:
expect5.45.tar.gz
https://sourceforge.net/projects/expect/files/latest/download

 

#cd
#cd mypackage 切换到压缩包存放位置

 

 2.安装expect

#tar zxf expect5.45.tar.gz -C /usr/src  解压到指定位置
#cd /usr/src/expect5.45/ 切换到解压目录
#./configure --prefix=/usr/local/expect --with-tcl=/usr/local/tcl/lib  --with-tclinclude=/usr/src/tcl8.4.20/generic/
#make -j 4
#make install 

 注:prefix指定安装路径 ;with 添加对tcl支持 指定tcl lib库的地址 ;引用一下tcl的配置文件,使两者关联起来。

3.配置tcl和expect(创建软链接)

1.复制tcl的配置文件到上一级目录文件夹中

#cd /usr/src/tcl8.4.20/unix/ 注:tcl  本身是支持windows和mac的,需切换一下目录
#ls 
#cp tclUnixPort.h  ../generic/ 把配置文件复制一份到上级目录的文件夹下(稍后用到)

 2.给expect创建软链接(便于记忆和使用)

#ln -s /usr/local/tcl/bin/expect  /usr/local/expect/bin/expect
#ln -s /usr/local/tcl/bin/expect  /bin/expect  同时也可以连接到bin下面和/bin/bash同级目录
#ln -s /usr/local/tcl/bin/expect  /usr/bin/expect   做软链接(便于使用)

注:expect的命令实际是在Tcl包里,并不是在expect包里。
所以需要把tcl中的expect命令做一个软链接,链接到expect包里。 

附:以上是手动安装,当然也可以用yum安装,yum可以自动解决依赖问题。
#yum -y install expect 

4.编写脚本

#vim ssh.exp

#!/usr/bin/expect -f  #注:这里的-f不是必须
set timeout 30
spawn ssh root@127.0.0.1
expect {   #注:关键字与大括号之间有个空格
"yes/no"  {send "yes\r" ; exp_continue }
"password:" {send "root\r" }
}
interact

5.问题还原与处理

#chmod +x ssh.exp 给脚本添加执行权限
#. ssh.exp 注:点后面有空格。这种写法是shell的第四种执行方法(执行expet就是错误方法)

[root@RHEL7 expect]# . ssh.exp 
bash: spawn: command not found...
couldn't read file "{": no such file or directory
bash: yes/no: No such file or directory

就是这个错误,各种找度娘,问题没能找到合适的答案,这是一个细节问题。

expect的执行和bash shell的执行,无论是语法还是执行方式,都有着差异。

运行expect时,需要采用第二种,采用当前路径#./的方式执行,

三、expect深入详解

1.expect 详解

expect是一个自动化交互套件,主要应用于执行命令和程序时,系统以交互形式要求输入指定字符串,实现交互通信。

expect自动交互流程:

spawn启动指定进程---expect获取指定关键字---send向指定程序发送指定字符---执行完成退出.

2.expect常用命令

spawn               交互程序开始后面跟命令或者指定程序
expect               等待一个进程的反馈,获取匹配信息匹配成功则执行expect后面的程序动作
send    exp_send    用于发送指定的字符串信息
exp_continue        在expect中多次匹配就需要用到
send_user           用来打印输出 相当于shell中的echo
exit                 退出expect脚本
eof                  expect执行结束 退出
set                  定义变量
puts                输出变量
set timeout      设置超时时间
interact    退出自动化,开始允许用户交互

3.案例中,每行代码详解

1. [#!/usr/bin/expect]   
    这一行告诉操作系统脚本里的代码使用那一个shell来执行。这里的expect其实和linux下的bash、windows下的cmd是一类东西。   
注意:这一行需要在脚本的第一行。   

2. [set timeout 30]     

设置超时时间的,现在你只要记住他的计时单位是:秒   

注:如果设置为-1,不予超时

3. [spawn ssh root@127.0.0.1]    
    spawn是进入expect环境后才可以执行的expect内部命令,如果没有装expect或者直接在默认的SHELL下执行是找不到spawn命令的。所以不要用 “which spawn“之类的命令去找spawn命令。   它主要的功能是给ssh运行进程加个壳,用来传递交互指令。   

4. [expect "password:"]   

expect {"yes/no" { send "yes\r" ;exp_continue}"password:" { send "123456\r"}}

这里的expect也是expect的一个内部命令,expect的shell命令和内部命令是一样的,但不是一个功能,习惯就好了。

这个命令的意思是判断上次输出结果里是否包含“yes/no”的字符串,如果有则立即通过send返回一个值,继续判断是否带有password字符串 有则通过send返回一个值 ,否则就等待一段时间后返回,这里等待时长就是前面设置的30秒   

5. [send "root\r"]   
    这里就是执行交互动作,与手工输入密码的动作等效。   
    温馨提示: 命令字符串结尾别忘记加上 “\r”,如果出现异常等待的状态可以核查一下。  

6. [interact]    
    执行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。如果没有这一句登录完成后会退出,而不是留在远程终端上。

四、expect精美案例

1.写一个简略版的远程登录

#vim 1.exp

#!/usr/bin/expect
spawn ssh saneri@192.168.56.103 df -Th
expect "*password"
send "123456\n"
expect eof

2.写一个expect脚本,执行多条命令

#vim 2.exp

#!/usr/bin/expect -f
set timeout 10
spawn sudo su - root
expect "*password*"
send "123456\r"
expect "#*"
send "ls\r"
expect "#*"
send "df -Th\r"
send "exit\r"
expect eof

3.内部自定义参数的远程登录

#vim 3.exp

#!/usr/tcl/bin/expect
set timeout 30
set host "101.200.241.109"
set username "root"
set password "123456"

spawn ssh $username@$host
expect "*password*" {send "$password\r"}
interact

4.expect外部参数个数和外部参数的取用

#vim 4.exp

#!/usr/tcl/bin/expect
if {$argc < 3} {
    puts "Usage:cmd <host> <username> <password>"
    exit 1
}
set timeout -1
set host [lindex $argv 0] 
set username [lindex $argv 1]
set password [lindex $argv 2]
spawn ssh $username@$host
expect "*password*" {send "$password\r"}
interact

在expect中,$argc表示参数个数,而参数值存放在$argv中,

比如取第一个参数就是[lindex $argv 0],以此类推。

标签:spawn,couldn,send,tcl,expect,exp,usr
来源: https://blog.csdn.net/xp871038951/article/details/120580232

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有