setUpClass和tearDownClass进行测试进行前和结束后进行一次操作,setUpClass用来为测试准备环境,tearDownClass用来清理环境,以备之后的测试(如连接数据库测试前备份一下数据库,测试完成后,将数据库直接还原,这样就可以不用重复造数据)。
#!/usr/bin/env python # -*- coding: utf-8 -*- import unittest class MyTestCase(unittest.TestCase): @classmethod def setUpClass(cls): print '开始执行' @classmethod def tearDownClass(cls): print '结束执行' # 测试用例 def test_case1(self): print '用例1' # 测试用例 def test_case2(self): print '用例2' if __name__ == '__main__': unittest.main()
testsuit测试套件
转自: https://www.cnblogs.com/eastonliu/p/9145255.html
在前面一章中示例了如何编写一个简单的测试,但有两个问题:
我们知道测试用例的执行顺序是根据测试用例名称顺序执行的,在不改变用例名称的情况下,我们怎么来控制用例执行的顺序呢?一个测试文件,我们直接执行该文件即可,但如果有多个测试文件,怎么进行组织,总不能一个个文件执行吧?要解决上面两个问题,我们就要用到测试套件(TestSuite)了
# coding = utf-8 import unittest import warnings from selenium import webdriver from time import sleep # 驱动文件路径 driverfile_path = r'D:\coship\Test_Framework\drivers\IEDriverServer.exe' class CmsLoginTest(unittest.TestCase): def setUp(self): # 这行代码的作用是忽略一些告警打印 warnings.simplefilter("ignore", ResourceWarning) self.driver = webdriver.Ie(executable_path=driverfile_path) self.driver.get("http://172.21.13.83:28080/") def tearDown(self): self.driver.quit() def test_login1(self): '''用户名、密码为空''' self.driver.find_element_by_css_selector("#imageField").click() error_message1 = self.driver.find_element_by_css_selector("[for='loginName']").text error_message2 = self.driver.find_element_by_css_selector("[for='textfield']").text self.assertEqual(error_message1, '用户名不能为空') self.assertEqual(error_message2, '密码不能为空') def test_login3(self): '''用户名、密码正确''' self.driver.find_element_by_css_selector("[name='admin.loginName']").send_keys("autotest") self.driver.find_element_by_css_selector("[name='admin.password']").send_keys("111111") self.driver.find_element_by_css_selector("#imageField").click() sleep(1) self.driver.switch_to.frame("topFrame") username = self.driver.find_element_by_css_selector("#nav_top>ul>li>a").text self.assertEqual(username,"autotest") def test_login2(self): '''用户名正确,密码错误''' self.driver.find_element_by_css_selector("[name='admin.loginName']").send_keys("autotest") self.driver.find_element_by_css_selector("[name='admin.password']").send_keys("123456") self.driver.find_element_by_css_selector("#imageField").click() error_message = self.driver.find_element_by_css_selector(".errorMessage").text self.assertEqual(error_message, '密码错误,请重新输入!') def test_login4(self): '''用户名不存在''' self.driver.find_element_by_css_selector("[name='admin.loginName']").send_keys("test007") self.driver.find_element_by_css_selector("[name='admin.password']").send_keys("123456") self.driver.find_element_by_css_selector("#imageField").click() error_message = self.driver.find_element_by_css_selector(".errorMessage").text self.assertEqual(error_message, '用户不存在!') if __name__ == "__main__": # 构造测试套件 suite = unittest.TestSuite() suite.addTest(CmsLoginTest("test_login1")) suite.addTest(CmsLoginTest("test_login2")) suite.addTest(CmsLoginTest("test_login4")) suite.addTest(CmsLoginTest("test_login3")) # 执行测试 runner = unittest.TextTestRunner(verbosity=2) runner.run(suite)
注:verbosity 参数可以控制输出的错误报告的详细程度,默认是 1;如果设为 0,则不输出每一用例的执行结果;如果设为 2,则输出详细的执行结果
执行结果:
"C:\Program Files\Python36\python.exe" D:/Git/Test_Framework/utils/1.py test_login1 (__main__.CmsLoginTest) 用户名、密码为空 ... ok test_login2 (__main__.CmsLoginTest) 用户名正确,密码错误 ... ok test_login4 (__main__.CmsLoginTest) 用户名不存在 ... ok test_login3 (__main__.CmsLoginTest) 用户名、密码正确 ... ok ---------------------------------------------------------------------- Ran 4 tests in 44.818s OK Process finished with exit code 0从用例的执行结果中我们可以看到,用例的执行顺序是按照添加用例时的顺序来执行的
一个一个地添加测试用例到测试套件中,有点麻烦,其实我们可以把要执行的测试用例用个列表来管理,然后再把这个列表添加到测试套件中,如下代码:
if __name__ == "__main__": # 构造测试套件 suite = unittest.TestSuite() test_cases = [CmsLoginTest("test_login1"),CmsLoginTest("test_login2"),CmsLoginTest("test_login4"), CmsLoginTest("test_login3")] suite.addTests(test_cases) # 执行测试 runner = unittest.TextTestRunner(verbosity=2) runner.run(suite)下面我们就来讲讲多个文件的测试用例组织。
假如我有两个系统的登录需要测试,测试用例分别放在两个文件中(cmslogin.py,smelogin.py),现在我需要把这两个文件中的用例添加到一个测试套件中来执行,为此我们要重新建立一个叫run_all.py的文件
import unittest from cmslogin import CmsLoginTest from smelogin import SmeLoginTest if __name__ == "__main__": # 构造测试套件 suite = unittest.TestSuite() test_cases = [CmsLoginTest("test_login1"),CmsLoginTest("test_login2"),CmsLoginTest("test_login4"), CmsLoginTest("test_login3"),SmeLoginTest("test_login1"),SmeLoginTest("test_login2")] suite.addTests(test_cases) # 执行测试 runner = unittest.TextTestRunner(verbosity=2) runner.run(suite)还可以用addTests + TestLoader方法来添加用例,但是这种方法是无法对case进行排序的
import unittest from cmslogin import CmsLoginTest from smelogin import SmeLoginTest if __name__ == "__main__": # 构造测试套件 suite = unittest.TestSuite() # 第一种方法:传入'模块名.TestCase名' suite.addTests(unittest.TestLoader().loadTestsFromName('cmslogin.CmsLoginTest')) suite.addTests(unittest.TestLoader().loadTestsFromName('smelogin.SmeLoginTest')) # 这里还可以把'模块名.TestCase名'放到一个列表中 suite.addTests(unittest.TestLoader().loadTestsFromNames(['cmslogin.CmsLoginTest','smelogin.SmeLoginTest'])) # 第二种方法:传入TestCase suite.addTests(unittest.TestLoader().loadTestsFromTestCase(CmsLoginTest)) # 执行测试 runner = unittest.TextTestRunner(verbosity=2) runner.run(suite)
unittest参数化
import requests import unittest from parameterized import parameterized as pd # 文件内容为 # http://xxx/api/user/stu_info,stu_name=矿泉水,矿泉水 def read_file(file_name): result = [] with open(file_name,encoding='utf-8') as fr: for line in fr: line = line.strip() if line: result.append(line.split(',')) return result def read_excel(): pass def get_data_from_mysql(sql): pass class InterFaceCase(unittest.TestCase): # @pd.expand([ # ['http://api.nnzhp.cn/api/user/stu_info','stu_name=矿泉水','矿泉水'], # ['http://api.nnzhp.cn/api/user/stu_info','stu_name=小黑','小黑'], # ['http://api.nnzhp.cn/api/user/stu_info','stu_name=','小黑'], # ]) @pd.expand(read_file('login.txt')) def testlogin(self,url,data,check): # username=abc&password=1234 url = url+'?'+data result = requests.get(url).text self.assertIn(check,result) # 检查check是否在result里面
查找测试用例
import unittest import BeautifulReport as bf suite = unittest.defaultTestLoader.discover('all_cases', '*.py') # 查找all_cases文件夹下面所有以 .py 为结尾的文件 report = bf.BeautifulReport(suite) report.report('端午节测试报告','dwj_report.html') # 报告标题,报告文件名 print(report.failure_count) # 已经封装了统计失败次数 print(report.success_count) # 成功次数
参数传递,上下关联关系的接口
import unittest class BaseTest(unittest.TestCase): def register(self,username,password): username,password = 'user1','password1' print('注册成功!') return username,password def login(self,username,password): print('test login',username,password) return 'token1' def add_basket(self,sku_id,count,token): print('加入购物车',sku_id,count) def order(self,token): print('产生订单!!') return 'order_id' def pay(self,order_id): print('支付') class TestOrder(BaseTest): def testOrder(self): '''生成订单的流程''' # 这个注释可以在beautifulreport中显示用例备注 token = self.login('user1','test1') order_id = self.order(token) def testOrder2(self): '''生成订单的异常流程''' token = self.login('user1', 'test1') order_id = self.order(token) class TestPay(BaseTest): def testPay(self): self.register('u','pass') 装饰器可以在不改变原代码情况下添加新功能,其实是将原来代码加入到新功能的函数里面进行实现 import time def log_w(str): with open('a.log', 'a+') as f: f.write(str) def TimeStampToStr(timestamp=None, format='%Y-%m-%d %H:%M:%S'): if timestamp: time_tuple = time.localtime(timestamp) res = time.strftime(format, time_tuple) return res else: return time.strftime(format) def write_record(func): def new_func(): func() print('%s 调用了 %s 函数' % (TimeStampToStr(), func)) log_w('%s 调用了 %s 函数\n' % (TimeStampToStr(), func)) return new_func() # a.log say 2019-06-07 18:33:44 调用了xx函数 # @write_record @write_record def say_hello(): print('Hello World!') @write_record def get_time(): print(time.localtime())
转载于:https://www.cnblogs.com/fatenet/p/11021499.html
