Python+Selenium使用Page Object实现页面自动化测试

 Page Object模式是Selenium中的一种测试设计模式,主要是将每一个页面设计为一个Class,其中包含页面中需要测试的元素(按钮,输入框,标题 等),这样在Selenium测试页面中可以通过调用页面类来获取页面元素,这样巧妙的避免了当页面元素id或者位置变化时,需要改测试页面代码的情况。 当页面元素id变化时,只需要更改测试页Class中页面的属性即可。

    Page Object模式是一种自动化测试设计模式,将页面定位和业务操作分开,分离测试对象(元素对象)和测试脚本(用例脚本),提高用例的可维护性。

    unittest是一种单元测试框架,用于设计各式各样的测试用例,可调用PageObject设计的页面类(对象),设计出更加可维护的用例。它提供用例组织与执行,提供丰富的比较(断言)方法,提供丰富的日志,统一适用于web自动化用例的开发与执行。

使用PO模式设计思路如下:

1.定义页面基础类,封装所有页面公用的方法。

命名为test_8_3_2_BasePage.py

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

# coding=utf-8

'''

Created on 2016-8-13

@author: Jennifer

Project:基础类BasePage,封装所有页面都公用的方法,

定义open函数,重定义find_element,switch_frame,send_keys等函数。

在初始化方法中定义驱动driver,基本url,title

WebDriverWait提供了显式等待方式。

'''

from selenium.webdriver.support.wait import WebDriverWait

from selenium.webdriver.support import expected_conditions as EC

class BasePage(object):

  """

  BasePage封装所有页面都公用的方法,例如driver, url ,FindElement等

  """

  #初始化driver、url、pagetitle等

  #实例化BasePage类时,最先执行的就是__init__方法,该方法的入参,其实就是BasePage类的入参。

  #__init__方法不能有返回值,只能返回None

  #self只实例本身,相较于类Page而言。

  def __init__(self, selenium_driver, base_url, pagetitle):

    self.driver = selenium_driver

    self.base_url = base_url

    self.pagetitle = pagetitle

      

  #通过title断言进入的页面是否正确。

  #使用title获取当前窗口title,检查输入的title是否在当前title中,返回比较结果(True 或 False)

  def on_page(self, pagetitle):

    return pagetitle in self.driver.title

   

  #打开页面,并校验页面链接是否加载正确

  #以单下划线_开头的方法,在使用import *时,该方法不会被导入,保证该方法为类私有的。

  def _open(self, url, pagetitle):

    #使用get打开访问链接地址

    self.driver.get(url)

    self.driver.maximize_window()

    #使用assert进行校验,打开的窗口title是否与配置的title一致。调用on_page()方法

    assert self.on_page(pagetitle), u"打开开页面失败 %s"%url

   

  #定义open方法,调用_open()进行打开链接

  def open(self):

    self._open(self.base_url, self.pagetitle)

   

  #重写元素定位方法

  def find_element(self,*loc):

#    return self.driver.find_element(*loc)

    try:

      #确保元素是可见的。

      #注意:以下入参为元组的元素,需要加*。Python存在这种特性,就是将入参放在元组里。

#      WebDriverWait(self.driver,10).until(lambda driver: driver.find_element(*loc).is_displayed())

      #注意:以下入参本身是元组,不需要加*

      WebDriverWait(self.driver,10).until(EC.visibility_of_element_located(loc))

      return self.driver.find_element(*loc)

    except:

      print u"%s 页面中未能找到 %s 元素"%(self, loc)

   

  #重写switch_frame方法

  def switch_frame(self, loc):

    return self.driver.switch_to_frame(loc)

   

  #定义script方法,用于执行js脚本,范围执行结果

  def script(self, src):

    self.driver.execute_script(src)

   

  #重写定义send_keys方法

  def send_keys(self, loc, vaule, clear_first=True, click_first=True):

    try:

      loc = getattr(self,"_%s"% loc) #getattr相当于实现self.loc

      if click_first:

        self.find_element(*loc).click()

      if clear_first:

        self.find_element(*loc).clear()

        self.find_element(*loc).send_keys(vaule)

    except AttributeError:

      print u"%s 页面中未能找到 %s 元素"%(self, loc)

2.定义登录页面的基本操作方法。

所有页面元素定位都在此层定义,UI一旦有更改,只需在修改这一层页面对象属性即可。

命名为test_8_3_2_LoginPage.py

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

# coding=utf-8

'''

Created on 2016-8-13

@author: Jennifer

Project:页面基本操作方法:如open,input_username,input_password,click_submit

'''

from selenium.webdriver.common.by import By

from test_8_3_2_BasePage import BasePage

#继承BasePage类

class LoginPage(BasePage):

  #定位器,通过元素属性定位元素对象

  username_loc =(By.NAME,'email')

  password_loc =(By.NAME,'password')

  submit_loc =(By.ID,'dologin')

  span_loc =(By.CSS_SELECTOR,"div.error-tt>p")

  dynpw_loc =(By.ID,"lbDynPw")

  userid_loc =(By.ID,"spnUid")

   

  #操作

  #通过继承覆盖(Overriding)方法:如果子类和父类的方法名相同,优先用子类自己的方法。

  #打开网页

  def open(self):

  #调用page中的_open打开连接

    self._open(self.base_url, self.pagetitle)

   

  #输入用户名:调用send_keys对象,输入用户名

  def input_username(self, username):

#    self.find_element(*self.username_loc).clear()

    self.find_element(*self.username_loc).send_keys(username)

   

  #输入密码:调用send_keys对象,输入密码

  def input_password(self, password):

#    self.find_element(*self.password_loc).clear()

    self.find_element(*self.password_loc).send_keys(password)

     

  #点击登录:调用send_keys对象,点击登录

  def click_submit(self):

    self.find_element(*self.submit_loc).click()

   

  #用户名或密码不合理是Tip框内容展示

  def show_span(self):

    return self.find_element(*self.span_loc).text

   

  #切换登录模式为动态密码登录(IE下有效)

  def swich_DynPw(self):

    self.find_element(*self.dynpw_loc).click()

   

  #登录成功页面中的用户ID查找

  def show_userid(self):

    return self.find_element(*self.userid_loc).text

3.使用unittest框架编写测试用例 

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

# coding=utf-8

'''

Created on 2016-8-13

@author: Jennifer

Project:使用unittest框架编写测试用例。

'''

import unittest

from test_8_3_2_LoginPage import LoginPage

from selenium import webdriver

class Caselogin126mail(unittest.TestCase):

  """

     登录126邮箱的case

  """

  def setUp(self):

    self.driver = webdriver.Firefox()

    self.driver.implicitly_wait(30)

    self.url ="http://mail.126.com"

    self.username ="XXX"

    self.password ="XXX"

   

  #用例执行体

  def test_login_mail(self):

    #声明LoginPage类对象

    login_page = LoginPage(self.driver, self.url, u"网易")

    #调用打开页面组件

    login_page.open()

    #切换到登录框Frame

    login_page.switch_frame('x-URS-iframe')

    #调用用户名输入组件

    login_page.input_username(self.username) 

    #调用密码输入组件

    login_page.input_password(self.password)   

    #调用点击登录按钮组件

    login_page.click_submit()

  def tearDown(self):

    self.driver.quit()

     

if __name__ == "__main__":

  unittest.main()

结语:

这样分层的好处是,不同层关心不同的问题。页面对象层只关心元素定位问题,测试用例只关心测试的数据。

​现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
分享他们的经验,还会分享很多直播讲座和技术沙龙
可以免费学习!划重点!开源的!!!
qq群号:485187702【暗号:csdn11】

最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走! 希望能帮助到你!【100%无套路免费领取】

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/718852.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

记一次:android学习笔记一(学习目录-不要看无内容)

学习目录如下 B站学习的名称--Android开发从入门到精通(项目案例版) 网址:https://www.bilibili.com/video/BV1jW411375J/ 第0章:安装 android stoid 参考地址https://blog.csdn.net/adminstate/article/details/130542368 第一章:第一个安卓应用 第二章:用户界面设…

idea插件开发的时候找不到com.intellij.psi.PsiClass

最近在使用idea上传接口带yapi(可视化管理平台)时遇到com.intellij.psi.PsiClass,在网上看了找到几种解决方案,这里总结记录一下: 方法一:在 build.gradle 中的 intellij plugins属性添加 ‘java’ intellij {version 2020.X.Xpl…

直接修改zynq petalinux编译出来的rootfs.cpio.gz文件内容

xilinx zynq petalinux 默认编译打包出的SPI flash烧写启动文件是BOOT.BIN,然而每次需要修改rootfs内的文件时都要重新build rootfs 然后再 package一次才能生成新的BOOT.bin文件,地球人都知道petalinux编译一次是很耗时间的,那么有没有什么简…

OpenCV 4基础篇| OpenCV图像的拆分和合并

目录 1. 通道拆分1.1 cv2.split1.1.1 语法结构1.1.2 注意事项1.1.3 代码示例 1.2 NumPy切片1.2.1 代码示例 2. 通道合并2.1 cv2.merge2.1.1 语法结构2.1.2 注意事项2.1.3 代码示例 1. 通道拆分 1.1 cv2.split 1.1.1 语法结构 b,g,r cv2.split(img[, mv]) #图像拆分为 BGR 通…

【开发工具】GIF 录屏工具推荐 ( GIF123 - 推荐使用 | GifCam | LICEcap )

文章目录 一、GIF 录屏工具推荐1、GIF123 ( 推荐使用 )2、GifCam3、LICEcap 本博客中介绍的 3 款 GIF 录屏工具下载地址 : https://download.csdn.net/download/han1202012/88905642 也可以到对应的官网独立下载 : GIF123 : https://gif123.aardio.com/ ;GifCam : https://bl…

FAST-LIO系列-阅读笔记

近期,阅读了FAST-LIO、FAST-LIO2以及Faster_LIO论文,这三篇论文都属于滤波器的SLAM算法,下面记录一下三个工作的主要贡献和不同。 FAST-LIO 1.提出了一种计算效率高、鲁棒性强的激光雷达-惯性里程测量框架。使用紧密耦合的迭代扩展卡尔曼滤…

报错:/bin/sh: warning: setlocale: LC_ALL: cannot change locale (zh_CN.UTF-8)

解释:这是shell 警告你无法将当前的区域设置(locale)更改为 zh_CN.UTF-8,这个警告可能不会影响 fc-cache 命令的实际运行,但它确实表明系统在某些方面可能无法正确地处理与 zh_CN.UTF-8 相关的内容。 1.检查当前的区域…

2024年口腔护理市场行业未来前景预测:正畸护理用品市场行业分析报告

口腔护理是维护口腔健康的重要步骤,近年来,随着大众口腔健康意识的不断增强,人们对于口腔护理的消费意愿也不断增加,由此,口腔护理市场的市场规模也比较大。 根据鲸参谋电商数据分析平台的相关数据显示,20…

OSCP靶场--Walla

OSCP靶场–Walla 考点(1.hydra http基本认证爆破: 2.sudo -l:python导入外部模块提权 3.Linux内核提权:cve-2021-4034) 1.nmap扫描 ## ┌──(root㉿kali)-[~/Desktop] └─# nmap -sV -sC -p- 192.168.181.97 --min-rate 2000 Starting N…

Linux网络编程:Socket套接字

一、socket地址API 1、主机字节序和网络字节序 小端字节序&#xff08;主机字节序&#xff09;是指一个整数的高位字节存储在内存的高地址处 大端字节序&#xff08;网络字节序&#xff09;是指一个整数的高位字节存储在内存的低地址处 判断机器字节序 #include <stdio.…

RT-DETR算法优化改进: 特征融合涨点篇 | 广义高效层聚合网络(GELAN) | YOLOv9

💡💡💡本文独家改进:即结合用梯度路径规划(CSPNet)和(ELAN)设计了一种广义的高效层聚合网络(GELAN),高效结合RT-DETR,实现涨点。 💡💡💡在多个私有数据集和公开数据集VisDrone2019、PASCAL VOC实现涨点 RT-DETR魔术师专栏介绍: https://blog.csdn.net/…

使用postman测试若依登录接口API-2

请求方式 由于登录控制器可知&#xff1a;该请求方式为Post请求 请求地址 在请求路径栏输入请求地址&#xff0c;如下图所示&#xff1a; 参数体 在Body键入所需参数&#xff0c;类型选择raw,数据格式选择"JSON"&#xff1a;如下图所示&#xff1a; 认证成功与失败…

解释存储过程和函数的区别,以及它们在MySQL中的用途。如何创建和使用存储过程和函数?

解释存储过程和函数的区别&#xff0c;以及它们在MySQL中的用途。 存储过程和函数在MySQL中的区别及用途 区别&#xff1a; 返回值&#xff1a; 函数&#xff1a;必须有一个返回值&#xff0c;这可以是一个标量值或一个表。如果没有明确的RETURN语句&#xff0c;函数将返回N…

香杆箐骑行记,春回大地

2024年3月2日春回大地之际我们校长骑行群再次踏上征程前往香杆箐。这次骑行不仅是一次对身体的锻炼更是一次心灵的洗礼。 清晨的阳光洒满大地我们从郊野公园后门出发踏上了前往香杆箐的道路。沿途的风景如画绿树成荫鲜花盛开让人心旷神怡。我们沿着山路蜿蜒前行感受着大自然的韵…

正则表达式-分组

1、oracle-正则表达式&#xff1a;将09/29/2008 用正则表达式转换成2008-09-29 select regexp_replace(09/29/2008, ^([0-9]{2})/([0-9]{2})/([0-9]{4})$, \3-\1-\2) replace from dual; 解析&#xff1a;regexp_replace-替换&#xff0c; 第一个参数&#xff1a;需要进行处…

Golang Copy()方法学习

前言 主要是涉及到深浅拷贝相关的&#xff0c;但是在看的一个资料过程中发现他有错…并且一系列&#xff0c;复制粘贴他的&#xff0c;也都错了。 错误文章指路 很显然&#xff0c;Copy是深拷贝啊&#xff01;&#xff01;&#xff01; Copy功能 copy的代码很少&#xff0c…

chatgp4 教我学搭建网站1-课程目录

Prerequisite 让我们为学习如何建立网站规划一个先修课程。我们将从0.1开始&#xff0c;不直接进入网站建设本身&#xff1a; 0.1 网络技术基础&#xff1a;了解互联网如何工作&#xff0c;包括域名系统&#xff08;DNS&#xff09;、HTTP/HTTPS协议等。 0.2 HTML基础&#x…

OAuth2:保障现代应用程序安全的关键协议

OAuth2&#xff1a;保障现代应用程序安全的关键协议 摘要&#xff1a;本文将为您详细介绍OAuth2&#xff0c;这一保障现代应用程序安全的关键协议。我们将探讨OAuth2的概念、工作流程、应用场景以及优势&#xff0c;帮助您更好地理解并应用这一强大的安全认证机制。 一、引言…

玩转地下管网三维建模:MagicPipe3D系统

地下管网是保障城市运行的基础设施和“生命线”。随着实景三维中国建设的推进&#xff0c;构建地下管网三维模型与地上融合的数字孪生场景&#xff0c;对于提升智慧城市管理至关重要&#xff01;针对现有三维管线建模数据差异大、建模交互弱、模型效果差、缺乏语义信息等缺陷&a…

Bert Encoder和Transformer Encoder有什么不同

前言&#xff1a;本篇文章主要从代码实现角度研究 Bert Encoder和Transformer Encoder 有什么不同&#xff1f;应该可以帮助你&#xff1a; 深入了解Bert Encoder 的结构实现深入了解Transformer Encoder的结构实现 本篇文章不涉及对注意力机制实现的代码研究。 注&#xff1a;…