python page object怎么实现页面切换
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了python page object怎么实现页面切换相关的知识,希望对你有一定的参考价值。
页面切换也是对对象操作。所以页面切换可以封装在页面的方法中。 参考技术A Page Object模式是Selenium中的一种测试设计模式,主要是将每一个页面设计为一个Class,其中包含页面中需要测试的元素(按钮,输入框,标题 等),这样在Selenium测试页面中可以通过调用页面类来获取页面元素,这样巧妙的避免了当页面元素id或者位置变化时,需要改测试页面代码的情况。 当页面元素id变化时,只需要更改测试页Class中页面的属性即可。Page Object模式是一种自动化测试设计模式,将页面定位和业务操作分开,分离测试对象(元素对象)和测试脚本(用例脚本),提高用例的可维护性。
unittest是一种单元测试框架,用于设计各式各样的测试用例,可调用PageObject设计的页面类(对象),设计出更加可维护的用例。它提供用例组织与执行,提供丰富的比较(断言)方法,提供丰富的日志,统一适用于web自动化用例的开发与执行。
使用PO模式设计思路如下:
1.定义页面基础类,封装所有页面公用的方法。
命名为test_8_3_2_BasePage.py
# 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
# 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框架编写测试用例
# 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 =""
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()
结语:
这样分层的好处是,不同层关心不同的问题。页面对象层只关心元素定位问题,测试用例只关心测试的数据。
python+ selenium&APPium page Object 设计模式
题记: 之前公司项目比较稳定, 在进行了系统测试,想用自动化测试进行冒烟测试,或者对主要功能进行测试, 因此用到了PO模式 因此做个记录:
Page Object
Page Object模式是使用Selenium的广大同行最为公认的一种设计模式。在设计测试时,把元素和方法按照页面抽象出来,分离成一定的对象,然后再进行组织。
Page Object模式,创建一个对象来对应页面的一个应用。故我们可以为每个页面定义一个类,并为每个页面的属性和操作构建模型。体现在对界面交互细节的封装,测试在更上层使用页面对象,在底层的属性或者操作的更改不会中断测试。减少代码重复,提高测试代码的可读性和可维护性。
我理解就是:把重复用的代码封装成个方法 或者类,在维护脚本的时候只需要改一个地方,有点像解耦(请大神指教是这个意思吗?)
下面以登录为案例:使用的是unitest框架
image:放置每次执行用例的截图,方便调试
report:放置每次执行的测试报告
test_case:
-drvier:封装了driver类
-model:封装了从文件读取数据等一下工具类
-page_object:存放一些页面的元素和定位文件,base.py 把selenium的定位重新封装;
case1
case2 : 真正的测试用例
run_all_test 执行用例
主要思路:在driver.py 封装了driver的方法, myunit.py 调用driver方法,在case类中继承myunit 类 就等于每次用例都有生产driver
driver.py:
def browser(): desired_caps = { ‘platformName‘:‘Android‘, ‘deviceName‘: ‘1f9XXXX08b‘, ‘platformVersion‘:"5.1", ‘appPackage‘:‘com.XXXXXss‘, ‘appActivity‘:‘com.meXXXXXXXXs.activity.LoginActivity‘, ‘unicodeKeyboard‘: True, ‘resetKeyBoard‘: True, "automationName":"uiautomator2" } driver = webdriver.Remote(‘http://127.0.0.1:4723/wd/hub‘,desired_caps) time.sleep(10) # d = driver.current_activity # print(d) return driver
myunit.py
import driver class MyTest(unittest.TestCase): def setUp(self): self.driver = browser() #调用driver self.driver.implicitly_wait(10) def tearDown(self): self.driver.quit()
login_case.py
class LoginTest(myunit.MyTest): def test_login_user_null(self): ‘‘‘用户名为空‘‘‘ #配置IP地址 ipo = IPConfigPage(self.driver) lpo = LoginPage(self.driver) lpo.click_ip_btn() ipo.ipconfig_action("127.0.0.1") #配置登录信息 lpo.login_action("","") result = driverwait.find_toast(self.driver, u‘账号为空‘) print(u"用户名和密码为空校验结果%s"%result) function.insert_image(self.driver, "login_user_null.jpg")
最后总结:个人ui自动化测试局限性比较大,相对于接口自动化,俗话说越后介入成本越高,ui自动化属于中后期,需要在项目稳定情况,小公司很难用的上,如果遇到项目迭代快,需求变化大,维护成本过高。而接口自动化能更早介入测试较早介入测试,维护成本低,而且能部分解决安全问题,人力资源成本高相对
以上是关于python page object怎么实现页面切换的主要内容,如果未能解决你的问题,请参考以下文章
python3+selenium框架设计01-Page Object
python+playwright 学习-9.页面对象模型Page Object Models
selenium Object Page 设计模式理解及实现!