继承能解决什么问题?
unittest每个模块都要用到前提条件以及清理,如果有上百个模块,我们要改域名和浏览器,就会工作量很大特别麻烦,这时我们可以用继承的思想只用改一次
我们可以将前提和清理提出来单独放到一个文件里,具体代码如下
- from selenium import webdriver
- import unittest
-
- class Init(unittest.TestCase):
- @classmethod
- def setUpClass(cls) -> None:
- cls.driver=webdriver.Chrome()
- cls.driver.maximize_window()
- cls.driver.get('http://www.baidu.com')
- cls.driver.implicitly_wait(30)
-
- @classmethod
- def tearDownClass(cls) -> None:
- cls.driver.quit()
然后我们写测试用例的时候可以继承它,具体代码如下
- from selenium import webdriver
- import unittest
-
- from 单元测试框架.test.init import Init
- class BaiduTest(Init):
- def test_baidu_title(self):
- '''百度测试:验证百度首页的title'''
- # assert self.driver.title=='百度一下,你就知道'
- self.assertEqual(self.driver.title,'百度一下,你就知道')
-
- def test_baidu_url(self):
- '''百度测试:验证百度首页的url'''
- assert self.driver.current_url=='https://www.baidu.com/'
-
- def test_baidu_video(self):
- '''百度测试:验证点击视频后跳转到视频的页面'''
- nowhandler=self.driver.current_window_handle
- self.driver.find_element_by_link_text('视频').click()
- allhandlers=self.driver.window_handles
- for handler in allhandlers:
- if handler!=nowhandler:
- self.driver.switch_to.window(handler)
- self.assertTrue(self.driver.current_url,'https://haokan.baidu.com/?sfrom=baidu-top')
- self.driver.close()
- self.driver.switch_to.window(nowhandler)
-
- def test_baidu_map(self):
- '''百度测试:验证点击地图后跳转到地图的页面'''
- nowhandler=self.driver.current_window_handle
- self.driver.find_element_by_link_text('地图').click()
- allhandlers=self.driver.window_handles
- for handler in allhandlers:
- if handler!=nowhandler:
- self.driver.switch_to.window(handler)
- self.assertTrue(self.driver.current_url.startswith('https://map.baidu'))
- self.driver.close()
- self.driver.switch_to.window(nowhandler)
-
- if __name__ == '__main__':
- unittest.main()
在unittest测试框架中,参数化使用的库为:parameterized 安装方式为:pip3 install parameterized
参数化:
我们把相同的测试步骤,应用于不同的测试场景,那么我们就可以使用参数化了
可以解决的问题是可以使用少量的测试代码,来覆盖更多的测试场景
例如:我们测一下sina邮箱的登录模块,代码如下:
- from selenium import webdriver
- import unittest
- import time as t
- class BaiduTest(unittest.TestCase):
- def setUp(self) -> None: #前提
- self.driver=webdriver.Chrome()
- self.driver.get('https://mail.sina.com.cn/')
- self.driver.maximize_window()
- self.driver.implicitly_wait(30)
-
- def tearDown(self) -> None: #清理
- self.driver.quit()
- def test_sina_null(self):
- '''sina邮箱验证:登录账户为空'''
- self.driver.find_element_by_class_name('loginBtn').click()
- divText=self.driver.find_element_by_xpath('/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]')
- self.assertEqual(divText.text,'请输入邮箱名')
-
- def test_sina_email_format(self):
- '''sina邮箱验证:登录邮箱格式不正确'''
- self.driver.find_element_by_id('freename').send_keys('qwert')
- self.driver.find_element_by_class_name('loginBtn').click()
- divText=self.driver.find_element_by_xpath('/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]')
- self.assertEqual(divText.text,'您输入的邮箱名格式不正确')
-
- def test_sina_username_error(self):
- '''sina邮箱验证:登录账户不匹配'''
- self.driver.find_element_by_id('freename').send_keys('asdf@sina.com')
- self.driver.find_element_by_id('freepassword').send_keys('asdfg')
- self.driver.find_element_by_class_name('loginBtn').click()
- t.sleep(3)
- divText=self.driver.find_element_by_xpath('/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]')
- self.assertEqual(divText.text,'登录名或密码错误')
-
- if __name__ == '__main__':
- unittest.main()
由于登录模块主要是⽤户名和密码的input表单的验证以及错误信息的验证,我们可以把用户名、密码、错误信息的验证参数化,具体实现的代码如下:
- from selenium import webdriver
- import unittest
- import time as t
- from parameterized import parameterized,param
- class BaiduTest(unittest.TestCase):
- def setUp(self) -> None: #前提
- self.driver=webdriver.Chrome()
- self.driver.get('https://mail.sina.com.cn/')
- self.driver.maximize_window()
- self.driver.implicitly_wait(30)
-
- def tearDown(self) -> None: #清理
- self.driver.quit()
-
- @parameterized.expand([
- param('','','请输入邮箱名'),
- param('wertasd', 'asdf', '您输入的邮箱名格式不正确'),
- param('wertasd@sina.cn', 'asdf', '登录名或密码错误')
- ])
- def test_sina_login(self,username,password,result):
- self.driver.find_element_by_id('freename').send_keys(username)
- t.sleep(3)
- self.driver.find_element_by_id('freepassword').send_keys(password)
- t.sleep(3)
- self.driver.find_element_by_class_name('loginBtn').click()
- t.sleep(3)
- divText=self.driver.find_element_by_xpath('/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]')
- self.assertEqual(divText.text,result)
assertEqual
assertEqual()是验证两个值相等,值的是数据类型与内容也是相等的,⻅案例代码:
- from selenium import webdriver
- import unittest
- class BaiduTest(unittest.TestCase):
- def setUp(self) -> None:
- self.driver=webdriver.Chrome()
- self.driver.get('http://www.baidu.com')
- self.driver.maximize_window()
- self.driver.implicitly_wait(30)
-
- def tearDown(self) -> None:
- self.driver.quit()
-
- def test_baidu_title(self):
- '''百度测试:验证百度首页的title'''
- # assert self.driver.title=='百度一下,你就知道'
- self.assertEqual(self.driver.title,'百度一下,你就知道')
assertTrue
返回的是bool类型,也就是对被测试的对象进⾏验证,如果返回的是boolean类型并且是true,那么结果验证通过,那么⽅法assertFlase()验证的是被测试对象返回的内容是false,⻅案例代码:
- from selenium import webdriver
- import unittest
- import time as t
- class BaiduTest(unittest.TestCase):
- def setUp(self) -> None: #前提
- self.driver=webdriver.Chrome()
- self.driver.get('https://mail.sina.com.cn/')
- self.driver.maximize_window()
- self.driver.implicitly_wait(30)
-
- def tearDown(self) -> None: #清理
- self.driver.quit()
-
- def test_sina_isLogin(self):
- '''sina邮箱验证,判断自动登录是否勾选'''
- isLogin=self.driver.find_element_by_id('store1')
- self.assertTrue(isLogin.is_selected())
assertIn
assertIn()值的是⼀个值是否包含在另外⼀个值⾥⾯,在这⾥特别的强调⼀下,在assertIn()的⽅法⾥⾯,有两个参数,那么值的包含其实就是第⼆个实际参数包含第⼀个实际参数。与之相反的⽅法是assergNotIn(),⻅案例代码:
- import unittest
- from selenium import webdriver
- class UiTest(unittest.TestCase):
- def setUp(self) -> None:
- self.driver=webdriver.Chrome()
- self.driver.maximize_window()
- self.driver.get('http://www.baidu.com')
- self.driver.implicitly_wait(30)
- def tearDown(self) -> None:
- self.driver.quit()
- def test_baidu_title_001(self):
- self.assertIn('百度',self.driver.title)
- def test_baidu_title_002(self):
- self.assertIn('百度⼀下,你就知道',self.driver.title)
- if __name__ == '__main__':
- unittest.main()
在unittest的框架中,⽣成测试报告需要使⽤到HTMLTestRunner
- import unittest
- import os
- from 单元测试框架.HTMLTestRunner import HTMLTestRunner #从HTMLTestRunner模块调用HTMLTestRunner类
-
- def getTests():
- '''加载所有的测试模块'''
- suite=unittest.TestLoader().discover(
- #找到被执行模块的路径
- start_dir=os.path.dirname(__file__),
- #加载路径下所有以test_开头的测试模块的文件
- pattern='test_*.py' #正则表达式
- )
- return suite
-
- def runSuite():
- unittest.TextTestRunner().run(getTests())
-
- def base_dir():
- return os.path.dirname(os.path.dirname(__file__))#获取当前目录的上级目录
-
- def run():
- fp=open(os.path.join(base_dir(),'report','report.html'),'wb')#拼接report.html的路径 wb 二进制的方式写入
- runner=HTMLTestRunner(
- stream=fp, #流 执行一个写入一个
- title='UI自动化测试报告',
- description=''
- )
- runner.run(getTests())
-
- if __name__ == '__main__':
- run()
怎么解决每次生成的测试报告不覆盖之前的测试报告,同时被保留:
解决方法:引入时间库获取时间戳
代码如下:
- import time
- import unittest
- import os
- from HTMLTestRunner import HTMLTestRunner
-
- def getTests():
- '''加载所有的测试模块'''
- suite=unittest.TestLoader().discover(
- #找到被执行模块的路径
- start_dir=os.path.dirname(__file__),
- #加载路径下所有以test_开头的测试模块的文件
- pattern='test_*.py' #正则表达式
- )
- return suite
-
- def getNowTime():
- return time.strftime('%y-%m-%d %H-%M-%S',time.localtime(time.time()))
-
- def base_dir():
- return os.path.dirname(os.path.dirname(__file__))
-
- def run():
- fp=open(os.path.join(base_dir(),'report',getNowTime()+'report.html'),'wb')
- runner=HTMLTestRunner(
- stream=fp,
- title='UI自动化测试报告',
- description=''
- )
- runner.run(getTests())
-
- if __name__ == '__main__':
- run()
这样每次执行的都会生成一个测试报告:
最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:
这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!