【python selenium的web自动化】- PageObject模式解析及案例

https://www.cnblogs.com/miki-peng/category/1942527.html

PO模式

Page Object(简称PO)模式,是Selenium实战中最为流行,并且是自动化测试中最为熟悉和推崇的一种设计模式。在设计自动化测试时,把页面元素和元素的操作方法按照页面抽象出来,分离成一定的对象,然后再进行组织。

做web自动化最头疼的一个问题,莫过于页面变化了,如果没有使用PO设计模式,页面一变化就意味着之前的元素定位甚至元素的操作方法不能用了,需要重新修改。你需要一个一个从测试脚本中把需要修改的元素定位方式、元素的操作方法找出来,然后一一地修改。这样的自动化脚本不但繁琐,维护成本也极高。

而page object模式就可以很好地解决这个问题,优点🔻:

  • 🍍 减少代码冗余

  • 🍍 业务和实现分离

  • 🍍 降低维护成本

那到底什么是Page Object模式,见名知意,就是页面对象,在实际自动化测试中,一般对脚本分为三层:

  • 🍎 对象层: 用于存放页面元素定位

  • 🍎 逻辑层: 用于存放一些封装好的功能用例模块

  • 🍎 业务层: 用于存放我们真正的测试用例的操作部分

除了以上三层,还有一个基础层,基础层主要是针对selenium的一些常用方法,根据实际业务需要进行二次封装,如点击、输入等操作加入一些等待、日志输入、截图等操作,方便以后查看脚本的运行情况及问题排查。

基础层

基础层类名一般命名为BasePage,后续的对象层操作元素时都继承这个基础类,下面以点击、输入为例:

# basepage.pyimport osimport timeimport datetimefrom selenium.webdriver.remote.webdriver import WebDriverfrom selenium.webdriver.support.wait import WebDriverWaitfrom selenium.webdriver.support import expected_conditions as ECfrom common.logging import logfrom common.constant import IMG_DIRclass BasePage:    def __init__(self, driver: WebDriver):        self.driver = driver    def wait_ele_visible(self, loc, img_desc, timeout=20, frequency=0.5):        """等待元素可见"""        try:            WebDriverWait(self.driver, timeout, frequency).until(EC.visibility_of_element_located(loc))            log.info("等待:{} - 元素{}可见成功。".format(img_desc, loc))        except:            log.exception("等待:{} - 元素{}可见失败!".format(img_desc, loc))            self.save_img(img_desc)            raise    def get_element(self, loc, img_desc):        """查找元素"""        try:            ele = self.driver.find_element(*loc)        except:            log.exception("查找:{} - 元素{}失败!".format(img_desc, loc))            self.save_img(img_desc)            raise        else:            log.info("查找:{} - 元素{}成功".format(img_desc, loc))            return ele    def click_element(self, loc, img_desc, timeout=20, frequency=0.5):        """点击元素"""        self.wait_ele_visible(loc, img_desc, timeout, frequency)        ele = self.get_element(loc, img_desc)        try:            ele.click()            log.info("点击:{} - 元素{}成功".format(img_desc, loc))        except:            log.exception("点击:{} - 元素{}失败!".format(img_desc, loc))            self.save_img(img_desc)            raise    def input_text(self, loc, value, img_desc, timeout=20, frequency=0.5):        """在元素中输入文本"""        self.wait_ele_visible(loc, img_desc, timeout, frequency)        ele = self.get_element(loc, img_desc)        try:            ele.send_keys(value)            log.info("输入:在{} - 元素{}输入文本值({})成功".format(img_desc, loc, value))        except:            log.exception("输入:在{} - 元素{}输入文本值({})失败!".format(img_desc, loc, value))            self.save_img(img_desc)            raise    def save_img(self, img_description):        """保存异常截图"""        now = time.strftime("%Y-%m-%d %H-%M-%S ", time.localtime())        img_path = os.path.join(IMG_DIR, now   img_description   '.png')        try:            self.driver.save_screenshot(img_path)        except:            log.exception("异常截图失败!")        else:            log.info("异常截图成功,截图存放在{}".format(img_path))

以点击click_element()为例,这里二次封装时加入了等待操作、日志输入、异常截图,后面点击元素时就直接调用click_element()就可以一步到位,不需要再考虑等待、日志、异常的情况,这里都已经处理好了,虽然在初期写基础页面会比较耗时,但只要基础打好,在后续维护工作中会轻松很多。以上只是一个示例,可以根据自己的实际需要进行优化。

对象层及逻辑层

对象层存放页面元素定位,逻辑层存放元素操作方法(页面功能),元素定位可以根据实际需要,可以单独放在一个模块来维护,也可以存放在excel中进行集中管理;下面演示的是元素定位和元素操作方法都存放到一个模块中,一个页面一个模块,后续页面元素发生变化,只需要修改在这个模块中修改对应的定位表达式或者操作方法即可。

演示以百度首页为例:

# baidu_page.pyfrom selenium.webdriver.common.by import Byfrom common.basepage import BasePageclass LoginPage(BasePage):    login_btn = (By.XPATH, '//div[@id="u1"]//a[@name="tj_login"]')  # 登录按钮    username_login_btn = (By.ID, 'TANGRAM__PSP_11__footerULoginBtn')    # 用户名登录按钮    user_input = (By.ID, 'TANGRAM__PSP_11__userName')  # 用户信息输入框    pwd_input = (By.ID, 'TANGRAM__PSP_11__password')  # 密码输入框    login_submit = (By.ID, 'TANGRAM__PSP_11__submit')   # 登录提交按钮    def login(self, user, pwd):        """        百度用户名登录        :param user: 手机/邮箱/用户名        :param pwd: 密码        :return:        """        self.click_element(self.login_btn, '百度-登录')        self.click_element(self.username_login_btn, '百度登录-用户名登录')        self.input_text(self.user_input, user, '用户名登录-手机/邮箱/用户名')        self.input_text(self.pwd_input, pwd, '用户名登录-密码')        self.click_element(self.login_submit, '用户名登录-登录')

业务层

用于存放真正的测试用例操作,这里不会出现元素定位、页面功能,所有操作都是直接调用逻辑层的.

测试用例 = 测试对象的功能 测试数据,下面以百度登录为例(用于演示,简略写的):

import unittestimport pytestimport ddtfrom selenium import webdriverfrom PageObjects.baidu_login_page import LoginPagefrom testdatas import common_datas as com_dfrom testdatas import login_data as lo_dfrom common.logging import log@ddt.ddtclass TestLogin(unittest.TestCase):    def setUp(self):        log.info("-------用例前置工作:打开浏览器--------")        self.driver = webdriver.Chrome()        self.driver.get(com_d.baidu_url)        self.driver.maximize_window()    def tearDown(self):        self.driver.quit()        log.info("-------用例后置工作:关闭浏览器--------")    @pytest.mark.smoke    def test_login_success(self):        # 用例:登录页的登录功能        # 步骤        LoginPage(self.driver).login(lo_d.success_data['user'], lo_d.success_data['pwd'])        # 断言.....

运行结果:

Testing started at 11:50 ...C:\software\python\python.exe "C:\Program Files\JetBrains\PyCharm Community Edition 2019.1.3\helpers\pycharm\_jb_unittest_runner.py" --path D:/learn/test/testcases/test_baidu_login.pyLaunching unittests with arguments python -m unittest D:/learn/test/testcases/test_baidu_login.py in D:\learn\test\testcasesProcess finished with exit code 02021-03-14 11:50:47,238-【test_baidu_login.py-->line:27】-INFO:-------用例前置工作:打开浏览器--------2021-03-14 11:50:51,327-【basepage.py-->line:38】-INFO:等待:百度-登录 - 元素('xpath', '//div[@id="u1"]//a[@name="tj_login"]')可见成功,耗时0:00:00.056843秒2021-03-14 11:50:51,339-【basepage.py-->line:77】-INFO:查找:百度-登录 - 元素('xpath', '//div[@id="u1"]//a[@name="tj_login"]')成功2021-03-14 11:50:51,414-【basepage.py-->line:86】-INFO:点击:百度-登录 - 元素('xpath', '//div[@id="u1"]//a[@name="tj_login"]')成功2021-03-14 11:50:53,463-【basepage.py-->line:38】-INFO:等待:百度登录-用户名登录 - 元素('id', 'TANGRAM__PSP_11__footerULoginBtn')可见成功,耗时0:00:02.048293秒2021-03-14 11:50:53,474-【basepage.py-->line:77】-INFO:查找:百度登录-用户名登录 - 元素('id', 'TANGRAM__PSP_11__footerULoginBtn')成功2021-03-14 11:50:53,535-【basepage.py-->line:86】-INFO:点击:百度登录-用户名登录 - 元素('id', 'TANGRAM__PSP_11__footerULoginBtn')成功2021-03-14 11:50:53,576-【basepage.py-->line:38】-INFO:等待:用户名登录-手机/邮箱/用户名 - 元素('id', 'TANGRAM__PSP_11__userName')可见成功,耗时0:00:00.040890秒2021-03-14 11:50:53,584-【basepage.py-->line:77】-INFO:查找:用户名登录-手机/邮箱/用户名 - 元素('id', 'TANGRAM__PSP_11__userName')成功2021-03-14 11:50:53,714-【basepage.py-->line:98】-INFO:输入:在用户名登录-手机/邮箱/用户名 - 元素('id', 'TANGRAM__PSP_11__userName')输入文本值(15692004245)成功2021-03-14 11:50:53,759-【basepage.py-->line:38】-INFO:等待:用户名登录-密码 - 元素('id', 'TANGRAM__PSP_11__password')可见成功,耗时0:00:00.043882秒2021-03-14 11:50:53,771-【basepage.py-->line:77】-INFO:查找:用户名登录-密码 - 元素('id', 'TANGRAM__PSP_11__password')成功2021-03-14 11:50:53,925-【basepage.py-->line:98】-INFO:输入:在用户名登录-密码 - 元素('id', 'TANGRAM__PSP_11__password')输入文本值(phang0209)成功2021-03-14 11:50:53,958-【basepage.py-->line:38】-INFO:等待:用户名登录-登录 - 元素('id', 'TANGRAM__PSP_11__submit')可见成功,耗时0:00:00.031914秒2021-03-14 11:50:53,969-【basepage.py-->line:77】-INFO:查找:用户名登录-登录 - 元素('id', 'TANGRAM__PSP_11__submit')成功2021-03-14 11:50:54,051-【basepage.py-->line:86】-INFO:点击:用户名登录-登录 - 元素('id', 'TANGRAM__PSP_11__submit')成功2021-03-14 11:50:56,426-【test_baidu_login.py-->line:35】-INFO:-------用例后置工作:关闭浏览器--------Ran 1 test in 9.191sOK

从输出日志来看,每一步操作都清晰可见,出现问题也能快速定位,这些都可以根据实际需要来优化。

来源:https://www.icode9.com/content-1-893051.html

(0)

相关推荐

  • 自动抢票之 12306 登录篇

    来源:Python 技术「ID: pythonall」 逢年过节 12306 的票总是要靠抢,前几天小编就在抢周一去上海的票,实在是抢不到呀,就撸了一个自动抢票的脚本. 抢票的思路就是使用 selen ...

  • appium+python自动化62-webview元素click失效问题解决

    前言 Appium 在切换到 webview 后,正确定位到元素,但是click () 事件后界面无响应,脚本运行正常不会报错. 主要原因是:混合APP 时监听全用的是tap事件,不是click事件 ...

  • Selenium+python自动化83-chrome手机wap模式

    一.前言 这里是群里(QQ群:226296743)风神出60大洋悬赏的问题,学会这篇就是赚了60大洋了! 我的环境: - chrome 62 - chromedriver 2.33 二.遇到问题 1. ...

  • (Python)自动生成代码(方法一)

    在写某个平台的自动化脚本时,笔者把全部的操作都封装到了两个类中,page.py和commonpage.py: page.py部分代码:class BasePage(object): ''' 页面基础类 ...

  • 自动化搭建环境及基础理论

    在dos命令中输入python,在dos命令中输入pip Pip是Python官方推荐的包管理工具,提供了对Python包的查找.下载.安装.卸载的功能,属于python的一部分. Python3.0 ...

  • Python爬虫模拟登陆哔哩哔哩(bilibili)并突破点选验证码功能

    写在前面   今天带给大家一个突破点选验证码的案例,利用爬虫模拟登陆哔哩哔哩,并且把一些采坑的地方给大家强调一下! 一.需求分析   模拟登陆哔哩哔哩   网站链接: https://passport ...

  • Python爬取网易云音乐辑的图片、专辑名和专辑出版时间

    本文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,如有问题请及时联系我们以作处理. 作者:阿里波特 来源:CSDN Python爬虫.数据分析.网站开发等案例教程视频免费在线观看 h ...

  • appium+python自动化33-解锁九宫格(TouchAction)

    一.TouchAction 1.源码可以在这个路径找到:Lib\site-packages\appium\webdriver\common\touch_action.py ``` class Touc ...

  • 第8章——自动化测试综合实战

    自动化测试综合实战 项目背景 在 http://localhost/news/ 新闻子页面进行登录测试. 功能实现 · 自动运行用例 · 自动生成测试报告 · 自动断言与截图 · 自动将最新测试报告发 ...

  • 《Selenium+Pytest Web自动化实战》前100名立减300

    课程介绍 课程主题:<Selenium+Pytest Web自动化实战> 适合人群: 1.功能测试转型自动化测试 2.web自动化零基础的小白 3.对python 和 selenium 有 ...

  • Cypress web自动化35-cy.exec()执行python命令操作数据库

    前言 cy.exec()可以执行系统命令,获取到stdout内容,当我们要操作数据库,准备测试数据的时候,通常用python连数据库操作会非常方便. 我们可以先把操作数据库的方法封装到一个py文件,这 ...

  • Cypress web自动化36-cy.exec()执行python查询数据库获取结果

    前言 cy.exec() 可以执行系统命令行,那么用 python 写个查询 sql 的时候,返回结果是 json 格式. cypress 的脚本是 javascript 语言写的,没法直接识别pyt ...

  • docker下使用selenium+testng实现web自动化

    Windows下selenium+testng的web自动化环境搭建 做过自动化的人,肯定对selenium web环境的搭建非常熟悉了,特别是selenium在java中的使用. 先搭建好安装好JD ...

  • Python|Helium实现web的自动化

    Helium库介绍自动化,通俗来讲是通过程序的自动运行来替代人为的操作,即在没有人的参与下同样达到人们想要达到的目的.在网上搜索web自动化,最先看到的一定是Selenium,这是一个专门的web自动 ...

  • 聚丙烯的供需格局及销售模式解析

    聚丙烯,是由丙烯聚合制得的一种热塑性树脂,可用于很多领域.这个产品是2000万吨级别的大产品,且较有金融属性.聚丙烯的供需格局是怎样的?它的销售模式又是如何?<摩贝视野>将在下文中为您解析 ...

  • Python 最快Web框架

    点击关注

  • PS图层混合模式教程,点光模式解析

    点光图层混合模式:根据混合色的明暗去替换颜色,如果混合色(光源)大于50%的中性灰,也就是说混合色比中性灰色亮,则混合色会替换掉比混合色暗的基色像素:如果混合色小于50%的中性灰,也就是说混合色比中性 ...

  • python word excel ppt自动化办公教程

    模块导入 import openpyxl 读取Excel文件 打开Excel文件 workbook = openpyxl.load_workbook('test.xlsx') 输出表单名字 # 输出工 ...