• day16-测试自动化之selenium的PO模式


    一、PO模式介绍       

            PO(Page Object)模式是一种在自动化测试中常用的设计模式,将页面的每个元素封装成一个对象,通过操作对象来进行页面的交互。

            一般分为六个版本,现在大部分企业都用的V4版本,三层结构(base+page+scripts)

            V1:不使用任何设计模式和单元测试框架

                    问题:无法批量运行

                    代码例子

    1. # 导包
    2. from time import sleep
    3. from selenium import webdriver
    4. from selenium.webdriver.chrome.service import Service
    5. from selenium.webdriver.common.by import By
    6. from selenium.webdriver.support.ui import WebDriverWait
    7. # 获取浏览器对象
    8. chromedriver_path = r"C:\Program Files\Google\Chrome\Application\chromedriver.exe"
    9. service = Service(executable_path=chromedriver_path)
    10. driver = webdriver.Chrome(service=service)
    11. def main():
    12. # 打开页面
    13. driver.get("http://www.tpshop.com/index.php")
    14. # 网页最大化
    15. driver.maximize_window()
    16. # 隐式等待
    17. driver.implicitly_wait(30)
    18. # 点击登录链接
    19. driver.find_element(By.CSS_SELECTOR,"body > div.tpshop-tm-hander > div.top-hander.clearfix > div > div > div.fl.nologin > a.red").click()
    20. # 输入用户名
    21. driver.find_element(By.CSS_SELECTOR,"#username").send_keys("admin")
    22. # 输入密码
    23. driver.find_element(By.CSS_SELECTOR,"#password").send_keys("123456")
    24. # 输入验证码
    25. driver.find_element(By.CSS_SELECTOR,"#verify_code").send_keys("8888")
    26. # 点击登录按钮
    27. driver.find_element(By.CSS_SELECTOR,"#loginform > div > div.login_bnt > a").click()
    28. # 获取错误提示信息
    29. msg = driver.find_element(By.CSS_SELECTOR,".layui-layer-content").text
    30. # 断言
    31. # assert msg == "用户名不存在!"
    32. # assertNotIn()
    33. # 点击提示框确定按钮
    34. driver.find_element(By.CSS_SELECTOR,".layui-layer-btn0").click()
    35. # 暂停两秒钟
    36. sleep(2)
    37. # 关闭浏览器驱动
    38. driver.quit()
    39. if __name__ == '__main__':
    40. main()

            V2:使用UnitTeSt管理用例

                    问题:业务脚本没有与页面对象分开

            V3:使用方法封装的思想,对代码进行优化

                    问题:代码冗余量太大了

            V4:采用PO模式的分层思想对代码进行拆分

                    结构:

                            base(基类):page页面一些公共的方法;

                                    1.初始化方法

                                    2.查找元素方法

                                    3.点击元素方法

                                    4.输入方法

                                    5.获取文本方法

                                    6.截图方法

                                    扩展:loc变量:类型为元组:*loc为解包

                                    注意:

                                    1.以上方法封装时候,解包只需1此,在查找元素解包

                                    2.driver为虚拟,谁调用base时,谁传入,无需关注从哪里来

                                    3.loc:真正使用1oc的方法只有查找元素方法使用

                                    代码                

    1. import time
    2. import page
    3. from selenium.webdriver.support.wait import WebDriverWait
    4. from base.get_logger import GetLogger
    5. # 获取log日志器
    6. log = GetLogger().get_logger()
    7. class Base:
    8. def __init__(self, driver):
    9. log.info("[base]: 正在获取初始化driver对象:{}".format(driver))
    10. self.driver = driver
    11. # 查找元素方法 封装
    12. def base_find(self, loc, timeout=30, poll=0.5):
    13. log.info("[base]: 正在定位:{} 元素,默认定位超时时间为: {}".format(loc, timeout))
    14. # 使用显示等待 查找元素
    15. return WebDriverWait(self.driver,
    16. timeout=timeout,
    17. poll_frequency=poll).until(lambda x: x.find_element(*loc))
    18. # 点击元素 方法封装
    19. def base_click(self, loc):
    20. log.info("[base]: 正在对:{} 元素实行点击事件".format(loc))
    21. self.base_find(loc).click()
    22. # self.driver.execute_script("arguments[0].click();", self.base_find(loc))
    23. # 输入元素 方法封装
    24. def base_input(self, loc, value):
    25. # 获取元素
    26. el = self.base_find(loc)
    27. # 清空
    28. log.info("[base]: 正在对:{} 元素实行清空".format(loc))
    29. el.clear()
    30. # 输入
    31. el.send_keys(value)
    32. # 获取文本信息 方法封装
    33. def base_get_text(self, loc):
    34. log.info("[base]: 正在获取:{} 元素文本值".format(loc))
    35. return self.base_find(loc).text
    36. # 截图 方法封装
    37. def base_get_image(self):
    38. log.info("[base]: 断言出错,调用截图")
    39. self.driver.get_screenshot_as_file("./image/{}.png".format(time.strftime("%Y_%m_%d %H_%M_%S")))
    40. # 判断元素是否存在 方法封装
    41. def base_element_is_exist(self, loc):
    42. try:
    43. self.base_find(loc, timeout=5)
    44. log.info("[base]: {} 元素查找成功,存在页面".format(loc))
    45. return True # 代表元素存在
    46. except Exception as e:
    47. log.error("[base]:发生错误{},{} 元素查找失败,不存在当前页面".format(e, loc))
    48. return False # 代表元素不存在
    49. # 回到首(页购物车、下订单、支付)都需要用到此方法
    50. def base_index(self):
    51. time.sleep(5)
    52. self.driver.get(page.URL)
    53. # 切换frame表单方法
    54. def base_switch_frame(self, name):
    55. self.driver.switch_to.frame(name)
    56. # 回到默认目录方法
    57. def base_default_content(self):
    58. self.driver.switch_to.default_content()
    59. # 切换窗口 方法 调用此方法
    60. def base_switch_to_window(self, title):
    61. log.info("正在执行切换title值为:{}窗口 ".format(title))
    62. self.driver.switch_to.window(self.base_get_title_handle(title))
    63. # 获取指定title页面的handle方法
    64. def base_get_title_handle(self, title):
    65. # 获取当前页面所有的handles
    66. for handle in self.driver.window_handles:
    67. log.info("正在遍历handles:{}-->{}".format(handle, self.driver.window_handles))
    68. # 切换 handle
    69. self.driver.switch_to.window(handle)
    70. log.info("切换 :{} 窗口".format(handle))
    71. # 获取当前页面title 并判断 是否等于 指定参数title
    72. log.info("判断当前页面title:{} 是否等于指定的title:{}".format(self.driver.title, title))
    73. if self.driver.title == title:
    74. log.info("条件成立! 返回当前handle{}".format(handle))
    75. # 返回 handle
    76. return handle

                            page(页面对象):一个页面封装成一个对象;

                                    应用:继承base;

                                    实现:

                                            1.模块名:page+实际操作模块名称 如:page_login.py

                                            2.页面对象名:以大驼峰方法将模块名抄进来,有下划线去掉下划线

                                            3.方法:涉及元素,将每个元素操作单独封装一个操作方法

                                            4.组装:根据需求组装以上操作步骤

                                    代码

    1. from base.base import Base
    2. import page
    3. from base.get_logger import GetLogger
    4. # 获取log日志器
    5. log = GetLogger().get_logger()
    6. class PageLogin(Base):
    7. # 点击 登录链接
    8. def page_click_login_link(self):
    9. log.info("[page_loging] 执行:{} 点击链接操作".format(page.login_link))
    10. self.base_click(page.login_link)
    11. # 输入用户名
    12. def page_input_username(self, username):
    13. log.info("[page_loging] 对:{} 元素 输入用户名:{} 操作".format(page.login_username, username))
    14. self.base_input(page.login_username, username)
    15. # 输入密码
    16. def page_input_pwd(self, pwd):
    17. log.info("[page_loging] 对:{} 元素 输入密码:{} 操作".format(page.login_pwd, pwd))
    18. self.base_input(page.login_pwd, pwd)
    19. # 输入验证码
    20. def page_input_verify_code(self, verify_code):
    21. log.info("[page_loging] 对:{} 元素 输入验证码:{} 操作".format(page.login_verify_code, verify_code))
    22. self.base_input(page.login_verify_code, verify_code)
    23. # 点击登录按钮
    24. def page_click_login_btn(self):
    25. self.base_click(page.login_btn)
    26. # 获取 错误提示信息
    27. def page_get_error_info(self):
    28. return self.base_get_text(page.login_err_info)
    29. # 点击 错误提示框 确定按钮
    30. def page_click_error_alert(self):
    31. self.base_click(page.login_err_ok_btn)
    32. # 判断是否登录成功
    33. def page_if_login_success(self):
    34. # 注意 一定要将找元素的结果返回,True:存在
    35. return self.base_element_is_exist(page.login_logout_link)
    36. # 点击 安全退出
    37. def page_click_logout_link(self):
    38. self.base_click(page.login_logout_link)
    39. # 判断是否退出成功
    40. def page_if_logout_success(self):
    41. return self.base_element_is_exist(page.login_link)
    42. # 组合业务方法 -->登录业务直接调用
    43. def page_login(self, username, pwd, verify_code):
    44. log.info("[page_loging] 正在执行登录操作, 用户名:{} 密码:{}, 验证码:{}".format(username, pwd, verify_code))
    45. # 调用 输入用户名
    46. self.page_input_username(username)
    47. # 调用 输入密码
    48. self.page_input_pwd(pwd)
    49. # 调用 输入验证码
    50. self.page_input_verify_code(verify_code)
    51. # 调用 点击登录
    52. self.page_click_login_btn()
    53. # 组合登录业务方法 给(购物车模块、订单模块、支付模块)依赖登录使用
    54. def page_login_success(self, username="13812345678", pwd="123456", verify_code="8888"):
    55. # 点击登录连接
    56. self.page_click_login_link()
    57. log.info("[page_loging] 正在执行登录操作, 用户名:{} 密码:{}, 验证码:{}".format(username, pwd, verify_code))
    58. # 调用 输入用户名
    59. self.page_input_username(username)
    60. # 调用 输入密码
    61. self.page_input_pwd(pwd)
    62. # 调用 输入验证码
    63. self.page_input_verify_code(verify_code)
    64. # 调用 点击登录
    65. self.page_click_login_btn()

                            sripts/cases(业务层):导包调用page页面

                                    实现

                                            1.模块:test+实际操作模块名称如:test_login.py

                                            2.测试业务名称:以大驼峰方法将模块名抄进来,有下划线去掉下划线

                                            3.方法

                                                    1).初始化方法setup()注:在unittest框架中不能使用def__init_()方法

                                                            1.1).实例化页面对象

                                                            1.2).前置操作如:打开等等

                                                    2).结束方法teardown

                                                            2.1).关闭驱动

                                                    3).测试方法

                                                            3.1).根据要操作的业务实现

                                    代码

    1. import unittest
    2. from base.get_driver import GetDriver
    3. from page.page_login import PageLogin
    4. from parameterized import parameterized
    5. from tool.read_txt import read_txt
    6. from base.get_logger import GetLogger
    7. # 获取log日志器
    8. log = GetLogger().get_logger()
    9. def get_data():
    10. arrs = []
    11. for data in read_txt("login.txt"):
    12. arrs.append(tuple(data.strip().split(",")))
    13. return arrs[1:]
    14. # 新建 登录测试类 并 继承 unittest.TestCase
    15. class TestLogin(unittest.TestCase):
    16. # 新建 setupClass
    17. @classmethod
    18. def setUpClass(cls):
    19. try:
    20. # 实例化 并获取driver
    21. cls.driver = GetDriver().get_driver()
    22. # 实例化 PageLogin()
    23. cls.login = PageLogin(cls.driver)
    24. # 点击登录连接
    25. cls.login.page_click_login_link()
    26. except Exception as e:
    27. log.error("错误:{}".format(e))
    28. # 截图
    29. cls.login.base_get_image()
    30. # 新建 tearDownClass
    31. @classmethod
    32. def tearDownClass(cls):
    33. # 关闭drier驱动对象
    34. GetDriver().quit_driver()
    35. # 新建 登录测试方法
    36. @parameterized.expand(get_data())
    37. def test_login(self, username, pwd, verify_code, expect_result, status):
    38. try:
    39. # 调用 登录业务方法
    40. self.login.page_login(username, pwd, verify_code)
    41. # 判断是否为正向
    42. if status == "true":
    43. # 断言是否登录成功
    44. try:
    45. self.assertTrue(self.login.page_if_login_success())
    46. except Exception as e:
    47. # 截图
    48. self.login.base_get_image()
    49. log.error("错误:{}".format(e))
    50. raise # 主动抛出异常,否则HTMLTestRunner无法显示该错误
    51. # 点击 安全退出
    52. self.login.page_click_logout_link()
    53. # 点击登录连接
    54. self.login.page_click_login_link()
    55. # 逆向用例
    56. else:
    57. # 获取错误提示信息
    58. msg = self.login.page_get_error_info()
    59. print("msg:", msg)
    60. try:
    61. self.assertEqual(msg, expect_result)
    62. except Exception as e:
    63. # 截图
    64. self.login.base_get_image()
    65. log.error("错误:{}".format(e))
    66. raise
    67. # 点击错误提示框 确定按钮
    68. self.login.page_click_error_alert()
    69. except Exception as e:
    70. log.error("错误:{}".format(e))
    71. # 截图
    72. self.login.base_get_image()
    73. raise

            V5:对PO分层之后的代码继续优化

            V6:PO模式深入封装,把共同操作提取封装到父类中,子类直接调用父类的方法                

    二、今日学习思维导图

  • 相关阅读:
    CSS选择器练习小游戏
    MonoSDF学习笔记
    【kubernetes系列学习】如何使pod和host主机的时间保持一致?
    [含lw+源码等]计算机毕业论文Java项目源码下载微信小程序记事本+后台管理系统[包运行成功]
    解决 net core 3.x 跨域问题
    基于JAVA的RSA文件加密软件的设计与实现
    基于 Webpack5 Module Federation 的业务解耦实践
    用java编写图书管理系统
    SpringBoot整合redis
    HDU_7149
  • 原文地址:https://blog.csdn.net/weixin_61319245/article/details/141093349