Python + Selenium PO 设计模式实践

前面讲了 PO 模式的思想,接下来我们讲一下常见的 PO 模式的实践。

在 Python 领域(Java 中 Selenium 提供了 page factory)中,由于没有官方或者说非常标准的实践模式,再加之 Python 又是一种支持函数式编程的语言,所以对 PO 的实践有很多实现方式。

目前我所知的有三种模式:

  • 以模块(页面文件)的方式收集页面元素定位和操作;
  • 以类为单位,在类中以类属性的形式只收集页面定位而不包含操作;
  • 以类为单位,包含页面元素定位和操作。

以上三种方式都有实践,第一种方式没有使用面向对象并非真正意义上的 PO;第二种方式只提取了定位,并没有封装操作;第三种模式更接近真正意义上的 PO 模式。

我们以前面讲过的登录为例,为了方便理解,我这里简化了步骤和代码。

第一种方式:

将所有的定位方式和操作都写在页面文件中,定位方式以常量(为阅读方便常量用了小写,这不太符合约定)的形式编写,操作以函数的形式编写:

页面文件 login_page.py

from selenium.webdriver.common.by import By

# 定位方式
username_input = By.NAME, 'username'
password_input = By.NAME, 'password'
login_button = By.NAME, 'submit'
logout_link = By.LINK_TEXT, '退出'

# 页面操作
def login(driver, username, password):
    '''登录操作'''
    driver.find_element(*username_input).send_keys(username)
    driver.find_element(*password_input).send_keys(password)
    driver.find_element(*login_button).click()

def logout(driver):
    '''登出操作'''
    driver.find_element(*logout_link )

用例文件 test_login.py

import unittest
from selenium import webdriver
from login_page import login

class TestLogin(unittest.TestCase):
    def setUp(self):
        self.driver = webdriver.Chrome()
    
    def test_login(self):
        login(self.driver, 'rightuser','123456')
        self.assertEqual(login_text, '登录成功')

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

这种方式,你不得不在函数参数中加入 driver 参数。如果多个页面涉及到类似的操作,你依然得每个页面都重复编写。

其抽象程度不高,也就是面向过程级别的。相对于面向对象的难于理解,这种方式显得很直观,所以初学者选择这种方式也可以提高一定的可维护性。

第二种方式:

使用面向对象的方式,只是抽象的页面类中只包含元素定位。

页面文件 login_page.py

from selenium.webdriver.common.by import By

# 定位方式
class Login_Page:
    username_input = By.NAME, 'username'
    password_input = By.NAME, 'password'
    login_button = By.NAME, 'submit'
    logout_link = By.LINK_TEXT, '退出'

用例文件 test_login.py

import unittest
from selenium import webdriver
from login_page import Login_Page as lp

class TestLogin(unittest.TestCase):
    def setUp(self):
        self.driver = webdriver.Chrome()
    
    def test_login(self):
        driver = self.driver
        driver.find_element(*lp.username_input).send_keys('rightuser')
        driver.find_element(*lp.password_input).send_keys('123456')
        driver.find_element(*lp.logout_link).click()
        self.assertEqual(login_text, '登录成功')

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

这种方式虽然用了面向对象,其实只是用了类将定位方式集中到一起。如果页面的操作方式发生变化,也意味着测试用例的更改。

同第二种方式一样,抽象程度不高,容易理解。问题依然是抽象程度不够,页面变化依然会引起测试代码的变化。

第三种方式:

第三种方式,相当于是前两种的集合,也是更符合 PO 模式的实践方式。通过将定位方式和操作全部封装,这样可以通过面向对象的特性很容易的进行更高层次的抽象和封装。

页面文件 page.py

可以在这里将每个页面都会涉及到的内容进行封装,比如等待、select 元素处理、重新定义 find_element 方法等。

class Page:
    def __init__(self, driver):
        self.driver = driver

    def wait(self, locator, timeout=10):
        pass
    
    def select(self, locator, option):
        pass
        
    def find(self, locator):
        pass

页面文件 login_page.py

继承 Page 类,就可以运用自己定义的方法,提高代码灵活性和开发速度:


class Login_Page(Page):
    # 定位方式
    username_input = By.NAME, 'username'
    password_input = By.NAME, 'password'
    login_button = By.NAME, 'submit'
    logout_link = By.LINK_TEXT, '退出'
    
    # 页面操作
    def login(self, username, password):
        '''登录操作'''
        self.wait(self.username_input)
        driver.find(self.username_input).send_keys(username)
        driver.find(self.password_input).send_keys(password)
        driver.find(self.login_button).click()

    def logout(self):
        '''登出操作'''
        driver.find(self.logout_link )

用例文件 test_login.py

import unittest
from selenium import webdriver
from login_page import Login_Page

class TestLogin(unittest.TestCase):
    def setUp(self):
        self.driver = webdriver.Chrome()
    
    def test_login(self):
        lp = Login_Page(self.driver)
        lp.login('righteruser', '123456')
        self.assertEqual(login_text, '登录成功')
        lp.logout()

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

网上讲的最多应该就是第三种,因为前面两种算不上真正意义上的 PO。当然,这也不见得是最好的,最终还要取决于你的项目情况和框架。只要能领会到这种设计模式的意义和目的,至于具体代码各有各的变化。

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

推荐阅读更多精彩内容