用python做自动化测试--Python实现远程性能监控(1)

来源:互联网 发布:淘宝客服绩效管理流程 编辑:程序博客网 时间:2024/06/01 08:14

在性能测试中,监控被测试服务器的性能指标是个重要的工作,包括CPU/Memory/IO/Network,但大多数人估计都是直接在被测试服务器的运行监控程序。我们开始也是这样做的。但这样做带来一个问题是,测试人员需要在每台被测试服务器上部署监控程序,增加了部署的工作量,而且经常因为python版本的问题,有些模块不兼容,或者第三方模块需要再次安装。

       改进性能测试监控工具:

1. 能远程监控被测试服务器,这样测试人员就不需要在每个被测试机器安装监控工具了。

         2. 被测试服务器上不需要安装agent,监控结果能取到本地。

        3. 本地服务器上的python模块和兼容性问题,可以通过Python  virtualenv解决,每个测试人员维护自己的一套Python环境。


Google了下,找到了pymeter(thttp://pymeter.sourceforge.net/), 看了下源代码,很多工作还没完成,但这个思路和我的是一样的。而且我在其他项目中已经实现了远程发送命令的模块。 所以不如直接在自己的项目上扩展。


远程发送命令的模块开始是基于Pexpect(http://www.noah.org/wiki/Pexpect)实现的, Pexpect很强大,它是一个用来启动子程序,并使用正则表达式对程序输出做出特定响应,以此实现与其自动交互的 Python 模块。用他来可以很容易实现telnet,ftp,ssh的操作。 但Pexpect无windows下的版本,这是我抛弃它的原因,无法达到测试工具兼容所有系统的要求。 所以就用telent模块替换了Pexpect,实现了远程发送命令和获取结果。

#file name: telnetoperate.py

[python] view plaincopy在CODE上查看代码片派生到我的代码片
  1. #!/usr/bin/env python  
  2. #coding=utf-8  
  3.   
  4. import time,sys,logging,traceback,telnetlib,socket  
  5.   
  6.   
  7. class TelnetAction:  
  8.     def __init__(self,host,prompt,account,accountPasswd,RootPasswd=""):  
  9.         self.log=logging.getLogger()  
  10.         self.host=host  
  11.         self.account=account  
  12.         self.accountPasswd=accountPasswd  
  13.         self.RootPasswd=RootPasswd  
  14.         self.possible_prompt = ["#","$"]  
  15.         self.prompt=prompt  
  16.         self.default_time_out=20  
  17.         self.child =None  
  18.         self.login()  
  19.       
  20.     def expand_expect(self,expect_list):  
  21.         try:  
  22.             result=self.child.expect(expect_list,self.default_time_out)  
  23.         except EOFError:  
  24.             self.log.error("No text was read, please check reason")  
  25.         if result[0]==-1:  
  26.             self.log.error("Expect result"+str(expect_list)+" don't exist")  
  27.         else:  
  28.             pass  
  29.         return result  
  30.   
  31.     def login(self):  
  32.         """Connect to a remote host and login. 
  33.              
  34.         """  
  35.         try:  
  36.             self.child = telnetlib.Telnet(self.host)  
  37.             self.expand_expect(['login:'])  
  38.             self.child.write(self.account+ '\n')  
  39.             self.expand_expect(['assword:'])  
  40.             self.child.write(self.accountPasswd + '\n')  
  41.             self.expand_expect(self.possible_prompt)  
  42.             self.log.debug("swith to root account on host "+self.host)  
  43.             if self.RootPasswd!="":  
  44.                 self.child.write('su -'+'\n')  
  45.                 self.expand_expect(['assword:'])  
  46.                 self.child.write(self.RootPasswd+'\n')  
  47.                 self.expand_expect(self.possible_prompt)  
  48.             #self.child.write('bash'+'\n')  
  49.             #self.expand_expect(self.possible_prompt)  
  50.             self.child.read_until(self.prompt)  
  51.             self.log.info("login host "+self.host+" successfully")  
  52.             return True  
  53.         except:  
  54.             print "Login failed,please check ip address and account/passwd"  
  55.             self.log.error("log in host "+self.host+" failed, please check reason")  
  56.             return False  
  57.   
  58.     def send_command(self,command,sleeptime=0.5):  
  59.         """Run a command on the remote host. 
  60.              
  61.         @param command: Unix command 
  62.         @return: Command output 
  63.         @rtype: String 
  64.         """  
  65.         self.log.debug("Starting to execute command: "+command)  
  66.         try:  
  67.             self.child.write(command + '\n')  
  68.             if self.expand_expect(self.possible_prompt)[0]==-1:  
  69.                 self.log.error("Executed command "+command+" is failed, please check it")  
  70.                 return False  
  71.             else:  
  72.                 time.sleep(sleeptime)  
  73.                 self.log.debug("Executed command "+command+" is successful")  
  74.                 return True  
  75.         except socket.error:  
  76.             self.log.error("when executed command "+command+" the connection maybe break, reconnect")  
  77.             traceback.print_exc()  
  78.             for i in range(0,3):  
  79.                 self.log.error("Telnet session is broken from "+self.host+ ", reconnecting....")  
  80.                 if self.login():  
  81.                     break  
  82.             return False  
  83.   
  84.     def get_output(self,time_out=2):  
  85.         reponse=self.child.read_until(self.prompt,time_out)  
  86.         #print "response:",reponse  
  87.         self.log.debug("reponse:"+reponse)  
  88.         return  self.__strip_output(reponse)  
  89.           
  90.     def send_atomic_command(self, command):  
  91.         self.send_command(command)  
  92.         command_output = self.get_output()  
  93.         self.logout()  
  94.         return command_output  
  95.       
  96.     def process_is_running(self,process_name,output_string):      
  97.         self.send_command("ps -ef | grep "+process_name+" | grep -v grep")  
  98.         output_list=[output_string]  
  99.         if self.expand_expect(output_list)[0]==-1:  
  100.             return False  
  101.         else:  
  102.             return True  
  103.       
  104.     def __strip_output(self, response):  
  105.         #Strip everything from the response except the actual command output.  
  106.           
  107.         #split the response into a list of the lines  
  108.           
  109.         lines = response.splitlines()  
  110.         self.log.debug("lines:"+str(lines))  
  111.         if len(lines)>1:  
  112.             #if our command was echoed back, remove it from the output  
  113.             if self.prompt in lines[0]:  
  114.                 lines.pop(0)  
  115.             #remove the last element, which is the prompt being displayed again  
  116.             lines.pop()  
  117.             #append a newline to each line of output  
  118.             lines = [item + '\n' for item in lines]  
  119.             #join the list back into a string and return it  
  120.             return ''.join(lines)  
  121.         else:  
  122.             self.log.info("The response is blank:"+response)  
  123.             return "Null response"  
  124.       
  125.     def logout(self):  
  126.           
  127.         self.child.close()  
0 0