unittest详解(三) 测试套件(TestSuite)
在前面一章中示例了如何编写一个简单的测试,但有两个问题:
- 我们知道测试用例的执行顺序是根据测试用例名称顺序执行的,在不改变用例名称的情况下,我们怎么来控制用例执行的顺序呢?
 - 一个测试文件,我们直接执行该文件即可,但如果有多个测试文件,怎么进行组织,总不能一个个文件执行吧?
 
要解决上面两个问题,我们就要用到测试套件(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("")
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("")
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("")
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详解(三) 测试套件(TestSuite)的更多相关文章
- .NET DLL 保护措施详解(三)最终效果
		
针对.NET DLL 保护措施详解所述思路完成最终的实现,以下为程序包下载地址 下载 注意: 运行环境为.net4.0,需要安装VS2015 C++可发行组件包vc_redist.x86.exe.然后 ...
 - Android 之窗口小部件详解(三)  部分转载
		
原文地址:http://blog.csdn.net/iefreer/article/details/4626274. (一) 应用程序窗口小部件App Widgets 应用程序窗口小部件(Widget ...
 - linux下getsockopt和setsockopt详解及测试
		
linux下getsockopt和setsockopt详解及测试 NAME 名字 getsockopt, setsockopt - get and set options on sockets 获取或 ...
 - unittest详解(四) 批量执行用例(discover)
		
前面我们说了,对于不同文件用例,我们可以通过addTest()把用例加载到一个测试套件(TestSuite)来统一执行,对于少量的文件这样做没问题,但是如果有几十上百个用例文件,这样做就太浪费时间了. ...
 - WebSocket安卓客户端实现详解(三)–服务端主动通知
		
WebSocket安卓客户端实现详解(三)–服务端主动通知 本篇依旧是接着上一篇继续扩展,还没看过之前博客的小伙伴,这里附上前几篇地址 WebSocket安卓客户端实现详解(一)–连接建立与重连 We ...
 - logback -- 配置详解 -- 三 -- <encoder>
		
附: logback.xml实例 logback -- 配置详解 -- 一 -- <configuration>及子节点 logback -- 配置详解 -- 二 -- <appen ...
 - python设计模式之装饰器详解(三)
		
python的装饰器使用是python语言一个非常重要的部分,装饰器是程序设计模式中装饰模式的具体化,python提供了特殊的语法糖可以非常方便的实现装饰模式. 系列文章 python设计模式之单例模 ...
 - Python操作redis字符串(String)详解  (三)
		
# -*- coding: utf-8 -*- import redis #这个redis不能用,请根据自己的需要修改 r =redis.Redis(host=") 1.SET 命令用于设置 ...
 - pika详解(三)SelectConnection及其他Connection
		
pika详解(三)SelectConnection及其他Connection 本文链接:https://blog.csdn.net/comprel/article/details/94661147 ...
 
随机推荐
- java-selenium上传
			
一.sendkeys()上传 HTML源码 <td>sendkeys上传</td> <div id='pf'><input type='file' id='p ...
 - python 变量 (全面不一样的变量)
			
变量 一:什么是变量? 变量即变化的量,核心是"变"与"量"二字,变即变化,量即衡量状态 变:状态是会发生改变的 量:记录现实世界中的状态,让计算机能够像人一样 ...
 - vue-router动态路由设置参数可选
			
在日常工作中,我们需要将匹配到的所有路由,映射到一个组件上. 如下代码想要达到的效果: 不传page和id,则映射到user默认list页面 传page和id,根据page不同,显示不同的页面 问题 ...
 - CentOS 7 关闭Selinux
			
临时关闭: [root@localhost ~]# getenforceEnforcing [root@localhost ~]# setenforce 0[root@localhost ~]# ge ...
 - LoadRunner之使用JSEESIONID访问网站
			
LoadRunner使用笔记 JSESSIONID的含义:https://www.cnblogs.com/caiwenjing/p/8081391.html 1.使用JSESSIONID访问网站 Ac ...
 - spring boot @Value源码解析
			
Spring boot 的@Value只能用于bean中,在bean的实例化时,会给@Value的属性赋值:如下面的例子: @SpringBootApplication @Slf4j public c ...
 - leetcode 1267. Count Servers that Communicate
			
You are given a map of a server center, represented as a m * n integer matrix grid, where 1 means th ...
 - MySQL主从延时这么长,要怎么优化?
			
MySQL主从复制,读写分离是互联网常见的数据库架构,该架构最令人诟病的地方就是,在数据量较大并发量较大的场景下,主从延时会比较严重. 为什么主从延时这么大? 答:MySQL使用单线程重放RelayL ...
 - reduce方法的封装使用
			
reduce()方法 语法: arr.reduce( function(previousValue, item, index, arr) { }, initialValue) previousValu ...
 - document.body.scrollTop无效的解决方法
			
1.document.body.scrollTop = 0 有时候不生效,两种解决方案,试试看. 1-1.设置:document.documentElement.scrollTop = 0;1-2.设 ...