0


SeleniumUI自动化的POM三层架构

简介:
企业级自动化测试中,要写的代码太多了,写着写着会发现很多的代码都重复的写,而且硬编码导致了代码很冗余,一大坨一大坨,当出现了问题后,不好剖析原因以及快速找到对应的解决办法。
写成POM三层架构就方便管理代码,而且避免了代码的冗余。
目的:使代码结构清晰,方便管理代码,将代码和业务分开来,一定程度实现了解耦合性。

文件夹层级如下

1、定义log.py日志

①定义了日志的存放的位置,
②定义的日志的最大存放大小,
③定义了日志的输出格式,
④日志即打印在控制台上也写入日志文件中,

import os
import logging
import time

from logging.handlers import RotatingFileHandler

def log(log_level="DEBUG"):
    logger = logging.getLogger()
    logger.setLevel(log_level)
    log_size = 1024*1024*20
    dir_name = "../../logs/"
    if not os.path.exists(dir_name):
        os.mkdir(dir_name)
    time_str = time.strftime("%Y%m%d",time.localtime())
    fh = RotatingFileHandler(dir_name+f"{time_str}.log",encoding="utf-8",maxBytes=log_size,backupCount=100)

    #将日志输出到控制台
    ch = logging.StreamHandler()
    #设置输出日志格式
    formatter = logging.Formatter(
        fmt="%(asctime)s [%(levelname)s] %(filename)s line:%(lineno)s %(message)s",
    )
    #为handler指定输出格式,注意大小写
    fh.setFormatter(formatter)
    ch.setFormatter(formatter)
    #为logger添加的日志处理器
    logger.addHandler(fh)
    logger.addHandler(ch)

2、在基础封装层初始化类uitls.init.py中选择了日志的打印级别,

在browser.py中会使用,使用日志时只需导入logging包,使用logging.info("日志信息")的方式即可

from .log import log
log("INFO")

3、定义页面基础类base_page.py,定义了页面找元素的方法,

①专门服务于页面类,用于寻找元素,
②使用了显示等待,设置了没找到元素时重复寻找的次数和隔多长时间找一次
③定义了driver驱动,则继承他的类在实例化时需要传入驱动

class BasePage(object):
    def __init__(self,driver):
        self.__driver = driver
    def find_element(self,by,value,times=10,wait_time=1)->object:
        return self.__driver.until_find_element(by,value,times=times,wait_time=wait_time)

4、将百度页面的元素以代码形式保存,baidu.py

①HomePage类继承了BasePage类,则可使用他的方法
②方法上添加@property,则调用这方法时可以像调用属性的方式一样(如:eles=对象.link_xinwen)

from automation.demo1.src.utils.base_page import BasePage

class HomePage(BasePage):
    @property
    def link_xinwen(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[1]')
    @property
    def link_hao123(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[2]')
    @property
    def link_ditu(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[3]')
    @property
    def link_tieba(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[4]')
    @property
    def link_shipin(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[5]')
    @property
    def link_tupian(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[6]')
    @property
    def link_wangpan(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/a[7]')
    @property
    def link_gengduo(self):
        return self.find_element('xpath', '//*[@id="s-top-left"]/div/a')
    @property
    def shurukuang(self):
        return self.find_element('xpath', '//*[@id="kw"]')
    @property
    def baiduyixia(self):
        return self.find_element('xpath', '//*[@id="su"]')
    @property
    def shezhi(self):
        return self.find_element('xpath', '//*[@id="s-usersetting-top"]')
    @property
    def denglu(self):
        return self.find_element('xpath', '//*[@id="s-top-loginbtn"]')
    @property
    def link_fanyi(self):
        return self.find_element('xpath','//*[@id="s-top-more"]/div[1]/a[1]')

class NewsPage(BasePage):
    @property
    def help_link(self):
        return self.find_element("xpath",'//*[@id="sbox"]/tbody/tr/td[2]/table/tbody/tr/td[2]/a')

5、重新封装浏览器方法, browser.py

①初始化寻找驱动
②打开浏览器,输入地址,且最大化
③切换窗口
④悬浮到元素
④JS下拉滚动条
⑤下拉滚动条到某位置
⑥下拉滚动条到某元素
⑦JS点击元素
⑧JS使元素变为可点击状态
⑨JS修元素属性
⑩获取浏览器名字和版本
11,智能寻找元素
12,智能点击元素

import logging
import time

from selenium.webdriver import Chrome, Firefox, ActionChains
from selenium.common.exceptions import NoSuchElementException, ElementNotInteractableException

class Browser(Chrome, Firefox):
    def __init__(self, browser_type="chrome", driver_path=None, *args, **kwargs):
        """
        :param browser_type:浏览器类型,如果不传则默认是谷歌浏览器
        :param driver_path:驱动的地址
        :param args:可变列表参数
        :param kwargs:可变字典参数
        """
        # 检查browser_type是否合法
        if browser_type not in ["chrome", "firefox"]:
            logging.error("browser_type浏览器类型即不是谷歌也不是火狐")
            raise ValueError("browser_type浏览器类型既不是谷歌也不是火狐")
        self.__browser_type = browser_type

        # 根据__browser_type的类型选择对于的驱动
        if self.__browser_type == "chrome":
            if driver_path:
                Chrome.__init__(self, executable_path=f"{driver_path}/chromedirver.exe", *args, **kwargs)
            else:
                Chrome.__init__(self, *args, **kwargs)
        elif self.__browser_type == 'firefox':
            if driver_path:
                Firefox.__init__(self, executable_path=f"{driver_path}/geckodriver.exe", *args, **kwargs)
            else:
                Firefox.__init__(self, *args, **kwargs)
    #打开页面且最大化,习惯性需要最大化,不然容易出现问题
    def open_browser(self, url):
        self.get(url)
        self.maximize_window()

    # 切换窗口
    def switch_to_new_page(self):
        # 获取目前窗口与最后窗口的对比
        current_handle = self.current_window_handle
        handles = self.window_handles
        if current_handle != handles[-1]:
            self.switch_to.window(handles[-1])

    # 悬浮到元素
    def suspension(self, ele):
        ActionChains(self).move_to_element(ele).perform()

    # 下拉滚动条-通过滚动滚动条
    def scrollTo1(self):
        self.execute_script("document.documentElement.scrollTop=10000")
        time.sleep(1)

    # 下拉滚动条-通过位置
    def scrollTo2(self):
        self.execute_script("window.scrollTo(0,100000)")
        time.sleep(1)

    # 下拉滚动条到某元素
    def scrollTo3(self, ele):
        self.execute_script("arguments[0].scrollIntoView()", ele)
        time.sleep(1)

    # 使用js点击元素
    def js_click(self, button):
        self.execute_script("$(arguments[0]).click();", button)

    # 使元素变成可点击状态
    def js_enable(self,ele):
        self.execute_script('arguments[0].removeAttribute(\"disabled\")',ele)
    # 修改元素不可点击的状态
    def js_disable(self,ele):
        self.execute_script('arguments[0].setAttribute(arguments[1],arguments[2])',ele,'disabled','enable')

    @property
    def browser_name(self):
        return self.capabilities["browserName"]

    @property
    def browser_version(self):
        return self.capabilities["browserVersion"]

    def until_find_element(self, by, value, times=10, wait_time=0.5):
        """
        用于定位元素
        :param by:定位元素的方式
        :param value:定位元素的值
        :param times:定位元素的重试次数
        :param wait_time:定位元素失败的等待时间
        :return:返回定位的元素
        """
        # 检查by的合法性
        if by not in ["id", "name", "class", "tag", "text", "partial_text", "css", "xpath"]:
            logging.error(f"无效的定位方式:{by},请输入id,name,class,tag,text,partial_text,css,xpath")
        for i in range(times):
            # 定位元素
            el = None
            try:
                if by == "id":
                    el = super().find_element_by_id(value)
                elif by == "name":
                    el = super().find_element_by_name(value)
                elif by == "class":
                    el = super().find_element_by_class_name(value)
                elif by == "tag":
                    el = super().find_element_by_tag_name(value)
                elif by == "text":
                    el = super().find_element_by_link_text(value)
                elif by == "partial_text":
                    el = super().find_element_by_partial_link_text(value)
                elif by == "css":
                    el = super().find_element_by_css_selector(value)
                elif by == "xpath":
                    el = super().find_element_by_xpath(value)
            except NoSuchElementException:
                logging.error(f"通过{by}未能找到元素【{value}】,正在进行第{i + 1}此重试...")
                time.sleep(wait_time)
            else:
                # 如果成功定位元素则返回元素
                logging.info(f"通过{by}成功定位元素【{value}】!")
                return el
        logging.error(f"通过{by}无法定位元素【{value}】,请检查...")
        raise NoSuchElementException(f"哦通过{by}无法定位元素【{value}】,请检查...")
    def click(self,button,repeat=5):
        for i in repeat:
            # 判断元素是否可见和可点击
            if(button.is_displayed() and button.is_enabled()):
                button.click()
                return
        logging.info(f"元素{button},是不可见得,或者是不可点击的")
        raise ElementNotInteractableException(f"元素{button},是不可见的,或者是不可点击的")

if __name__ == '__main__':
    try:
        browser = Browser()
        browser.open_browser('http://sahitest.com/demo/clicks.htm')
        ele_disable1 = browser.find_element('xpath','/html/body/form/input[5]')
        # browser.js_click(ele_disable1)
        browser.js_enable(ele_disable1)
        browser.addAttribute(browser,ele_disable1,'disable')
        time.sleep(2)
        ele_disable1.click()
    finally:
        time.sleep(15)
        browser.quit()

6、编写测试用例

①起始工作,打开浏览器输入地址
②结束工作,关闭浏览器
①用例1:测试百度输入框输入selenium能搜索出包含selenium相关的信息
②用例2:测试通过百度首页进入新闻界面
③用例3:测试能进入hao123界面
。。。。。。

import unittest
import time
import logging
from selenium.webdriver import ActionChains
from automation.demo1.src.pages.baidu import HomePage,NewsPage
from automation.demo1.src.utils.browser import Browser

class Baidu(unittest.TestCase):
    def setUp(self) -> None:
        self.driver = Browser()
        self.driver.open_browser("https://www.baidu.com")
        logging.info("打开浏览器")
        logging.info(f"浏览器名称:{self.driver.browser_name},浏览器版本:{self.driver.browser_version}")

        self.homepage =  HomePage(self.driver)
        self.newpage = NewsPage(self.driver)
    def tearDown(self) -> None:
        self.driver.quit()
        logging.info("关闭浏览器")
    def test_01_search(self):
        """用例1:测试百度输入框输入selenium能搜索出包含selenium相关的信息"""
        logging.info("用例1:测试百度输入框输入selenium能搜索出包含selenium相关的信息")

        #输入搜索信息
        self.homepage.shurukuang.send_keys("selenium_automation")
        logging.info("输入搜索信息")
        #点击按钮
        self.homepage.baiduyixia.click()
        time.sleep(2)
        #校验搜索结果
        els = self.driver.find_element_by_partial_link_text("Selenium")
        self.assertIsNotNone(els)
        logging.info("用例1测试通过")
    def test_02_access_help_news(self):
        """用例2:测试通过百度首页进入新闻界面"""
        logging.info("用例2:测试通过百度首页能进入新闻界面")
        #点击新闻链接
        self.homepage.link_xinwen.click()
        logging.info("点击新闻链接")
        #切换窗口
        self.driver.switch_to_new_page()
        logging.info("切换窗口")
        #点击游戏链接
        self.newpage.help_link.click()
        logging.info("点击游戏链接")
        #校验url
        current_rul = self.driver.current_url
        self.assertEqual(current_rul,"https://help.baidu.com/")
        logging.info("用例2测试通过")
    def test_03_access_hao123(self):
        """用例3:测试能进入hao123界面"""
        logging.info("用例3:测试能进入hao123界面")
        self.homepage.link_hao123.click()
        self.driver.switch_to_new_page()
        logging.info("切换窗口")
        #校验url
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://www.hao123.com/?src=from_pc")
        logging.info("用例3测试通过")
    def test_04_access_map(self):
        """用例4:测试能进入地图界面"""
        logging.info("用例4:测试能进入地图界面")
        self.homepage.link_ditu.click()
        logging.info("切换窗口")
        self.driver.switch_to_new_page()
        #校验url
        current_url = self.driver.current_url
        els= self.driver.find_element('xpath','//*[@id="componseLeft"]/p')
        self.assertIsNotNone(els)
        logging.info("用例4测试通过")
    def test_05_access_tieba(self):
        """用例5: 测试能进入贴啊界面"""
        logging.info("用例5:测试能进入贴吧界面")
        self.homepage.link_tieba.click()
        logging.info("切换窗口")
        self.driver.switch_to_new_page()
        #校验url
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://tieba.baidu.com/index.html")
        logging.info("用例5通过测试覅")
    def test_06_access_shiping(self):
        """用例6:能进入视频界面"""
        logging.info("用例6:能进入视频界面")
        self.homepage.link_shipin.click()
        logging.info("切换窗口")
        self.driver.switch_to_new_page()
        #校验url
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://haokan.baidu.com/?sfrom=baidu-top")
        logging.info("用例6测试通过")
    def test_07_access_tupian(self):
        """用例7:能进入图片界面"""
        logging.info("用例7:能进入视频界面")
        self.homepage.link_tupian.click()
        logging.info("切换窗口")
        self.driver.switch_to_new_page()
        #校验rul
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://image.baidu.com/")
        logging.info("用例7测试通过")
    def test_08_access_wangpan(self):
        """用例8:能进入网盘界面"""
        logging.info("用例8:能进入网盘界面")
        self.homepage.link_wangpan.click()
        logging.info("切换窗口")
        self.driver.switch_to_new_page()
        #校验url
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://pan.baidu.com/?from=1026962h")
        logging.info("用例8测试通过")
    def test_09_access_gengduo(self):
        """用例9:能进入更多界面"""
        logging.info("用例9:能进入更多界面")
        self.homepage.link_gengduo.click()
        logging.info("切换窗口")
        self.driver.switch_to_new_page()
        #校验url
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://www.baidu.com/more/")
        logging.info("用例9测试通过")
    def test_10_access_fanyi(self):
        """用例10,能进入翻译页面"""
        logging.info("用例10:能进入翻译页面")
        #悬浮到设置元素上,后点击进入翻译页面
        self.driver.suspension(self.homepage.link_gengduo)
        self.homepage.link_fanyi.click()
        #校验url
        self.driver.switch_to_new_page()
        current_url = self.driver.current_url
        self.assertEqual(current_url,"https://fanyi.baidu.com/")
        logging.info("用例10测试通过")

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

SeleniumGUI自动化POM三层架构结束,欢迎进我主页观看其它技术类文章~~


本文转载自: https://blog.csdn.net/weixin_45812996/article/details/128295117
版权归原作者 江云的博客 所有, 如有侵权,请联系我们删除。

“SeleniumUI自动化的POM三层架构”的评论:

还没有评论