Web自动化测试:POM设计模式的实现

关于pom设计模式(project Object model/PageObject),一种底层、逻辑、用例的分层,在项目还没有开发出来时,就可以开始写UI自动化脚本了,在开发完成后,再进行元素定位的适配以及调试;而且也可以多人共同维护开发脚本,更方便大家合作。

这一节主要来介绍一下如何从零开始搭建这几个层级。

一、driver层的封装
这一层主要是对于webdriver方法的封装,这里来举一个栗子,最常用的定位方法,之前讲过统一定位方法的三种传参格式:webdriver的所有定位方法,使用find_element()方法通过BY类、字符串、元组三种方法传递定位类型和数据,这里我使用元组的形式(例如locator = (“id”,“name_box”))来传递参数。

1.1 定位元素方法封装示例

这个定位元素的公共方法中,加了很多东西;如果每次定位的时候写这些异常捕获、打印操作的话,那么程序会非常臃肿,所以需要单独提出来,每次需要定位的时候统一调用这个方法。

有一个入参locator,格式为(“定位类型”,“定位参数值”),返回我们所定位到的元素
加入了元素等待,并判断该元素是否存在
对于关键信息的打印输出,方便定位监控
加入了异常捕获,定位失败后可以继续执行程序
def find_element(self, *locator):
try:
print(“定位元素:%s” % (locator,))
return WebDriverWait(self.driver, 10).until(EC.presence_of_element_located(locator))
except Exception as msg:
print(u"%s 页面中未能找到 %s 元素" % (self, locator))
print(“错误信息%s” % msg)
1.2 封装类的初始化

对于webdriver的封装,我们要先创建一个class,这样方便我们继承调用这些封装的方法。在class中,我这里设计了一个初始化,每次调用封装的driver时,传递三个参数,一个必填项:driver、两个非必填项:page_url、page_title,我这样的想法是每次引用这个封装类时,传递一个driver进来,如果有打开网址页面的需要,则传递网址和网址页面的title,这样也可以做一次校验。

def init(self, driver, page_url=None, page_title=None):
self.page_url = page_url
self.page_title = page_title
self.driver = driver
self.driver.maximize_window()
self.driver.implicitly_wait(30)
二、page层书写
page类在继承我们封装的webdriver后,主要写具体的操作步骤,例如输入登录名、输入登录密码、点击登录按钮等操作。

2.1继承pagedriver并初始化

这里的page层要继承pagedriver的类Action,然后在page层的初始化中,初始化Action。

from common.pagedriver import Action
class Login(Action):

def __init__(self, driver, page_url=None, page_title=None):
    Action.__init__(self, driver, page_url, page_title)

2.2 操作步骤
比如我要写打开页面、输入用户名这两个方法:

其中元素定位放在类变量中,而登录账号我们放在case层来输入。

from common.pagedriver import Action

class Login(Action):
input_name_loc = (“xpath”, “//input[@placeholder=‘邮箱帐号或手机号码’]”)
frame_loc = (0)

def __init__(self, driver, page_url=None, page_title=None):
    Action.__init__(self, driver, page_url, page_title)

def open(self):
    """打开页面"""
    self._open(self.page_url, self.page_title)

def input_name(self, login_name):
    """输入登录名"""
    self.send_keys(self.input_name_loc, login_name)

三、case层调用
终于到了第三层,这里我们要做的就是把page层的方法,像搭积木一样搭起来,并且连成完整的操作。

3.1 使用unittest,并初始化数据

在unittest的框架基础上,主要是在setUp()方法中初始化我们的数据,例如网址、账号、driver的初始化

import unittest
from selenium import webdriver

class Demo(unittest.TestCase):

def setUp(self):
    self.url = "https://mail.163.com/"
    self.title = "网易"
    self.user_name = ""  # 登录账户
    self.user_password = ""  # 登录密码
    self.driver = webdriver.Chrome()

def tearDown(self):
    self.driver.close()

if name == “main”:
unittest.main()
3.2 调用方法,完成用例
首先我们引用page层,然后使用page层的方法搭建case。

from page.login_page import Login

def test_login(self):
    login_page = Login(self.driver, self.url, self.title)
    login_page.open()
    login_page.input_name(self.user_name)

四、实例演示:登录163网易邮箱
通过上述的分层步骤,演示登录163邮箱的操作,登录后通过断言登陆成功页面title,来判断是否登录成功。

运行结果:

打开网址:https://mail.163.com/
网址预期标题: 网易
定位元素:(‘xpath’, “//input[@placeholder=‘邮箱帐号或手机号码’]”)
输入值:
定位元素:(‘xpath’, “//input[@placeholder=‘输入密码’]”)
输入值:
定位元素:(‘xpath’, “//input[@placeholder=‘输入密码’]”)
输入值:
(26封未读) 网易邮箱6.0版
.

Ran 1 test in 202.126s

OK
运行代码:

pagedriver.py

from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.ui import WebDriverWait

class Action(object):
“”"
Action封装所有页面都公用的方法
“”"

# 初始化driver、url、title等
def __init__(self, driver, page_url=None, page_title=None):
    self.page_url = page_url
    self.page_title = page_title
    self.driver = driver
    self.driver.maximize_window()
    self.driver.implicitly_wait(30)

def open(self):
    """
    定义open方法,调用_open()进行打开链接
    """
    self._open(self.page_url, self.page_title)

def on_page(self, page_title):
    """
    使用current_url获取当前窗口Url地址,进行与配置地址作比较,返回比较结果(True False)
    """
    return page_title in self.driver.title

def _open(self, page_url, page_title):
    """
    打开页面,校验页面链接是否加载正确
    """
    # 使用get打开访问链接地址
    if page_url and page_title is not None:
        self.driver.get(page_url)
        print("打开网址:%s" % page_url)
        print("网址预期标题: %s" % page_title)
        # 使用assert进行校验,打开的链接地址是否与配置的地址一致。调用on_page()方法
        assert self.on_page(page_title), u"打开页面%s失败" % page_url

def find_element(self, *locator):
    try:
        print("定位元素:%s" % (locator,))
        return WebDriverWait(self.driver, 20).until(EC.presence_of_element_located(locator))
    except Exception as msg:
        print(u"%s 页面中未能找到 %s 元素" % (self, locator))
        print("错误信息%s" % msg)

def send_keys(self, locator, value, clear_first=True):
    """
    重写定义send_keys方法
    """
    element = self.find_element(*locator)
    if clear_first:
        element.clear()
        element.send_keys(value)
    else:
        element.send_keys(value)
    print("输入值:%s" % value)

def switch_frame(self, frame_loc):
    """
    切换frame,
    :param frame_loc:id、name、element、index
    :return:
    """
    self.driver.switch_to.frame(frame_loc)

login_page.py

from common.pagedriver import Action
from selenium.webdriver.common.keys import Keys

class Login(Action):
input_name_loc = (“xpath”, “//input[@placeholder=‘邮箱帐号或手机号码’]”)
input_password_loc = (“xpath”, “//input[@placeholder=‘输入密码’]”)
enter_login_loc = Keys.ENTER
frame_loc = (0)

def __init__(self, driver, page_url=None, page_title=None):
    Action.__init__(self, driver, page_url, page_title)

def open(self):
    """打开页面"""
    self._open(self.page_url, self.page_title)

def change_frame(self):
    """切换frame"""
    self.switch_frame(self.frame_loc)

def input_name(self, login_name):
    """输入登录名"""
    self.send_keys(self.input_name_loc, login_name)

def input_password(self, login_password):
    """输入密码"""
    self.send_keys(self.input_password_loc, login_password)

def enter_login(self):
    """模拟登陆点击回车"""
    self.send_keys(self.input_password_loc, self.enter_login_loc, False)

def get_login_message(self):
    """获取登录后的信息以断言"""
    return self.driver.title

test_163_login.py

-- coding: utf-8 --

import unittest
from time import sleep
from page.login_page import Login
from selenium import webdriver

class Demo(unittest.TestCase):

def setUp(self):
    self.url = "https://mail.163.com/"
    self.title = "网易"
    self.user_name = ""  # 登录账户
    self.user_password = ""  # 登录密码
    self.driver = webdriver.Chrome()

def test_wangyi_login(self):
    """登录网易邮箱"""
    login_page = Login(self.driver, self.url, self.title)
    login_page.open()
    login_page.change_frame()
    sleep(3)
    login_page.input_name(self.user_name)
    login_page.input_password(self.user_password)
    sleep(2)
    login_page.enter_login()
    sleep(5)
    print(login_page.get_login_message())
    assert "网易邮箱6.0版" in login_page.get_login_message()

def tearDown(self):
    self.driver.close()

if name == “main”:
unittest.main()

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是基于Python的接口自动化POM模式的模板: ```python # 导入相关库和模块 import requests # 创建基础类 class BaseApi(object): # 定义构造函数 def __init__(self): self.session = requests.session() # 创建会话对象 self.base_url = "http://api.example.com" # 接口基础 URL # 定义接口请求方法 def send_api(self, url, method="get", data=None, json=None, **kwargs): # 组装请求 URL url = self.base_url + url # 根据不同请求方法发送请求 if method.lower() == "get": res = self.session.get(url, params=data, **kwargs) elif method.lower() == "post": res = self.session.post(url, data=data, json=json, **kwargs) else: raise ValueError("Invalid request method: {}".format(method)) # 返回响应结果 return res.json() # 创建页面类 class LoginPage(BaseApi): # 定义页面元素定位器 username_input_loc = ("id", "username") password_input_loc = ("id", "password") login_button_loc = ("id", "login-button") # 定义页面动作或操作方法 def input_username(self, username): self.session.cookies.clear() # 清除会话 Cookies self.session.headers.update({"User-Agent": "Mozilla/5.0"}) # 更新请求头部信息 self.session.headers.update({"Referer": "http://www.example.com"}) # 更新请求头部信息 self.session.headers.update({"Accept-Language": "en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7"}) # 更新请求头部信息 self.session.headers.update({"Accept-Encoding": "gzip, deflate"}) # 更新请求头部信息 self.session.headers.update({"Connection": "keep-alive"}) # 更新请求头部信息 self.session.headers.update({"Content-Type": "application/x-www-form-urlencoded"}) # 更新请求头部信息 self.session.headers.update({"Upgrade-Insecure-Requests": "1"}) # 更新请求头部信息 self.session.headers.update({"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8"}) # 更新请求头部信息 self.session.headers.update({"Cache-Control": "max-age=0"}) # 更新请求头部信息 self.session.headers.update({"Origin": "http://www.example.com"}) # 更新请求头部信息 self.session.headers.update({"Host": "www.example.com"}) # 更新请求头部信息 self.session.headers.update({"Pragma": "no-cache"}) # 更新请求头部信息 self.session.headers.update({"Referer": "http://www.example.com/login"}) # 更新请求头部信息 self.session.cookies.update({"username": username}) # 更新请求 Cookies self.session.cookies.update({"password": "password"}) # 更新请求 Cookies def input_password(self, password): self.session.cookies.update({"password": password}) # 更新请求 Cookies def click_login_button(self): self.send_api("/login", method="post", data=self.session.cookies.get_dict()) # 发送登录请求 # 创建测试类 class TestLoginPage(object): # 定义测试用例 def test_login_success(self): login_page = LoginPage() login_page.input_username("test_user") login_page.input_password("test_password") login_page.click_login_button() # 断言登录成功后的响应结果 assert login_page.send_api("/index")["status"] == "success" def test_login_fail(self): login_page = LoginPage() login_page.input_username("test_user") login_page.input_password("wrong_password") login_page.click_login_button() # 断言登录失败后的响应结果 assert login_page.send_api("/index")["status"] == "fail" ``` 以上是一个基于Python的接口自动化POM模式的模板,供你参考。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值