Selenium page object模式Python

news2025/1/23 15:00:28

目录

概述

优点

示例

项目结构:

基础页面类BasePage

业务页面类BaiduHomePage

测试类test_baidu:

文件工具类file_util

运行日志:

测试结果:


概述

在web应用程序的UI中,有一些区域可以与测试交互。页面对象仅将这些对象建模为测试代码中的对象。这减少了重复代码的数量,意味着如果UI发生更改,则只需在一个位置应用修复。

页面对象是一种在测试自动化中流行的设计模式,用于增强测试维护和减少代码重复。页面对象是一个面向对象的类,用作AUT页面的接口。然后,每当需要与该页面的UI交互时,测试就会使用该页面对象类的方法。好处是,如果页面的UI发生了更改,则测试本身不需要更改,只需要更改页面对象中的代码。随后,所有支持新UI的更改都位于同一位置。

优点

测试代码和特定于页面的代码之间有一个清晰的分离,例如定位器(或者如果您使用的是UI映射,则使用它们)和布局。

  • 页面提供的服务或操作只有一个存储库,而不是将这些服务分散在整个测试中。
  • 在这两种情况下,这都允许在一个地方进行由于UI更改而需要的任何修改。

示例

演示一个百度搜索功能搜索关键字selenium并做断言,然后保存断言结果截图到对应日期文件夹。

项目结构:

基础页面类BasePage

封装页面基本操作,主要是跟业务不相关的公共方法,如:元素查找、点击、文本输入、截图、双击、获取元素坐标等,跟页面相关的方法不建议封装在基础页面类中,以下是一个简单的实例:

import logging
import time
import traceback

from selenium.webdriver import ActionChains
from selenium.webdriver.common.actions.action_builder import ActionBuilder
from selenium.webdriver.common.by import By
from selenium.webdriver.support.wait import WebDriverWait
import selenium.webdriver.support.expected_conditions as EC
import os
from datetime import datetime


class Action:
    def __init__(self, dirver):
        self.driver = dirver

    def find_element(self, loc):
        try:
            # WebDriverWait(self.driver, 10).until(EC.visibility_of_all_elements_located(loc))
            WebDriverWait(self.driver, 15).until(lambda x: x.find_element(*loc).is_displayed())

            element = self.driver.find_element(*loc)
            self.get_element_coordinate(element)
        except Exception as e:
            traceback.print_exception(e)
            traceback.print_stack()
            logging.info("页面中没有%s %" % (self.loc[1]))
        else:
            return element

    def find_elements(self, loc):
        try:
            # 保证元素可见
            WebDriverWait(self.driver, 15).until(EC.visibility_of_all_elements_located(loc))

            elements = self.driver.find_elements(*loc)
        except Exception as e:
            traceback.print_exception(e)
            traceback.print_stack()
            logging.info("页面中没有%s %" % (self.loc[1]))
        else:
            return elements

    def get_element_coordinate(self, element):
        rect = element.rect
        x, y = rect['x'], rect['y']
        size = element.size
        width = size['width']
        height = size['height']
        left_up = (x, y)
        left_down = (x, y + height)
        center = (x + 0.5 * width, y + 0.5 * height)
        right_up = (x + width, y)
        right_down = (x + width, y + height)
        element_coordinate = dict(left_up=left_up, left_down=left_down, center=center, right_up=right_up,
                                  right_down=right_down)
        logging.info(element_coordinate)
        print(element_coordinate)
        return element_coordinate

    def click_element(self, loc):
        self.find_element(loc).click()

    def save_element_picture(self, loc, file_name):
        element = self.find_element(loc)
        element.screenshot(file_name)

    def save_picture(self, filename, browser='FireFox'):
        file_path = os.path.dirname(os.path.dirname(__file__))
        now = datetime.now()

        picture_date_dir = now.strftime("%Y-%m-%d")
        formatted_date = now.strftime("%Y-%m-%d-%H-%M-%S")
        file_path = file_path + '\\picture\\' + picture_date_dir
        picture_path = file_path + '\\' + filename + '_' + formatted_date + '.png'
        print(picture_path)
        # 检查文件路径是否存在
        if not os.path.exists(file_path):
            print('创建文件路径')
            # 如果文件路径不存在,创建它
            os.makedirs(file_path)
        if browser == 'FireFox':

            self.driver.save_full_page_screenshot(picture_path)
        elif browser == 'Chrome':
            self.driver.get_screenshot_as_file(picture_path)

    def save_long_picture(self, filename, browser='FireFox'):
        # file_path=

        # 截长图
        self.driver.save_full_page_screenshot(filename)

    def click_element_with_coordinate(self, x, y):
        action = ActionBuilder(self.driver)
        action.pointer_action.move_to_location(x, y).click()
        action.perform()

    def double_click(self, loc):
        element = self.find_element(loc)
        ActionChains(self.driver).double_click(element).perform()

    
   
    def set_high_light_elment(self, element):
        script = '''
                // 高亮显示元素
        arguments[0].style.backgroundColor = "yellow";

        // 设置红色边框
        arguments[0].style.border = "3px solid red";
        '''
        self.driver.execute_script(script, element)

        pass

    def browser_2_windows_coordinates_v2(self, browserX, browserY, screenWidth=1360, screenHeight=768,
                                         desktopScale=1):
        # location = self.get_element_location(element)
        # x, y = location['left_up'][0], location['left_up'][1]
        script = '''
        function getDesktopCoordinates(browserX, browserY,screenWidth,screenHeight,desktopScale) {{
        // 浏览器中的坐标(x, y)
        var browserX = browserX;
        var browserY = browserY;
        // 屏幕分辨率
        var screenWidth = screenWidth;
        var screenHeight = screenHeight;
        // 桌面缩放比例
        var desktopScale = desktopScale;
        //- 浏览器窗口左上角的桌面坐标为(win_x, win_y)。
        var win_x = window.screenX || window.screenLeft;
        var win_y = window.screenY || window.screenTop;
        //计算工具栏高度
        var toolbarHeight = window.outerHeight - window.innerHeight;
        // 计算桌面坐标
        var desktopX =(win_x+ browserX) * (screenWidth/window.innerWidth) ;
        var desktopY =(win_y+ browserY+toolbarHeight) * (screenHeight/ window.innerHeight );
        console.log("桌面坐标 (x, y):", desktopX, desktopY);
        // 创建包含坐标的对象
        var desktopCoordinates = {{
        desktopX: desktopX,
        desktopY: desktopY
        }};
        return desktopCoordinates;
        }}
        var coordinates = getDesktopCoordinates({browserX}, {browserY},{screenWidth},{screenHeight},{desktopScale});
        return coordinates;
        '''.format(browserX=browserX, browserY=browserY, screenWidth=screenWidth, screenHeight=screenHeight,
                   desktopScale=desktopScale)
        logging.info(script)
        desktopCoordinates = self.driver.execute_script(script)
        logging.info(desktopCoordinates)
        return desktopCoordinates

    def mark_dom(self, x, y, color='red'):
        script = '''
        // 创建黑点DOM
        const dot = document.createElement('div');
        dot.style.position = 'absolute';
        dot.style.width = '10px';
        dot.style.height = '10px';
        dot.style.backgroundColor = '{}';
        dot.style.borderRadius = '50%';
        dot.style.left = {} + 'px';
        dot.style.top = {} + 'px';
        document.body.appendChild(dot);
        '''.format(color, round(x, 0), round(y, 0))
        logging.info(script)
        print(script)

        self.driver.execute_script(script)

    def mark_dom_text(self, text, x, y):
        script = '''
        // 创建一个新的标记元素
        var newElement = document.createElement("span");
        // 设置标记元素的文本内容
        newElement.innerText = "{}";
        // 设置标记元素的位置样式
        newElement.style.position = "absolute";
        newElement.style.left = "{}px";
        newElement.style.top = "{}px";
        newElement.style.color = "red";
        // 将新的标记元素附加到目标元素中
        document.body.appendChild(newElement);
        '''.format(text, round(x, 0), round(y, 0))

        logging.info(script)
        self.driver.execute_script(script)

业务页面类BaiduHomePage

该类主要是描述待测页面中的元素及对应的操作,当页面元素发生变更时可以快速在该页面进行修改降低了业务代码和测试代码的耦合性

import time

import pyautogui as pyautogui
from selenium.webdriver.common.by import By

from PythonPractise.selenium_po.page import BasePage


class BaiduHomePage(BasePage.Action):
    search_input_loc = (By.ID, '''kw''')
    search_btn_loc = (By.ID, '''su''')

    def search_text(self):
        search_input=self.find_element(self.search_input_loc)
        self.set_high_light_elment(search_input)
        location=self.get_element_coordinate(search_input)
        browser_x,browser_y=location['center'][0],location['center'][1]
        self.mark_dom(browser_x,browser_y)
        desktopCoordinates=self.browser_2_windows_coordinates_v2(browser_x,browser_y)
        x,y=desktopCoordinates['desktopX'],desktopCoordinates['desktopY']
        # 移动到拖拽元素中心坐标
        pyautogui.moveTo(x, y, duration=1, tween=pyautogui.linear)
        time.sleep(10)
        self.find_element(self.search_input_loc).send_keys('Selenium')
        self.click_element(self.search_btn_loc)
        self.save_picture('selenium')

测试类test_baidu:

测试方法类,主要描述对页面逻辑测试的验证,在此方法中会初始话待测页面类,如本例中的百度首页的测试,本类主要关注业务逻辑的操作及验证,引入页面对象降低测试代码和业务代码的耦合性增强了代码的可读性,也降低了和页面代码的耦合性。

该方法首先打开百度首页,然后搜索对应的关键字并做校验,方法结束后关闭测试会话。

import time

from webdriver_manager.chrome import ChromeDriverManager
from selenium.webdriver.chrome.service import Service as ChromeService
from selenium.webdriver.firefox.service import Service as FirefoxService
from webdriver_manager.firefox import GeckoDriverManager

from PythonPractise.selenium_po.page import BaiduHomePage
from selenium import webdriver


class TestCase:

    def setup_class(self):
        # self.driver = webdriver.Chrome(service=ChromeService(ChromeDriverManager().install()))
        # self.driver = webdriver.Chrome(executable_path='F:\\PycharmProjects\\PythonPractise\\selenium_po\\driver\\chromedriver.exe')
        self.driver= webdriver.Firefox(service=FirefoxService(GeckoDriverManager().install()))
        self.driver.get("http://www.baidu.com")

    def test_baidu_search(self):
        baidu_home_page = BaiduHomePage.BaiduHomePage(self.driver)
        baidu_home_page.search_text()
        time.sleep(5)
        baidu_home_page.save_picture('selenium')
        assert 'selenium' in self.driver.page_source

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

文件工具类file_util

此方法主要是定义项目的文件位置方便后面的截图文件、日志文件等其他测试过程中产生的文件的保存,大家根据实际需求进行扩展。

import os
from datetime import datetime

# 获取当前日期和时间
now = datetime.now()

# 格式化为 yyyy-mm-dd
formatted_date = now.strftime("%Y-%m-%d :%H:%M:%S")

print(formatted_date)
project_path = os.path.dirname(os.path.dirname(__file__))
print(project_path)

运行日志:

E:\Python3.11\python.exe "E:/PyCharm Community Edition 2023.1.1/plugins/python-ce/helpers/pycharm/_jb_pytest_runner.py" --path F:\PycharmProjects\PythonPractise\selenium_po\testcase\test_baidu.py 
Testing started at 15:42 ...
Launching pytest with arguments F:\PycharmProjects\PythonPractise\selenium_po\testcase\test_baidu.py --no-header --no-summary -q in F:\PycharmProjects\PythonPractise\selenium_po\testcase

============================= test session starts =============================
collecting ... collected 1 item

test_baidu.py::TestCase::test_baidu_search 

============================= 1 passed in 42.88s ==============================
PASSED                        [100%]{'left_up': (353.0, 188.3999938964844), 'left_down': (353.0, 232.3999938964844), 'center': (628.0, 210.3999938964844), 'right_up': (903.0, 188.3999938964844), 'right_down': (903.0, 232.3999938964844)}
{'left_up': (353.0, 188.3999938964844), 'left_down': (353.0, 234.3999938964844), 'center': (629.0, 211.3999938964844), 'right_up': (905.0, 188.3999938964844), 'right_down': (905.0, 234.3999938964844)}

        // 创建黑点DOM
        const dot = document.createElement('div');
        dot.style.position = 'absolute';
        dot.style.width = '10px';
        dot.style.height = '10px';
        dot.style.backgroundColor = 'red';
        dot.style.borderRadius = '50%';
        dot.style.left = 629.0 + 'px';
        dot.style.top = 211.0 + 'px';
        document.body.appendChild(dot);
        
{'left_up': (353.0, 188.3999938964844), 'left_down': (353.0, 234.3999938964844), 'center': (629.0, 211.3999938964844), 'right_up': (905.0, 188.3999938964844), 'right_down': (905.0, 234.3999938964844)}
{'left_up': (725.0, 15.0), 'left_down': (725.0, 55.0), 'center': (781.0, 35.0), 'right_up': (837.0, 15.0), 'right_down': (837.0, 55.0)}
F:\PycharmProjects\PythonPractise\selenium_po\picture\2023-12-02\selenium_2023-12-02-15-42-56.png
F:\PycharmProjects\PythonPractise\selenium_po\picture\2023-12-02\selenium_2023-12-02-15-43-01.png

进程已结束,退出代码0

测试结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1279517.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

神经网络模型预训练

根据神经网络各个层的计算逻辑用程序实现相关的计算,主要是:前向传播计算、反向传播计算、损失计算、精确度计算等,并提供保存超参数到文件中。 # coding: utf-8 import sys, os sys.path.append(os.pardir) # 为了导入父目录的文件而进行的…

2023年12月4日支付宝蚂蚁庄园小课堂小鸡宝宝考考你今日正确答案是什么?

问题:你知道电杆上安装的“小风车”有什么用途吗? 答案:防止鸟类筑巢 解析:小风车一般做成橙色,因为橙色是一种可令野鸟产生恐慌感的颜色;小风车在转动时,会发出令野鸟害怕的噪声;…

【iOS】数据持久化(三)之SQLite3及其使用

目录 数据库简介什么是SQLite?在Xcode引入SQLite APISQL语句的种类存储字段类型 SQLite的使用创建数据库创建表和删表数据表操作增(插入数据INSERT)删(删除数据DELETE)改(更新数据UPDATE)查&…

softmax实现

import matplotlib.pyplot as plt import torch from IPython import display from d2l import torch as d2lbatch_size 256 train_iter,test_iter d2l.load_data_fashion_mnist(batch_size) test_iter.num_workers 0 train_iter.num_workers 0 num_inputs 784 # 将图片…

【msg_msg】corCTF2021-msgmsg 套题

前言 该套题共两题,一道简单模式 fire_of_salvation,一道困难模式 wall_of_perdition,都是关于 msg_msg 的利用的。这题跟之前的 TPCTF2023 core 的很像(应该是 TPCTF2023 core 跟他很像,bushi)。 其中 f…

ISP算法简述-BLC

Black Level Calibration, 黑电平矫正 现象 1)在纯黑条件下拍张图,你会发现像素值不为0 2)或者你发现图像整体偏色 这些问题可能是黑电平导致的。 原因 存在黑电平的原因有2个: 1)sensor的电路本身存在暗电流。暗电流主要产生在光电信号转换过程中&#…

人工智能 - 人脸识别:发展历史、技术全解与实战

目录 一、人脸识别技术的发展历程早期探索:20世纪60至80年代技术价值点: 自动化与算法化:20世纪90年代技术价值点: 深度学习的革命:21世纪初至今技术价值点: 二、几何特征方法详解与实战几何特征方法的原理…

【C语言】深入理解指针(1)

前言 C语言是一种直接操作内存的编程语言,我们可以直接访问和操作计算机内存中的地址空间。 而C语言中存在的指针类型,指针指向的就是内存中的地址。我们可以通过指针来访问和修改内存中存储的数据。 因此,深入理解指针,并且理解内…

基于SSH的员工管理系统(一)——包结构

基于SSH的员工管理系统(一)——包结构 包结构 1、整体包结构 2、action包 3、domain实体包 4、service层 5、dao层 6、util工具包 7、页面层

【Oracle】数据库登陆错误:ORA-28000:the account is locked解决方法

问题描述 在连接Oracle数据库的时候出现了ORA-28000:the account is locked报错,登录账号被锁定,出现这种情况就需要将被锁定用户解锁。 解决方法 解锁方法就是通过用system账号登录数据库,然后修改被锁定账户状态,具体如下图所示…

03 数仓平台 Kafka

kafka概述 定义 Kafka 是一个开源的分布式事件流平台(Event Streaming Plantform),主要用于大数据实时领域。本质上是一个分布式的基于发布/订阅模式的消息队列(Message Queue)。 消息队列 在大数据场景中主要采用…

右值引用和移动语句(C++11)

左值引用和右值引用 回顾引用 我们之前就了解到了左值引用,首先我们要了解引用在编译器底层其实就是指针。具体来说,当声明引用时,编译器会在底层生成一个指针来表示引用,但在代码编写和使用时,我们可以像使用变量类…

鸿蒙绘制折线图基金走势图

鉴于鸿蒙下一代剥离aosp,对于小公司而言,要么用h5重构,要么等大厂完善工具、等华为出转换工具后跟进,用鸿蒙重新开发一套代码对于一般公司而言成本会大幅增加。但对于广大开发者来说,暂且不论未来鸿蒙发展如何&#xf…

中国消费电子行业发展趋势及消费者需求洞察|徐礼昭

一、引言 近年来,随着科技的飞速发展,消费电子行业面临着前所未有的挑战与机遇。本文将从行业发展趋势、消费者需求洞察以及企业数字化转型的方向和动作三个方面,对消费电子行业进行深入剖析。 二、消费电子行业发展趋势 5G技术的普及和应…

WEB安全之Python

WEB安全之python python-pyc反编译 python类似java一样,存在编译过程,先将源码文件*.py编译成 *.pyc文件,然后通过python解释器执行 生成pyc文件 创建一个py文件随便输入几句代码(1.py) 通过python交互终端 >>>import py_compil…

【C++练级之路】【Lv.1】C++,启动!(命名空间,缺省参数,函数重载,引用,内联函数,auto,范围for,nullptr)

目录 引言入门须知一、命名空间1.1 作用域限定符1.2 命名空间的意义1.3 命名空间的定义1.4 命名空间的使用 二、C输入&输出2.1 cout输出2.2 cin输入2.3 std命名空间的使用惯例 三、缺省参数3.1 缺省参数概念3.2 缺省参数分类 四、函数重载4.1 函数重载概念4.2 函数重载分类…

JavaSE自定义验证码图片生成器

设计项目的时候打算在原有的功能上补充验证码功能,在实现了邮箱验证码之后想着顺便把一个简单的图片验证码生成器也实现一下,用作分享。 注意,实际开发中验证码往往采用各种组件,通过导入依赖来在后端开发时使用相关功能&#xf…

泊车功能专题介绍 ———— 汽车全景影像监测系统性能要求及试验方法(国标未公布)

文章目录 术语和定义一般要求功能要求故障指示 性能要求响应时间图像时延单视图视野范围平面拼接视图视野平面拼接效果总体要求行列畸变拼接错位及拼接无效区域 试验方法环境条件仪器和设备车辆条件系统响应时间试验图像时延试验单视图视野范围试验平面拼接视图视野试验平面拼接…

【大学英语视听说上】Mid-term Test 2

Section A 【短篇新闻1】 You probably think college students are experts at sleeping, but parties, preparations for tests, personal problems and general stress can rack a students sleep habits, which can be bad for the body and the mind. Texas Tech Univer…

51爱心流水灯32灯炫酷代码

源代码摘自远眺883的文章,大佬是30个灯的,感兴趣的铁汁们可以去看看哦~(已取得原作者的许可):基于STC89C51单片机设计的心形流水灯软件代码部分_单片机流水灯代码_远眺883的博客-CSDN博客 由于博主是个小菜鸡&#xff…