一文詳解Python中PO模式的設計與實現

在使用 Python 進行編碼的時候,會使用自身自帶的編碼設計格式,比如說最常見的單例模式,稍微抽象一些的抽象工廠模式等等… 在利用 Python 做自動化測試的時候,是不是也有自己的設計模式呢?所以在今天這個小章節裡,需要續瞭解的就是 python 作為自動化測試裡面的一種設計模式,尤其是 UI自動化 的專屬模式 —> “PageObject” 自動化設計模式,簡稱 “PO模式” 。

瞭解並實現 “PageObject” 自動化設計模式

什麼是PO模式

一種在測試自動化中變得流行的設計模式,使得自動化測試腳本的代碼量減少,避免代碼重復,更加易讀,減少維護的成本。

其實簡單來說就是將頁面的操作、腳本的Case、通用的頁面元素分開的這樣一個模式。

一般 PO 設計模式多數分為三層

PO 三層模式

第一層:(核心、BasePage層)

  • 對 Selenium 的底層進行二次封裝,定義一個所有頁面都繼承的基礎屬性頁面 —> BasePage 。
  • 封裝 Selenium 的基本方法,例如:元素定位、元素等待、導航頁面、頁面跳轉等等…
  • PS:其實在使用的過程中不需要全部封裝,用到多少方法就封裝多少方法即可。(之前接觸過其他大佬的自動化框架,他把所有的 selenium 的底層的方法做瞭一層封裝,這樣做很好,能夠做很多的事情,但是比較繁重。實際上在真實使用的時候用不到那麼多,所以不建議全部封裝)。

第二層:(頁面層、也叫配置層)

  • 頁面元素進行分離,每個元素隻定位一次,隔離定位。如果頁面改變,隻需要改變相應的元素定位。
  • 如果存在一些業務的屬性、方法,需要將其通過業務方法的方式將業務與操作元素的動作分離開來。

第三層:(封裝測試層)

使用單元測試框架對業務邏輯進行封裝測試

PO 設計模式的優點

UI 頁面的頻繁變化,導致頁面 UI 元素頻繁的變動,PO設計模式便於元素定位改變的維護。

傳統線性自動化,多個用例腳本中需要反復的定位同一個元素,PO設計模式可以減少這部分頻繁定位元素的代碼量

小節:減少重復代碼的冗餘,便於UI頁面頻繁變更下的元素定位維護。

將改寫的腳本轉為PO設計模式

首先在項目裡創建一個 python package 命名為 pages ,然後在 pages 創建一個模塊 base_page.py 用來作為第一層的 base_page核心層 。

如下圖:

構建基礎的 BasePage 層

嘗試構建最基礎的 base_page 層,代碼示例如下:

# coding:utf-8
from selenium import webdriver

class BasePage(object):
    """
    1、第一層 - 核心層-BasePage層,定義一個所有頁面都繼承的page層
    2、對將要使用的 selenium 的底層方法進行二次封裝
    """

    def __init__(self, driver, path=None):     # 構造函數,類的初始化
        """
        為瞭方便編寫將 driver 初始化,
        先使用 "self.driver = webdriver.Chrome()" 後續改為 self.driver = driver
        """
        self.driver = webdriver.Chrome()
        # self.driver = driver
        self.driver.implicitly_wait(5)  # 定義全局的默認加載時間
        self.load_page(path)            # 訪問並加載網頁

    def load_page(self, path=None):     # 訪問並加載網頁,如果 path 不為空的話,直接傳給 driver.get() 訪問
        if path is not None:
            self.driver.get(path)

    def by_xpath(self, xpath):          # 二次封裝 selenium 的 xpath 元素定位
        return self.driver.find_element_by_xpath(xpath)

    def js_click(self, xpath):          # JavaScript 定位元素,並執行 click
        self.driver.execute_script('arguments[0].click()', self.by_xpath(xpath))

到這裡,base_page 層算是寫完瞭,這就是一個最底層、最基礎的類,這個類讓我們實現瞭 selenium 底層的 Xpath 定位方法 與 JavaScript 定位元素方法,這些方法能夠幫助我們更好的去完成後續的定位處理操作。

ok,接下我們再去編寫各個頁面層的東西。

構建首頁的 Page 層(HomePage)

代碼示例如下:

# coding:utf-8

from selenium import webdriver
from pages.base_page import BasePage    # 導入 base_page 層

class HomePage(BasePage):      # 定義 FirstPage(繼承 BasePage )
    """
    1、第二層 - 各個頁面單獨封裝成層,頁面的元素、操作、流程
    """
    def direct_to_login(self):      # 首頁跳轉至登錄頁
        return self.by_xpath("//*[@id='app']/div[1]/div[5]/div[3]")

    def direct_to_product(self):    # 登陸成功後,跳轉至首頁
        return self.by_xpath("//*[@id='app']/div[1]/div[5]/div[1]")

    # 方法流程
    def cross_to_login(self):
        self.direct_to_login().click()  # 點擊 "登錄" 按鈕進行登錄

    def cross_to_product(self):
        self.direct_to_product().click()    # 點擊 "首頁" 跳轉至首頁

構建登錄頁的 Page 層(LoginPage)

代碼示例如下:

# coding:utf-8
from selenium import webdriver
from pages.base_page import BasePage    # 導入 base_page 層

class LoginPage(BasePage):      # 定義 FirstPage(繼承 BasePage )
    """
    1、頁面層(登錄頁) - 各個頁面單獨封裝成層,頁面的元素、操作、流程
    """
    def login_username(self):    # 登錄頁 - 用戶名輸入框
        return self.by_xpath("//*[@id='app']/div[1]/form/div[1]/div[2]/div/input")

    def login_password(self):    # 登錄頁 - 密碼輸入框
        return self.by_xpath("//*[@id='app']/div[1]/form/div[2]/div[2]/div/input")

    def login_button(self):      # 登錄頁 - 登錄按鈕
        return self.by_xpath("//*[@id='app']/div[1]/form/div[3]/button")

    # 登錄Case
    def login(self, username, password):    # 登錄方法,傳入 username 與 password
        self.login_username().send_keys(username)
        self.login_password().send_keys(password)
        self.login_button().click()

構建 首頁 – 訂單 – 支付 流程的 Page 層(OrderPage)

# coding:utf-8

from time import sleep
from pages.base_page import BasePage    # 導入 base_page 層

class OrderPage(BasePage):      # 定義 FirstPage(繼承 BasePage )
    """
    1、頁面層(登錄頁) - 各個頁面單獨封裝成層,頁面的元素、操作、流程
    """
    def product(self):    # 下單 - 第一個產品
        return self.by_xpath("//*[@id='app']/div[1]/div[4]/div[2]/a[1]")

    def ticket_book(self):  # 門票 - 預定(按鈕)
        return self.by_xpath("//*[@id='app']/div[1]/div[5]/div[2]/div[2]/a")

    def book_date(self):    # 門票 - 選擇日期
        return self.by_xpath("//*[@id='app']/div[1]/form/div[1]/div[1]/div[2]/div/input")

    def to_order(self):     # 門票下單
        return self.by_xpath("//*[@id='app']/div[1]/form/div[4]/div/button")

    def pay_off(self):      # 門票下單 - 支付
        return self.by_xpath("//*[@id='app']/div[1]/form/div/div/button")

    def confirm(self):      # 門票下單 - 確認支付
        return self.by_xpath("/html/body/div[5]/div[3]/button[2]")

    # 下單成功Case
    def place_order(self):
        self.product().click()
        self.ticket_book().click()
        self.book_date().send_keys("2022-06-16")
        self.to_order().click()
        sleep(2)
        element = self.pay_off()
        self.driver.execute_script('arguments[0].click()', element)
        sleep(2)

以上,我們準備的所有頁面需要準備的元素定位、基線流程算是寫完瞭,但是具體的用例,應該如何實現呢?繼續往下看。

PO 設計模式下測試Case的改造

代碼示例如下:

# coding:utf-8

import unittest
from time import sleep
from selenium import webdriver
from pages.home_page import HomePage
from pages.login_page import LoginPage
from pages.order_page import OrderPage

'''
1、初始化 - 打開瀏覽器,設置瀏覽器大小
2、最終操作 - 關閉瀏覽器
3、用例部分 - 登錄 與 購買操作、下訂單、支付
'''

class TestTravel(unittest.TestCase):
    @classmethod
    def setUpClass(cls):                    # 每個測試類在加載之前執行一次 setUpClass ,初始化方法
        cls.driver = webdriver.Chrome()
        cls.driver.maximize_window()

    def test_a_order(self):
        #初始化參數
        username = '13500000001'
        password = 'Success@2020'

        #初始化界面
        home_page = HomePage(driver=self.driver, path="http://django.t.mukewang.com/#/")
        login_page = LoginPage(driver=self.driver)
        order_page = OrderPage(driver=self.driver)

        #跳轉登錄
        home_page.cross_to_login()

        #登錄
        login_page.login(username, password)

        # 跳轉至訂單頁
        home_page.cross_to_product()

        #下單
        order_page.place_order()


    @classmethod
    def tearDownClass(cls):
        cls.driver.quit()       # 徹底退出瀏覽器


if __name__ == '__main__':
    unittest.main()

這裡改造完成之後,記得將 "BasePage 層" 的 '# self.driver = driver' 取消註釋,並將 'self.driver = webdriver.Chrome()' 註釋掉 。

以上就是一個比較完整的通過 PO 的方式來連接三個頁面與基礎的 base_page 來寫出的更簡潔一些的測試用例。

運行結果如下:(速度可能過快,擔待一下,gif 隻有15秒的時間)

到此這篇關於一文詳解Python中PO模式的設計與實現的文章就介紹到這瞭,更多相關Python PO模式內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: