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()

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342