TCL脚本实例解读

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

TCL脚本实例解读
作者:杨帆、老卢
前言
Sigma的这段日子在技术方面感觉提高的方面可能就是脚本的编写吧,我感觉写一个可用的脚本,并不一定非的在于你对脚本有了多了解之后,然后再去实现一个切合实际的脚本,最主要是思路,当你对所需要使用的脚本工具有一定的理解(如:TCL),在一个实际环境,首先能有个明确的想法,想实现个具体的什么东西,怎么样的思路可以实现,大脑里具备了这些条件,就可以尝试去写一些脚本。

当然了,在实现的过程中肯定会遇到这样或者那样的问题,但一般来说,基本都是一些对脚本语法以及命令不熟悉的原因造成,这些问题很好解决,可以跟熟悉脚本的同事讨论,来帮你解决当时的问题,所以,千万不要因为还没有开始,就将脚本看得非常困难,导致自己一直无法迈出第一步,其实有些东西还是比较容易实现的。

所以在此将我写的几个脚本在此解读解读。

文档目的
这篇文档中所附带的脚本,主要是根据Sigma这边搭建的系统测试环境而撰写出来的,脚本内容可能与今后Sigma公司为我们所开发的系统测试脚本无关,当撰写完这几个脚本之后,各人感觉有些东西还是比较有价值的,因此本人将此脚本撰写为文档,将这些东西共享出来供大家分享、借鉴、参考,相信看完这篇文档,因该会提供很多切合实际测试的一些脚本开发思路。

还有,这篇文档并非一篇解释TCL命令的文档,有许多脚本中的相关命令不熟悉的地方需要查找其它资料,部分命令只做了解释或者介绍。

感谢!:)
这些脚本的撰写过程中感谢小康同学与小井同学的帮助(很多地方错误的查找、以及提供了一些解决关键问题的命令,如果没有他们的帮助,这个脚本完成的难度就大了,有很多也是他们的功劳)
环境介绍
这里提到的环境主要介绍的是系统测试的物理网络TOP环境与逻辑网络TOP环境,因为脚本里面的内容跟这个具体环境有关:
物理网络TOP,物理网络TOP所描述的实际是所有测试PC与DUT连接的物理
线路图,实际物理TOP分为两个部分,控制网络与测试网络:
(在这个实际环境中要求所有参与测试PC拥有两块可用的物理网卡,一块接控制网络交换机,另一块网卡连接测试网络交换机,测试网络交换机必须支持Vlan 的划分,在某种特殊情况要求有trunk功能的支持,如测试网络交换机接口不足的情况,需要添加新的交换机,不同的交换机相同VLAN之间需要通信的需求,额外话题)
控制网络:我们可以从上图看到,所有PC的网卡eth1同在同一数据链路,IP 也在相同网段,控制网络的目的是为了方便通过其中任意一台PC来控制其他的PC做一些事情,比如让某些PC在Testing网络建立一些连接,或者发送一些数据包,如何来控制,其实看完这篇文档实现的脚本,也就没有那么神秘了。

测试网络:实际这个网络就是专门构建我们DUT测试环境的网络TOP,根据划分VLAN创建不同的逻辑TOP。

比如把DUT的某个接口eth0放入某个VLAN,将一些测试PC划入此VLAN,将DUT的另一个接口eth1放入另一个VLAN,其中此VLAN划入一些PC,这个TOP相当于DUT两个接口连接了两
个不同,逻辑上的网桥。

(一个VLAN相当于是一个独立的网桥)
如下图,我们系统测试测试网络的逻辑TOP(列出供大家参考):
从上图可以见到,在实际环境中搭建,至少可能需要六台交换机(少画两台),但在这个环境只用上来两台交换机,使用Vlan和Trunk实现。

脚本介绍
在做详细的脚本分析之前我们首先先对这些脚本做一个简单的介绍。

这篇文档脚本实例中有部分是存在一定关系的,目的为执行不同的功能,但可以调用其它脚本执行所产生的变量。

以下列出脚本的文件名称,对以下脚本的功能以及实现的目的做一个简单介绍:
reach_host.tcl:
检测控制网络存在的PC,将所有存在(可ping通)的PC放入一个变量形成一个列表。

这个文档的目的是因为控制所有控制网络PC去做一些测试操作时,使用的都是IP通信,因此我们要保持测试网络的连通性,可能需要经常的去做一些ping的操作,因此做了一个这样的脚本来执行这件事情。

这个脚本是借鉴Sigma的同事修改过来的,其他所有的脚本思路都是从这个脚本出发。

这个脚本在这一系列脚本中,目的并不只是单纯的测试连通性,其实还有其他用途,比如我们的系统测环境所有PC分为四个区域,每个区域之间的PC的IP并不连续,我们可以通过这个脚本将所有可用的PC做成一个
列表,将这个列表应用在其它的脚本。

✧setup_tools.tcl:
这是一个单独得脚本,目的是将所有控制网络中可以Ping通的设备安装一些工具。

(其实这个脚本可以扩展,只需要了解其他需要安装的linux软件步骤,即可向此脚本中补充,这里面只是拿出几个例子来提供一些思路)
✧ipaddr_configure.tcl
这个脚本运用到之前reach_host.tcl运行输出的控制网络存活PC的列表,然后在对这些列表中的PC配置eth0(测试网络)的IP地址。

✧hostname_configure.tcl(此脚本由老卢完成,脚本的解释后续添加)
这个脚本运用到之前reach_host.tcl脚本运行输出的结果(所有控制网络可连同PC的IP列表),然后在对这些列表中的PC配置主机名,但每个PC的主机名的第一个字符必须相关区域的字符,比如:A区的第2台PC IP地址为10.11.105.12,B区的第8台PC地址为10.11.105.28,我们根据相同颜色来分析他们的对应关系,A对10,B对20,C对30等(具体可以看系统测试TOP的IP列表)。

✧pc_list.tcl
设置所有控制网络的IP变量,可以让其他脚本直接调用,使用其变量,避免每次新的脚本,使用到一些主机名字与IP对应关系得变量,每次都需要申明,避免麻烦就建立了一个这么列表。

相当于在C语言中,我将所有需要经常用到的常量在一个文件中逐个define,然后其他的文件include。

✧traffic_control.tcl
这个脚本可能是代码与内容最多的一个了(也是我最有成就感的一个)。

这个脚本的目的是随时地控制TOP图中存在的任意PC与另一台PC建立任意通信(不过目前脚本中只是包含ping与ftp,要将其他的通性方式加入目前非常简单只需要将步骤往脚本中填入),在执行这个脚本时,只需相应主机控制网络存活,不需要了解测试网络中的IP地址,就可以让双方建立通信。

比如:在这个系统网络中,如测试网络逻辑TOP(图二)所列出的PC,我只需要在其中某台PC运行此脚本,如FTP:
./traffic.control.tcl A1 ftp A2 get aaa 10
执行这个脚本的人不需要知道这两台主机的IP是多少(当然脚本会知道),就可以实现A1主机与A2进行FTP通信,get(可以put)一个aaa的文件,get这个操作循环10次
忘了说明,这些脚本都由TCL结合Linux下的命令实现,对以上脚本做一个简单描述只是带领大家能对脚本产生一定的兴趣,其实当看了脚本之后,你们会感觉实际实现起来并非那么神秘,就只是将Linux下的命令调来调去!
首先我们先进入第一个脚本的解释吧!
!#/usr/bin/tclsh
set reach ""
#设置变量reach,内容为空
set unreach ""
#设置变量unreach,内容为空
set hostbase "10.11.105"
#因为我们所有ip的前三个字节相等,变化的只是第四个字节,因此将前三个字节放入变量for {set i 11} {$i<49} {incr i} {
#我们的主机第四位字节最低为11,最高为48,设置i为11,小于49,incr代表i递增1 catch {exec ping $hostbase.$i -c 2} r
#exec代表在linux执行ping这个命令,-c 2代表只发送两次icmp报文(linux下默认ping是#不会停止的),catch将所有打印付给变量r。

注意:当catch “{}”中如果内容出错,此#程序不会停止,将打印全部交给r,然后继续执行下一条tcl语句,至于$hostbase.$i,我#也是当时才知道原来可以这么连着使用,在执行ping命令之前,将被解释为此次循环的ip #地址,如果此时为第三次循环$hostbase.$i为10.11.105.13
if [regexp {ttl} $r] {
#判断语句,使用regexp来匹配变量r中是否存在ttl字段(只有ping通系统才会打印此字段)puts "Host $hostbase.$i is Active!"
#如果为第三次循环将打印Host 10.11.105.13 is Active!的信息,其中$hostbase.$i将会#被替换为相应得数值
lappend reach $hostbase.$i
#lappend将这个IP地址追加到reach这个变量,建立一个列表,如果此时为第三次循环,
#reach中的变量将是”10.11.105.12 10.11.105.12 10.11.105.13”,其实所谓的列表就是在
#变量中各个字符串之间使用空格隔开
} else {
puts "Host $hostbase.$i is Down!"
#如果为假,同样也会打印
lappend unreach $hostbase.$i
#然后将这个ip追加到unreach这个变量中(行成列表)
}
}
puts "Reach Host Are:$reach"
#循环完毕,将所有ping成功的设备ip打印
puts "Unreach Host Are:$unreach"
#循环完毕,将所有ping失败的设备ip打印
看了上面这个脚本是否觉得比较简单?虽然简单,但实际用途中这个脚本是非常有用的,在系统测试实施中,既省时间又省力!而且在其他的脚本中都会用到此脚本首先做个探测,将所有可以连通的主机做一些操作,其实还有个用处,要知道,我们定义的主机区域之间的IP并不是连续的,通过这种方法我们可以把中间不连续的ip空缺给排除,这后去案例中可以看到。

在讲述第二个脚本之前,我首先在此简单的提一下关于expect这个命令,这个命令在实际运用中非常有用,在这篇文档之后的脚本中,几乎没有离开expect,expect几乎可以把你目前可以做的是用全部作完。

在执行一些脚本时,难免会碰到一些人机交互的过程,最常见的就是登陆的过程,对方要求输入用户名密码,通过shell脚本,这一部分可能需要用户本人来实现,而在TCL中使用到expect这个特性,此问题得到解决。

(首先,这篇文档并非一篇详细的关于expect文档,因此只在这简单介绍到expect的部分使用与个人理解)
在安装完TCL之后expect是需额外安装的,给人感觉就像个外挂、插件、TCL的扩展,怎么理解都行,因为在此目的是把它用上。

但安装完expect之后
(请查找相关安装文档),在脚本中用到expect,脚本执行的时候同样需要通过TCL解释器来做。

我们详细的单独拿ftp到其他设备来做个例子,让脚本自动输入用户名密码,然后退出,如下:
spawn ftp 192.168.1.1
expect “User (10.11.20.36:(none)):” {send “yangfan\r”}
expect “Password:” {send “yangfan\r”}
expect “ftp>” {send “get test.txt\r”}
expect “ftp>” {send “bye\r”}
以上是一个简单的expect过程,如果是用普通的shell脚本,其中包含ftp的通信过程,可能这一部分需要用户手动来输入相关信息。

在这个脚本中,重要提到的是spawn、expect、send这几个关键字,至于语法可以按此例照搬。

Spawn 与expect是因该成对出现,通过spawn的执行相当与打开一个expect的通道,之后的expect结果将根据spawn后面所执行的命令来做一系列的交互式操作(在此例中通过spawn将针对ftp 192.168.1.1做一系列的expect交互式操作)。

当通过spawn打开此ftp之后,将有一个全局变量spawn_id被赋值,以上脚本为例,会将系统中此ftp运行的进程id赋值给spawn_id。

当脚本执行ftp 192.168.1.1之后,第二句脚本含义是将User (10.11.20.36:(none)):字符串与expect中的buffer 匹配(每当执行一个动作,会将系统输出信息放入expect_out(buffer)这个变量),如果匹配,将执行“{}”中send语句,将此内容向ftp进程送出,”\r”代表回车。

说白了就是当ftp后,如果出现字符串User (10.11.20.36:(none)):,然后输入用户名yangfan回车到下一步,然后期待出现字符串Password:,将输入密码yangfan 回车,然后如果出现ftp>字符穿,继续输入get test.txt,最后在出现ftp>的时候再输入bye退出ftp交互过程。

简单讲述了上面的expect,我们继续看下面的脚本实例:
在讲述在各脚本之前,我首先描述一下执行环境,我当时在在控制网络的某一个pc上执行的,我首先将需要安装的软件copy到我执行此脚本的pc/root/linuxtools目录中(在linux下可以通过scp这个命令将本地的文件cp到远程主机),我想通过脚本将此目录下所有软件包通过ssh拷贝到所有可以连通
的测试网络PC上,我使用的这些安装包都是tar包,tar包需要首先通过tar命令解压缩,解完压缩后一般都是安装包相关的一个安装目录,所有安装文件都在目录下面,因此当时我已经在本地解完压缩,因此我只需要将我那几个安装包的目录cp到其它PC,然后远程进行安装,安装文件目录为一下几个
需cp到远程主机的目录:libpcap-0.9.4/ libevent-1.1a/ libpcap-0.9.4/ fragroute-1.2/ vi setup_tools.tcl
#!/usr/bin/tclsh
package require Expect
#当脚本中使用到expect特性,需要注明以上
set hostbase "10.11.105"
set hosts ""
set reach ""
set unreach ""
for {set i 11} {$i<49} {incr i} {
lappend hosts $hostbase.$i
}
#上面这一步目的是让所有10.11.105.11至10.11.105.48的pc形成一个列表置入变量$hosts puts "Seach reach hosts,waiting a moment!"
#通过puts来打印一串字符串消息
foreach host $hosts {
#foreach的含义是将$hosts的列表中ip地址逐个取出放入变量host中,每次取完一个将交给下面#的语句执行,一直循环到最后一个列表成员取完(foreach详细查看TCL相关命令解释)catch {exec ping $host -c 2} output
#如果是第三次循环,$host变量中的值因该为10.11.105.13,然后执行ping,将结果通过catch #给变量output
if [regexp {ttl} $output] {
lappend reach $host
#如果$output中有ttl,将此ip追加到列表reach中
} else {
lappend unreach $host
#如果否,将此ip追加到unreach中
}
}
puts "Reach hosts:"
puts $reach
#将所有可达的主机地址全部打印。

实际上面所有操作结果与目的同reach_host.tcl脚本一致,#只是中间穿插了一下foreach的使用方法
foreach host $reach {
#在此,我们已知$reach中包含的主机地址列表全部为控制网络可以连通的地址,我们将所有可#连通地址逐个取出来付给变量host执行下面语句
catch {spawn scp -r /root/linuxtools root@$host:/root/} a
#scp是linux下的一个命令,这个命令在此的意义是将本地/root/linuxtools目录下的所有文件#拷贝到 root@$host:/root/下面,-r参数是包含目录,$host在此是个变量,当执行时它是个从#$reach中取得的Ip地址
set timeout 40
#设置timeout时间,此时间是个全局变量,控制expect的超时时间,当expect在执行到某部阻塞#无法继续往下执行的时候,通过此设置时间进行超时
expect {
"(yes/no)" {send "yes\r";exp_continue}
"password:" {send "talent\r" }
timeout
}
#此可能需要详细解释一下
expect{
}

expect “” {}的区别
#两个在同时使用的时候实际都是单独得一个expect执行命令,前者在这条命令中expect中有多
#条期待需要执行,比如在我们实际的环境,我们使用scp或者ssh到远程的一台机器时,本地启#动后每第一次连接对方的时候都回有(yes/no)的打印,但我们不能保证每次我们所做的操作
#都是第一次连接到对方,如上例,使用expect “yes/no” {send “yes”},如果没有所期待yes/no #的输出,此时将会阻塞,但使用前者的话,如果没有yes/no的输出,将继续执行下
#一条语句寻找所期待的字符串,如此例子,如果没有yes/no的期待,会使用Password:去
#expetc_buffer寻找匹配字符串(在我们的例子中,Password:在buffer中每次肯定是会存在的)#\r代表回车,exp_continue代表执行完成之后,在继续从expect重新开始寻找匹配的字符串(注#意:这里所指的从expect开始,也就是从头开始继续寻找expect{}中所可以匹配到的字符串)# 如,telnet到linux设备之后
Expect {
“username” {send “root\r”;exp_continue}
“password” {send “abcd\r”;exp_continue }
“]#” {send “ls\r”;exp_continue }
}
# 此时会出现死循环,因为在expect中所期待的第三句”]#”,当执行ls之后,此时执行
#exp_continue ,而exp_continuel将从expect最开始寻找期待匹配的字符串(从“username”#开始),ls的输出结果交给exp_buffer(此事buffer中肯定会有”]#”的字符串,如果你想象不到#这个场景,我们可以回忆,我们在windows dos下C:>下敲一个dir,目录打印出来后,最后的字#符串依然是C:>),因此当执行到”]#”又将开始执行ls,死循环开始了
catch {spawn ssh root@$host} a
#做完上面的安装文件copy过后,在通过ssh登陆到这台设备(目的,是通过ssh在远程的pc上做#安装软件的操作),ssh root@$host转变过来就是ssh root@$10.11.105.x,root@的意思表明用#什么用户登录,这是linux下的一个命令。

expect {
"yes/no" {send "yes\r";exp_continue}
"password:" {send "talent\r";exp_continue}
}
#当登录完成之后
expect "]#" {send "cp -rf /root/linuxtools/libdnet-1.11 /usr/local/\r"}
expect "]#" {send "cp -rf /root/linuxtools/libpcap-0.9.4 /usr/local/\r"}
expect "]#" {send "cp -rf /root/linuxtools/libevent-1.1a /usr/local/\r"}
expect "]#" {send "cp -rf /root/linuxtools/fragroute-1.2 /usr/local/\r"}
expect "]#" {send "cd /usr/local/libpcap-0.9.4/\r"}
expect "libpcap-0.9.4]#" {send "./configure&&make&&make install\r"}
expect "libpcap-0.9.4]#" {send "cd /usr/local/libdnet-1.11\r"}
expect "libdnet-1.11]#" {send "./configure&&make&&make install\r"}
expect "libdnet-1.11]#" {send "cd /usr/local/libevent-1.1a\r"}
expect "libevent-1.1a]#" {send "./configure&&make&&make install\r"}
expect "libevent-1.1a]#" {send "cd /usr/local/fragroute-1.2\r"}
expect "fragroute-1.2]#" {send "./configure&&make&&make install\r"}
expect "fragroute-1.2]#" {send "exit\r";exp_continue}
}
#以上的实际整个步骤就是将cp到/root/linuxtools/下的工具安装目录在cp到/usr/local/下进#行安装,这个在安装软件过程中不是必要的,我的安装操作步骤繁琐了点,但我借介绍的只是#实现思路,其它的并不重要,但所有软件安装目录cp完了之后,在cd到每个软件安装目录下执#./configure&&make&&make install,这个是软件的安装命令,然后逐个的cd到不同的软件目录,#逐个的执行./configure&&make&&make install这条命令进行安装
puts "Host:$host fragroute is setup over!"
#然后再打印出某个Ip主机软件安装完毕
}
实际到这一步我们可以看出,整个软件安装的思路:
取出需要安装软件的PC,在一个变量hosts中形成列表
通过foreach将hosts的Ip逐个取出放入host变量
将本地所有需要安装软件拷贝到host主机(通过expect与linux下scp这个命令实现)
通过ssh登录到host这台主机在host主机上进行安装软件(expect与linux 下ssh命令实现)
软件安装完毕,通过foreach在hosts变量中取得下一台需要安装软件的主机
再作1-4描述的操作。

其实如果明白了这个脚本的思路,后续的脚本是很容易看明白的,因此,后续的脚本中,一些雷同的步骤就不再做详细解释了。

ipaddr_configure.tcl
这个脚本目的是将所有控制网络中能连通的pc,eth0网卡配置上ip地址,但遵循一定规律,如下(别忘了,eth1上的ip地址已经手动配置好了):
A区域PC 4eth1 IP 10.11.105.14eth0 IP 192.168.14.2 GW 192.168.14.1
B 区域P
C 8 eth1 IP 10.11.105.28eth0 IP 192.168.28.2 GW 192.168.28.1
也就是eth1 IP的第四个字节与eth0 IP的第三个字节相同(请参考我们控制网络的IP规律,图1已经全部标出)。

下面这个脚本将会调用我们第一个实例的脚本reach_host.tcl。

ipaddr_configure.tcl
#!/usr/bin/tclsh
package require Expect
source /root/tclscript/reach_host.tcl
#其实思路一样,给所有控制网络可以连通的设备做以下的操作,因为本身就有这么一个功能的#脚本存在(第一个案例),在这个脚本中,我们使用source命令首先来执行这个文件,这个脚#本将会产生一个变量reach(所有可以ping通PC的列表)供以下的命令使用,相当于c语言的
#include语句.
foreach host $reach {
puts "Host $host start ip configure process:\n"
#取出其中一个ip
set ipaddr [split $host {.}]
#通过split命令将Ip分割为列表赋值给ipaddr。

如host中Ip为10.11.105.11,split通过”.”将此#Ip字符串分割为”10 11 105 11”赋给ipaddr
set net [lindex $ipaddr 3]
#从ipaddr列表中通过lindex取出第4个赋值给net变量(从0开始计算,因此为3)。

catch {spawn ssh root@$host} r
#然后ssh登录在这台pc
puts "spid = $spawn_id \n"
#这条语句在这个地方使用没有什么意义,只是为当时调试,不过spawn_id这边变量在某些情况#用的着,怕到时候忘掉这个单词,因此就把它留在这供以后查找
set timeout 40
expect {
"yes/no" {send "yes\r";continue}
"password:" {send "talent\r"}
timeout
}
#通过ssh登录到这台主机
expect "]#" {send "echo
DEVICE=eth0>/etc/sysconfig/network-scripts/ifcfg-eth0\r"}
#登录后同样也是使用linux下的命令echo向/etc/sysconfig/network-scripts/ifcfg-eth0写入#配置信息,ifcfg-eth0这个配置文件保存的网卡的一些基本信息,可以通过修改此配置文件来修#改ip等基本信息,在这里需要注意的是,使用的是echo>,后续使用的是echo>>,>代表覆盖,这#里设计的含义是首先将之前这个配置文件中的信息通过第一条覆盖,后续的信息将通过追加的#方式来向这个文件补充信息
expect "]#" {send "echo
ONBOOT=YES>>/etc/sysconfig/network-scripts/ifcfg-eth0\r"}
expect "]#" {send "echo
BOOTPROTO=static>>/etc/sysconfig/network-scripts/ifcfg-eth0\r"}
expect "]#" {send "echo
IPADDR=192.168.$net.2>>/etc/sysconfig/network-scripts/ifcfg-eth0\r"} expect "]#" {send "echo
NETMASK=255.255.255.0>>/etc/sysconfig/network-scripts/ifcfg-eth0\r"} expect "]#" {send "echo
GATEWAY=192.168.$net.1>>/etc/sysconfig/network-scripts/ifcfg-eth0\r"} expect "]#" {send "service network restart\r"}
#ifcfg-eth0的信息添加完毕之后,将执行linux下service network restart这条命令,来促使#上面的配置生效。

expect "]#" {send "exit\r"}
#然后在linux送出exit,退出登录的ssh,ssh进程结束,expect也就跟着结束puts "Host $host ip configure was finished!i\n"
}
看完上面这个脚本,是否有新的想法来实现以下自己想实现的东西?
接着介绍
pc_list.tcl
这个里面实际就是控制网络ip的列表,我在这里面定义了32个变量,每个变量对应一台主机,每个变量的内容是这台主机的实际IP,目的是位了让其它的脚本使用,免得每次都需要来给这些变量赋值。

如:后续要介绍的traffic_control.tcl 就会用到这个脚本。

pc_list.tcl
#!/usr/bin/tclsh
set A1 10.11.105.11
set A2 10.11.105.12
set A3 10.11.105.13
set A4 10.11.105.14
set A5 10.11.105.15
set A6 10.11.105.16
set A7 10.11.105.17
set A8 10.11.105.18
set B1 10.11.105.21
set B2 10.11.105.22
set B3 10.11.105.23
set B4 10.11.105.24
set B5 10.11.105.25
set B6 10.11.105.26
set B7 10.11.105.27
set B8 10.11.105.28
set C1 10.11.105.31
set C2 10.11.105.32
set C3 10.11.105.33
set C4 10.11.105.34
set C5 10.11.105.35
set C6 10.11.105.36
set C7 10.11.105.37
set C8 10.11.105.38
set D1 10.11.105.41
set D2 10.11.105.42
set D3 10.11.105.43
set D4 10.11.105.44
set D5 10.11.105.45
set D6 10.11.105.46
set D7 10.11.105.47
set D8 10.11.105.48
接着我们来说明以下traffic_control.tcl这个实例吧,但首先了解这个脚本的目的,我们可以看我们的测试网络的逻辑TOP(图二),我们只需要保证控制网络可连通,执行这脚本可以指定TOP中任意一台PC与另一台PC的进行通信(脚本中包含ping、ftp),执行脚本而并不需要知道测试逻辑TOP主机的IP地址。

目前缺点是,测试网络的TOP搭建要求正确,而且每个目的主机需要提供相关服务,而且脚本只可以执行两台设备的通信,如果要主机之间的通信需要将此脚本分别执行多次,目前不支持一些排错信息。

之前我的构想是:
执行./traffic_control.tcl A1 ftp B2 get(put) filename 10
当我执行这个脚本是A1对应得主机向B2对应的主机发起ftp通信,向B2上get或者put一个filename(可指定),然后指定这个get或put的动作做10次,首先了解我的想法然后我们在具体看这个脚本内容。

A1 ftp B2 get\put filename,均作为可选参数,ftp在此同样为一个可选参数。

这个脚本可以在控制网络任意一台主机上执行。

在设计这个脚本之前,我们来看看整个过程的思路:
首先当用户输入脚本时
脚本需要知道输入这个两台主机相应的控制网络的IP。

(pc_list已经定义) 通过控制网络ip,脚本至少必须知道发起通信目标主机的IP地址。

(通过ssh 上去取得)
判断通信类型,是ping还是ftp或者其它。

(每一个通信类型都可以给不同数量或者内容的参数)
思路就这么几条,但在实现的过程中却有很多问题需要等待解决,具体看下
面脚本的解释吧!
traffic_control.tcl
#!/usr/bin/tclsh
package require Expect
source /root/tclscript/pc_list.tcl
#执行这个pc_list.tcl脚本
#source /root/tclscript/reach_host.tcl
#这条命令无效,通过#注释无效
proc trff_control {} {
#定义一个trff_control的过程
global A1 A2 A3 A4 A5 A6 A7 A8 B1 B2 B3 B4 B5 B6 B7 C1 C2 C3 C4 C5 C6 C7 C8 D1 D2 D3 D4 D5 D6 D7 D8
#因为之前执行的pc_list.tcl所有变量是个全局的变量,在过程中需要使用到的话需要在这个过#程中使用global声明一下
set C_sou_hostname "" #控制网络的主机名,将来主动发起通信方
set C_des_hostname "" #控制网络的主机名,将来主动发起方的目
set C_sou_hostip "" #控制网络的ip,通信主动发起方
set C_des_hostip "" #控制网络的ip,通信接受方
set T_sou_hostip "" #逻辑网络的ip,通信发起方
set T_des_hostip "" #逻辑网络的ip,通信接受方
#定义部分变量,这样使得以后的编写过程中更容易区分
set trff_type "" #设置通信类型,如ftp、ping、nc、nmap等等
global argv argc argv0
#因为argv argc argv0这三个变量是全局变量,在这个过程中需要用到此变量需要用global申明#一下
set C_sou_hostname [lindex $argv 0]
#我将脚本执行输入的第一个参数定义为发起方的主机名
set trff_type [lindex $argv 1]
#我将脚本执行输入的第二个参数定义为通信类型
set C_des_hostname [lindex $argv 2]
#我将脚本执行输入的第一个参数定义为接受方
set arg1 [lindex $argv 3]
set arg2 [lindex $argv 4]
set arg3 [lindex $argv 5]
set arg4 [lindex $argv 6]
#定义其它的参数,考虑到今后通信类型的不同,参数也不一样,因此暂时定义了几个为将来通#信类型命令的参数先留着,少了在加,多了也没有关系
set C_sou_hostip [subst $$C_sou_hostname]
set C_des_hostip [subst $$C_des_hostname]
#subst这个命令很关键~,因为这个命令解决了很大的问题,如果没有这个命令,可能这个脚本#就比较繁琐。

比如,我们将通过lindex $argv 0得到一个字符串A1,然后又将这个A1字符串赋#给C_des_hostname,但我的目的是想输入的A1与我们这个脚本里定义的$A1联系起来,最终目#的是想当用户输入A1时,将$A1的ip直接赋值给C_sou_hostip,但TCL不支持
set C_sou_hostip $$C_sou_hostname。

相关文档
最新文档