需求分析
公司要求做自动化测试,在交换芯片流片前测试芯片的功能,在palladium上跑综合网表,仿真实现芯片的功能,具体如何实现的就不细说了,不是重点TAT。我目前的工作就是负责能够用脚本SSH远程服务器,执行Linux下执行一些之前写好的测试程序(芯片的配置代码),同时在配置好芯片后,使用脚本控制测试仪,使得测试仪构造报文,发流停流等功能。
由于我们租用的思博伦测试仪版本太老,只支持TCL脚本控制,不支持Python控制。由于之前没有用过TCL,没办法也得把TCL基本语法学一遍啦。不过好在各种编程语言有一定的想通性,稍微看看试试基本语法就能编写一些测试例啦,最主要的是TCL控制思博伦测试仪的一些API比较麻烦,需要慢慢找,不过测试例跑通之后也比较好添加。
得知老板需求之后就去查解决办法啦,当时也比较死脑筋,想着底层只能用TCL控制测试例,SSH等远程功能也就用TCL实现吧。于是找啊,试啊,看着TCL的expect包好像能行,但是用着我们能够适应测试仪版本的TCL版本怎么都不行,折腾了几天没有办法SSH。没办法SSH就没有办法自动化测试,不能自动回归测试例,测试效率就会低很多啊。
后来灵机一动,既然对Python比较熟悉,何不用Python完成这一切呢,远程控制,执行TCL。用了Python感觉进度势如破竹啊,半天就基本能够搞通啦。
虽然遇到点小问题纠结了好久,Python的Tinker里面的TCL控制不了测试仪啊,版本不对啊,折腾了一天,终于通过控制Python版本(32位python2.7版本)实现了调整TCL版本控制思博伦测试仪。
哈哈哈,环境全部搭建好之后自己封装了一些类啊,写了一些函数啊,以及针对我目前的工作环境如何完整得SSH和Telnet执行完一条配置命令。由于我们有测试shaper精度以及抖动的问题,写了个画折线图以及计算平均速度最大速度最小速度的函数。保存速度数据至CSV文件,以及取SSH以及Telnet到的shell的输出的某些字段。
Telnet,SSH连接以及发送命令代码
telnet代码
import logging
import telnetlib
import time
class TelnetClient(object):
def __init__(self, host_ip, username, port=23):
self.tn = telnetlib.Telnet()
self.host = host_ip
self.username = username
self.port = port
self._connect()
# telnet登录主机
def _connect(self):
try:
# self.tn = telnetlib.Telnet(host_ip,port=23)
self.tn.open(self.host, self.port)
except:
logging.warning('%s网络连接失败'%self.host)
return False
# 等待login出现后输入用户名,最多等待10秒
self.tn.read_until(b'login: ',timeout=10)
self.tn.write(username.encode('ascii') + b'\n')
# 等待Password出现后输入用户名,最多等待10秒
time.sleep(2)
# 获取登录结果
# read_very_eager()获取到的是的是上次获取之后本次获取之前的所有输出
command_result = self.tn.read_very_eager().decode('ascii')
if 'Login incorrect' not in command_result:
logging.warning('%s登录成功'%self.host)
return True
else:
logging.warning('%s登录失败,用户名错误'%self.host)
return False
# 此函数实现执行传过来的命令,并输出其执行结果
def execute_some_command(self,command):
# 执行命令
self.tn.write(command.encode('ascii')+b'\n')
time.sleep(2)
# 获取命令结果
command_result = self.tn.read_very_eager().decode('ascii')
logging.warning('命令执行结果:\n%s' % command_result)
def sendWithCheck(self, command, print_c = 1, timeout = 400):
self.tn.write(command+'\n')
recv = self.tn.read_until('kd5886>',timeout=timeout)
if print_c: print recv
return recv
def send(self, command, print_c = 1, sleeptime = 60):
self.tn.write(command+'\n')
time.sleep(sleeptime)
command_result = self.tn.read_very_eager()
if print_c : print command_result
return command_result
# 退出telnet
def close(self):
self.tn.write(b"exit\n")
if __name__ == '__main__':
host_ip = '192.168.220.129'
username = 'root'
password = 'abcd1234'
command = 'whoami'
telnet_client = TelnetClient()
SSH代码
class SSHConnection(object):
def __init__(self, host, username, password, port = 22):
self._host = host
self._port = port
self._username = username
self._password = password
self._ssh = None
self._shell = None
self._sftp = None
self._client = None
self._connect() # 建立连接
def _connect(self):
#创建一个ssh的白名单
ssh = paramiko.SSHClient()
know_host = paramiko.AutoAddPolicy()
#加载创建的白名单
ssh.set_missing_host_key_policy(know_host)
ssh.connect(hostname = self._host, port = self._port, username=self._username, password=self._password)
self._ssh = ssh
#执行命令
def exec_command(self, command):
if self._ssh is None:
self._ssh = paramiko.SSHClient()
stdin, stdout, stderr = self._ssh.exec_command(command)
data = stdout.read()
if len(data) > 0:
print(data.strip()) #打印正确结果
return data
err = stderr.read()
if len(err) > 0:
print(err.strip()) #输出错误结果
return err
#发送命令包含尾字符检查
def sendWithCheck(self, command, sleeptime = 1, print_c = 1, timeout = 20):
if self._ssh is None:
self._ssh = paramiko.SSHClient()
if self._shell is None:
self._shell = self._ssh.invoke_shell()
self._shell.settimeout(1000)
self._shell.send(command+'\n')
result = ''
timenow = 0
while True:
time.sleep(sleeptime)
timenow += 1
if timenow > timeout :
if print_c: print result
return result
try:
recv = self._shell.recv(9999)
result = result + recv
str_l = [line for line in result.split()]
if str_l[-1] == 'kd5886>':
if print_c: print result
return result
except:
print 'getting buff...'
#发送命令
def send(self, command, sleeptime = 1, print_c = 1):
if self._ssh is None:
self._ssh = paramiko.SSHClient()
if self._shell is None:
self._shell = self._ssh.invoke_shell()
self._shell.settimeout(1000)
self._shell.send(command+'\n')
result = ''
timenow = 0
time.sleep(sleeptime)
recv = self._shell.recv(9999)
if print_c: print recv
return recv
def close(self):
if self._ssh:
self._ssh.close()
一些保存画图等代码
# 折线图
def line_chart(list_t, label, title = 'title', xlabel = 'x', ylabel = 'y', filename = None):
# 数据准备
plt.figure(figsize=(8, 4))
length = len(list_t)
# 使用Matplotlib画折线图
x = np.arange(length)
plt.plot(x, list_t, label = label, linewidth=1, color='b')
plt.title("{}, max={},min={},average={}".format(title,max(list_t),min(list_t),sum(list_t)/len(list_t)))
# 横坐标描述
plt.xlabel(xlabel)
# 纵坐标描述
plt.ylabel(ylabel)
plt.legend()
fig = plt.gcf()
if filename:
fig.savefig(filename)
else:
fig.savefig("c:/wcb/%s"%title)
plt.show()
#port stati packet 时获得txrx count
def get_txrx_count(port, string):
str_l = [line for line in string.split()]
for i in range(len(str_l)):
if str_l[i] == str(55):
return int(str_l[i+1]),int(str_l[i+2])
if str_l[i] == str(port) and str_l[i+3]==str(port+1):
return int(str_l[i+1]),int(str_l[i+2])
#直接获取某个count
def getCountByStr(recvBuff, target):
recvStrList = [line for line in recvBuff.split()]
for i in range(len(recvStrList)):
if recvStrList[i] == target:
return int(recvStrList[i+1])
return None
#taget 为字符串list
def getCountByNStr(recvBuff, target):
recvStrList = [line for line in recvBuff.split()]
lengthT = len(target)
for i in range(len(recvStrList)):
if recvStrList[i:i+lengthT] == target:
return int(recvStrList[i + lengthT])
return None
def savetocsv(list1, filename = 'd:/wcb/test.csv',list2 = None):
df = pd.DataFrame()
df['list1'] = list1
if list2: df['list2'] = list2
df.to_csv(filename, encoding='gbk')
python控制TCL
import Tkinter
tcl = Tkinter.Tk()
tcl.evalfile('d:/wcb/case3.tcl')
tcl.tk.eval('package require SpirentTestCenter')
tcl.tk.eval('source d:/wcb/case3.tcl')
tcl.eval("""
array set colorcount {
red 1
green 5
blue 4
white 9
}
""")
print tcl.eval('array names colorcount')
#Tcl的list与Python的Tuple可相互使用
tcl.setvar('list_a',(1,2))
tcl.eval("""
puts [lindex $list_a 1]
set v [list a b [list x y ] "c d e " " f {g h}"]
""")
c = tcl.getvar('v') #('a', 'b', ('x', 'y'), 'c d e ', ' f {g h}')
#从TCL中获取list后将list画成折线图,目前用于shaper分析
tcl.eval('''
set var [list 1 4 5 6 7 8 9 2 34 55]
set num 7
set num [expr $num + 1]
#在列表末尾添加str
lappend var $num
''')
var = tcl.getvar('var')
num = tcl.getvar('num')
print num
print var
print type(var)
print type(var[0])
var = list(map(int, var))
print len(var)
print var
print var[0]
print type(var[0])
line_chart(var,'rxrate',title= 'shaper',xlabel= 'num',ylabel='rate/bps',filename="d:/wcb/rate")
savetocsv(var,'d:/wcb/test_1.csv')
后续
后面就开始自己编写一些自动化测试回归例啦,主要是调TCL代码实现测试仪的一些操作,写完之后回归测试例就只需要跑一遍就行啦,大大节省了时间同时可以利用晚上的时间进行回归。