本文整理归纳以往的工作中用到的东西,现汇总成基础测试框架提供分享。
框架采用python3 + selenium3 + PO + yaml + ddt + unittest等技术编写成基础测试框架,能适应日常测试工作需要。
1、使用Page Object模式将页面定位和业务操作分开,分离测试对象(元素对象)和测试脚本(用例脚本),一个页面建一个对象类,提高用例的可维护性;
2、使用yaml管理页面控件元素数据和测试用例数据。例如元素ID等发生变化时,不需要去修改测试代码,只需要在对应的页面元素yaml文件中修改即可;
3、分模块管理,互不影响,随时组装,即拿即用。
GitHub项目地址:https://github.com/yingoja/DemoUI
如下思维导图目录结构介绍:
login.yaml
testinfo: - id: test_login001 title: 登录测试 info: 打开抽屉首页 testcase: - element_info: login-link-a find_type: ID operate_type: click info: 打开登录对话框 - element_info: mobile find_type: ID operate_type: send_keys info: 输入手机号 - element_info: mbpwd find_type: ID operate_type: send_keys info: 输入密码 - element_info: //input[@class='keeplogin'] find_type: XPATH operate_type: click info: 单击取消自动登录单选框 - element_info: //span[text()='登录'] find_type: XPATH operate_type: click info: 单击登录按钮 - element_info: userProNick find_type: ID operate_type: perform info: 鼠标悬停账户菜单 - element_info: //a[@class='logout'] find_type: XPATH operate_type: click info: 选择退出 check: - element_info: //div[@class='box-mobilelogin']/div[1]/span find_type: XPATH info: 检查输入手机号或密码,登录异常提示 - element_info: userProNick find_type: ID info: 成功登录 - element_info: reg-link-a find_type: ID info: 检查退出登录是否成功
例如,我们要新增登录功能测试用例:
首先,只需在testyaml目录下新增一个页面对象yaml文件,参考login.yaml格式编写即可。这些文件是提供给封装页面对象类调用并执行定位识别操作。
login_data.yaml
- id: test_login001.1 detail : 手机号和密码为空登录 screenshot : phone_pawd_empty data: phone: "" password: "" check : - 手机号不能为空 - id: test_login001.2 detail : 手机号为空登录 screenshot : phone_empty data : phone: "" password : aa check : - 手机号不能为空 - id: test_login001.3 detail : 密码为空登录 screenshot : pawd_empty data : phone : 13511112222 password: "" check : - 密码不能为空 - id: test_login001.4 detail : 非法手机号登录 screenshot : phone_error data : phone : abc password: aa check : - 手机号格式不对 - id: test_login001.5 detail : 手机号或密码不匹配 screenshot : pawd_error data : phone : 13511112222 password: aa check : - 账号密码错误 - id: test_login001.6 detail : 手机号和密码正确 screenshot : phone_pawd_success data : phone : 13865439800 password: ******** check : - yingoja login_data.yaml login_data.yaml
其次,在testdata目录下新增一个login_data.yaml文件提供给登录接口传参的测试数据,编写格式参考login_data.yaml文件。
loginPage.py
#!/usr/bin/env python # _*_ coding:utf-8 _*_ __author__ = 'YinJia' import os,sys sys.path.append(os.path.dirname(os.path.dirname(os.path.dirname(__file__)))) from config import setting from selenium.webdriver.support.select import Select from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.common.by import By from public.page_obj.base import Page from time import sleep from public.models.GetYaml import getyaml testData = getyaml(setting.TEST_Element_YAML + '/' + 'login.yaml') class login(Page): """ 用户登录页面 """ url = '/' dig_login_button_loc = (By.ID, testData.get_elementinfo(0)) def dig_login(self): """ 首页登录 :return: """ self.find_element(*self.dig_login_button_loc).click() sleep(1) # 定位器,通过元素属性定位元素对象 # 手机号输入框 login_phone_loc = (By.ID,testData.get_elementinfo()) # 密码输入框 login_password_loc = (By.ID,testData.get_elementinfo()) # 取消自动登录 keeplogin_button_loc = (By.XPATH,testData.get_elementinfo()) # 单击登录 login_user_loc = (By.XPATH,testData.get_elementinfo()) # 退出登录 login_exit_loc = (By.ID, testData.get_elementinfo()) # 选择退出 login_exit_button_loc = (By.XPATH,testData.get_elementinfo()) def login_phone(self,phone): """ 登录手机号 :param username: :return: """ self.find_element(*self.login_phone_loc).send_keys(phone) def login_password(self,password): """ 登录密码 :param password: :return: """ self.find_element(*self.login_password_loc).send_keys(password) def keeplogin(self): """ 取消单选自动登录 :return: """ self.find_element(*self.keeplogin_button_loc).click() def login_button(self): """ 登录按钮 :return: """ self.find_element(*self.login_user_loc).click() def login_exit(self): """ 退出系统 :return: """ above = self.find_element(*self.login_exit_loc) ActionChains(self.driver).move_to_element(above).perform() sleep(2) self.find_element(*self.login_exit_button_loc).click() def user_login(self,phone,password): """ 登录入口 :param username: 用户名 :param password: 密码 :return: """ self.open() self.dig_login() self.login_phone(phone) self.login_password(password) sleep(1) self.keeplogin() sleep(1) self.login_button() sleep(1) phone_pawd_error_hint_loc = (By.XPATH,testData.get_CheckElementinfo(0)) user_login_success_loc = (By.ID,testData.get_CheckElementinfo(1)) exit_login_success_loc = (By.ID,testData.get_CheckElementinfo(2)) # 手机号或密码错误提示 def phone_pawd_error_hint(self): return self.find_element(*self.phone_pawd_error_hint_loc).text # 登录成功用户名 def user_login_success_hint(self): return self.find_element(*self.user_login_success_loc).text # 退出登录 def exit_login_success_hint(self): return self.find_element(*self.exit_login_success_loc).text
然后,在page_obj目录下新增一个loginPage.py文件,是用来封装登录页面对象类,执行登录测试流程操作。
login_sta.py
#!/usr/bin/env python # _*_ coding:utf-8 _*_ __author__ = 'YinJia' import os,sys sys.path.append(os.path.dirname(os.path.dirname(__file__))) import unittest,ddt,yaml from config import setting from public.models import myunit,screenshot from public.page_obj.loginPage import login from public.models.log import Log try: f =open(setting.TEST_DATA_YAML + '/' + 'login_data.yaml',encoding='utf-8') testData = yaml.load(f) except FileNotFoundError as file: log = Log() log.error("文件不存在:{0}".format(file)) @ddt.ddt class Demo_UI(myunit.MyTest): """抽屉新热榜登录测试""" def user_login_verify(self,phone,password): """ 用户登录 :param phone: 手机号 :param password: 密码 :return: """ login(self.driver).user_login(phone,password) def exit_login_check(self): """ 退出登录 :return: """ login(self.driver).login_exit() @ddt.data(*testData) def test_login(self,datayaml): """ 登录测试 :param datayaml: 加载login_data登录测试数据 :return: """ log = Log() log.info("当前执行测试用例ID-> {0} ; 测试点-> {1}".format(datayaml['id'],datayaml['detail'])) # 调用登录方法 self.user_login_verify(datayaml['data']['phone'],datayaml['data']['password']) po = login(self.driver) if datayaml['screenshot'] == 'phone_pawd_success': log.info("检查点-> {0}".format(po.user_login_success_hint())) self.assertEqual(po.user_login_success_hint(), datayaml['check'][0], "成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint())) log.info("成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint())) screenshot.insert_img(self.driver, datayaml['screenshot'] + '.jpg') log.info("-----> 开始执行退出流程操作") self.exit_login_check() po_exit = login(self.driver) log.info("检查点-> 找到{}元素,表示退出成功!".format(po_exit.exit_login_success_hint())) self.assertEqual(po_exit.exit_login_success_hint(), '注册',"退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint())) log.info("退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint())) else: log.info("检查点-> {0}".format(po.phone_pawd_error_hint())) self.assertEqual(po.phone_pawd_error_hint(),datayaml['check'][] , "异常登录,返回实际结果是->: {}".format(po.phone_pawd_error_hint())) log.info("异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint())) screenshot.insert_img(self.driver,datayaml['screenshot'] + '.jpg') if __name__=='__main__': unittest.main()
最后,在testcase目录下创建测试用例文件login_sta.py,采用ddt数据驱动读取yaml测试数据文件
综上所述,编写用例方法只需要按以上四个步骤创建->编写即可。
执行如下主程序,可看输出的实际结果。
#!/usr/bin/env python # _*_ coding:utf-8 _*_ __author__ = 'YinJia' import os,sys sys.path.append(os.path.dirname(__file__)) from config import setting import unittest,time from package.HTMLTestRunner import HTMLTestRunner from public.models.newReport import new_report from public.models.sendmail import send_mail # 测试报告存放文件夹,如不存在,则自动创建一个report目录 if not os.path.exists(setting.TEST_REPORT):os.makedirs(setting.TEST_REPORT + '/' + "screenshot") def add_case(test_path=setting.TEST_DIR): """加载所有的测试用例""" discover = unittest.defaultTestLoader.discover(test_path, pattern='*_sta.py') return discover def run_case(all_case,result_path=setting.TEST_REPORT): """执行所有的测试用例""" now = time.strftime("%Y-%m-%d %H_%M_%S") filename = result_path + '/' + now + 'result.html' fp = open(filename,'wb') runner = HTMLTestRunner(stream=fp,title="抽屉新热榜UI自动化测试报告", description='环境:windows 7 浏览器:chrome', tester='Jason') runner.run(all_case) fp.close() report = new_report(setting.TEST_REPORT) #调用模块生成最新的报告 send_mail(report) #调用发送邮件模块 if __name__ =="__main__": cases = add_case() run_case(cases)
HTML报告日志
HTML报告点击截图,弹出截图
测试报告通过的日志
自动截图存放指定的目录
邮件测试报告
到此这篇关于python selenium自动化测试框架搭建的方法步骤的文章就介绍到这了,更多相关python selenium自动化测试框架搭建内容请搜索小牛知识库以前的文章或继续浏览下面的相关文章希望大家以后多多支持小牛知识库!
作者:YinJia
出处:http://www.cnblogs.com/yinjia/
概况 背景 从开始打算写一个MV*,到一个简单的demo,花了几天的时间,虽然很多代码都是复制/改造过来的,然而It Works(nginx的那句话会让人激动有木有)。现在他叫lettuce,代码 https://github.com/phodal/lettuce,如果有兴趣可以加入我们。 虽然js还不够expert,但是开始了。 步骤 Step 1: 注册npm和bower包 一开始我做的3次c
selenium 是一个web的自动化测试工具,不少学习功能自动化的同学开始首选selenium ,相因为它相比QTP有诸多有点: * 免费,也不用再为破解QTP而大伤脑筋 * 小巧,对于不同的语言它只是一个包而已,而QTP需要下载安装1个多G 的程序。 * 这也是最重要的一点,不管你以前更熟悉C、 java、ruby、python、或都是C# ,你都可以通过selenium完成自动化测试,而QT
基础测试结构 # unittest_simple.py import unittest class SimplisticTest(unittest.TestCase): def test(self): a = 'a' b = 'a' self.assertEqual(a, b) 运行测试 $ python3 -m unittest u
WebUI automation testing framework based on Selenium 介绍: pyse基于selenium(webdriver)进行了简单的二次封装,比selenium提供的方法操作更简洁。 起因: python + selenium 自动化测试写久了发现selenium(webdriver)提供原生的方法并简便,于是,产生了二次封装的想法。想不到太炫酷的名
谈起软件的自动化测试,大部分测试工程师都会情不自禁的浮现起下述三种观点: ● 自动化测试比较高端 刚刚展开测试职业新篇章时,往往从纯手工功能测试入手。做了一段时间的黑盒测试,渐渐对黑盒测试的重复和低效感到乏味,想让工作更有效率一点,想开展自动化测试。但对如何开展软件自动化测试没有一点点思路,没有一丝想法,从而认为自动化测试很高端,那些做自动化的测试工程师都很牛。 ● 自动化测试脚本需
本文向大家介绍python3+selenium自动化测试框架详解,包括了python3+selenium自动化测试框架详解的使用技巧和注意事项,需要的朋友参考一下 背景 为了更好的发展自身的测试技能,应对测试行业以及互联网行业的迭代变化。自学python以及自动化测试。 虽然在2017年已经开始接触了selenium,期间是断断续续执行自动化测试,因为还有其他测试任务,培训任务要执行… 前期建议大