unittest为python语言自带的单元测试框架,python把unittest封装为一个标准模块封装在python开发包中。unittest中常用的类有:unittest.TestCase、unittest.TestSuite、unittest.TextTestRunner、test fixture
一、单元测试框架
单元测试框架并非只能用于单元测试,也可以用于编写和规范自动化测试用例。利用其组织测试用例、断言预期结果以及批量执行测试用例等功能。
单元测试框架主要功能:(源自虫师《Selenium2自动化测试实战 基于python语言》)
1、提供用例组织与执行
提供一定的规范来编写测试用例,方便扩展和维护测试用例;
2、提供丰富的比较方法
测试用例往往需要有一个实际结果和预期结果比较的过程,单元测试框架会提供丰富的断言(assert)方法,用以判断用例的执行是否通过;
3、提供丰富的日记
测试用例执行失败时能抛出失败原因,用例执行完成后提供丰富的执行结果,包括执行时间、失败用例数、成功用例数等
二、TestCase
完整的测试流程,包括测试环境的搭建、测试的执行、测试后环境的还原。
# coding:utf-8 from appium import webdriver import unittest,time class TestCalculator(unittest.TestCase): # 测试前初始化配置
def setUp(self): print('test start') # desired capabilities以字典形式存储,客户端将键值对发给服务器端 desired_caps = {} desired_caps['platformName'] = 'Android' # Android系统 desired_caps['platformVersion'] = '4.4.2' # 指定平台的系统版本 desired_caps['deviceName'] = 'Android Emulator' # 设备名称 desired_caps['appPackage'] = 'com.android.calculator2' # 包名 desired_caps['appActivity'] = '.Calculator' # launcherActivity self.driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_caps) #定义webdriver对象
def tearDown(self): print('test case end') self.driver.quit() #加号运算case def test_add(self): print('add case is running') # 点击运算操作 self.driver.find_element_by_name("5").click() self.driver.find_element_by_name("+").click() self.driver.find_element_by_name("8").click() self.driver.find_element_by_name("=").click() time.sleep(2) self.driver.find_element_by_name("CLR").click() if __name__ == '__main__': unittest.main()
setUp()方法用于测试用例执行前的初始化工作,tearDown()用于测试用例执行后的善后工作。unittest中所有以test开头的函数就是测试用例,我们编写测试用例时需要以test开头。
.py文件有两种使用方式:作为模块调用和直接使用,如果它等于“__main__”表示直接使用。
三、有多个测试用例
1、多个用例
在上面基础上,我们增加一个测试用例test_sub
# coding:utf-8 from appium import webdriver import unittest,time class TestCalculator(unittest.TestCase):
def setUp(self): pass #此处省略
def tearDown(self): print('test case end') self.driver.quit() #加号运算case def test_add(self): print('add case is running') # 点击运算操作 self.driver.find_element_by_name("1").click() self.driver.find_element_by_name("+").click() self.driver.find_element_by_name("8").click() self.driver.find_element_by_name("=").click() time.sleep(2) self.driver.find_element_by_name("CLR").click() #减运算case def test_sub(self): print('sub case is running') self.driver.find_element_by_name("9").click() #无法通过name属性定位到'-',使用xpath定位 self.driver.find_element_by_xpath("//android.widget.Button[contains(@content-desc,'minus')]").click() self.driver.find_element_by_name("1").click() self.driver.find_element_by_name("=").click() time.sleep(2) self.driver.find_element_by_name("CLR").click()
if __name__ == '__main__': unittest.main()
执行后输出结果如下:
test start add case is running test case end test start sub case is running test case end
可知:每执行一个case,都会执行一次setUp()和tearDown()。所以可以单独封装一个测试类,用于存放这两个函数,前提是所有的测试用例所需要的setUp()和tearDown()方法都一样;
PS:测试用例的执行顺序是根据用例名按照ASCII码来排序的,与代码中case的先后顺序无关。数字和字母的顺序为:0~9,A~Z,a~z。如test_a、test_b、test_c.
2、TestSuite和TextTestRunner
TestSuite是一个测试套件,用来组装多个测试用例,可通过addTest方法把testCase添加到TestSuite中,再通过TextTestRunner对象的run()方法执行用例。
如下,现在有两个测试用例test_add和test_sub,把他们依次加入TestSuite中再执行
if __name__ == '__main__': #构造测试集 suite = unittest.TestSuite() suite.addTest(TestCalculator('test_add')) suite.addTest(TestCalculator('test_sub')) #执行测试 runner = unittest.TextTestRunner() runner.run(suite)
使用TestSuite可以做到有选择地执行用例,不需要测试的case可以无需加入。TestSuite按照addTest()的先后顺序执行,需要先执行的case先添加到TestSuite中。