Selenium+Pytest自动化测试框架 ------ 禅道实战

前言

有人问我登录携带登录的测试框架该怎么处理,今天就对框架做一点小升级吧,加入登录的测试功能。

选用的测试网址为我电脑本地搭建的禅道

更改了以下的一些文件,框架为原文章框架主体

conftest.py更改

conftest.py

#!/usr/bin/env python3
# -*- coding:utf-8 -*-
import base64
import pytest
import allure
from py.xml import html
from selenium import webdriver
from page.webpage import WebPage
from common.readconfig import ini
from tools.send_mail import send_report
from tools.times import timestamp
from config.conf import cmdriver = None@pytest.fixture(scope='session', autouse=True)
def drivers(request):global driverif driver is None:driver = webdriver.Chrome()web = WebPage(driver)web.get_url(ini.url)def fn():driver.quit()request.addfinalizer(fn)return driver@pytest.hookimpl(hookwrapper=True)
def pytest_runtest_makereport(item):"""当测试失败的时候,自动截图,展示到html报告中:param item:"""pytest_html = item.config.pluginmanager.getplugin('html')outcome = yieldreport = outcome.get_result()extra = getattr(report, 'extra', [])if report.when == 'call' or report.when == "setup":xfail = hasattr(report, 'wasxfail')if (report.skipped and xfail) or (report.failed and not xfail):screen_img = _capture_screenshot()if screen_img:html = '<div><img src="data:image/png;base64,%s" alt="screenshot" style="width:1024px;height:768px;" ' \'onclick="window.open(this.src)" align="right"/></div>' % screen_imgextra.append(pytest_html.extras.html(html))report.extra = extrareport.description = str(item.function.__doc__)def pytest_html_results_table_header(cells):cells.insert(1, html.th('用例名称'))cells.insert(2, html.th('Test_nodeid'))cells.pop(2)def pytest_html_results_table_row(report, cells):cells.insert(1, html.td(report.description))cells.insert(2, html.td(report.nodeid))cells.pop(2)def pytest_html_results_table_html(report, data):if report.passed:del data[:]data.append(html.div('通过的用例未捕获日志输出.', class_='empty log'))def pytest_html_report_title(report):report.title = "pytest示例项目测试报告"def pytest_configure(config):config._metadata.clear()config._metadata['测试项目'] = "测试百度官网搜索"config._metadata['测试地址'] = ini.urldef pytest_html_results_summary(prefix, summary, postfix):# prefix.clear() # 清空summary中的内容prefix.extend([html.p("所属部门: XX公司测试部")])prefix.extend([html.p("测试执行人: 随风挥手")])def pytest_terminal_summary(terminalreporter, exitstatus, config):"""收集测试结果"""result = {"total": terminalreporter._numcollected,'passed': len(terminalreporter.stats.get('passed', [])),'failed': len(terminalreporter.stats.get('failed', [])),'error': len(terminalreporter.stats.get('error', [])),'skipped': len(terminalreporter.stats.get('skipped', [])),# terminalreporter._sessionstarttime 会话开始时间'total times': timestamp() - terminalreporter._sessionstarttime}print(result)if result['failed'] or result['error']:send_report()def _capture_screenshot():"""截图保存为base64"""now_time, screen_path = cm.screen_filedriver.save_screenshot(screen_path)allure.attach.file(screen_path, "测试失败截图...{}".format(now_time), allure.attachment_type.PNG)with open(screen_path, 'rb') as f:imagebase64 = base64.b64encode(f.read())return imagebase64.decode()

config.ini更改

[HOST]
HOST = http://127.0.0.1/zentao/user-login-L3plbnRhby9teS5odG1s.html

conf.py更改

#!/usr/bin/env python3
# -*- coding:utf-8 -*-
import os
from selenium.webdriver.common.by import By
from tools.times import datetime_strftimeclass ConfigManager(object):# 项目目录BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))# 日志目录LOG_PATH = os.path.join(BASE_DIR, 'logs')# 报告目录REPORT_PATH = os.path.join(BASE_DIR, 'report', 'report.html')ELEMENT_PATH = os.path.join(BASE_DIR, 'page_element')# 元素定位的类型LOCATE_MODE = {'css': By.CSS_SELECTOR,'xpath': By.XPATH,'name': By.NAME,'id': By.ID,'class': By.CLASS_NAME}# 邮件信息EMAIL_INFO = {'username': '1084502012@qq.com',  # 切换成你自己的地址'password': 'QQ邮箱授权码','smtp_host': 'smtp.qq.com','smtp_port': 465}# 收件人ADDRESSEE = ['1084502012@qq.com',]@propertydef ini_file(self):# 配置文件_file = os.path.join(self.BASE_DIR, 'config', 'config.ini')if not os.path.exists(_file):raise FileNotFoundError("配置文件%s不存在!" % _file)return _filedef element_file(self, name):"""页面元素文件"""element_path = os.path.join(self.ELEMENT_PATH, '%s.yaml' % name)if not os.path.exists(element_path):raise FileNotFoundError("%s 文件不存在!" % element_path)return element_path@propertydef log_path(self):log_path = os.path.join(self.BASE_DIR, 'logs')if not os.path.exists(log_path):os.makedirs(log_path)return os.path.join(log_path, "%s.log" % datetime_strftime())@propertydef screen_file(self):now_time = datetime_strftime("%Y%m%d%H%M%S")# 截图目录screenshot_dir = os.path.join(self.BASE_DIR, 'screen_capture')if not os.path.exists(screenshot_dir):os.makedirs(screenshot_dir)screen_path = os.path.join(screenshot_dir, "{}.png".format(now_time))return now_time, screen_pathcm = ConfigManager()
if __name__ == '__main__':print(cm.BASE_DIR)

page更改

webpage.py

添加了几个函数!

#!/usr/bin/env python3
# -*- coding:utf-8 -*-
"""
selenium基类
本文件存放了selenium基类的封装方法
"""
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.ui import WebDriverWait
from selenium.common.exceptions import TimeoutException, NoSuchElementException
from config.conf import cm
from tools.times import sleep
from tools.logger import Loggerlog = Logger(__name__).loggerclass WebPage(object):"""selenium基类"""def __init__(self, driver):# self.driver = webdriver.Chrome()self.driver = driverself.timeout = 20self.wait = WebDriverWait(self.driver, self.timeout)def get_url(self, url):"""打开网址并验证"""self.driver.maximize_window()self.driver.set_page_load_timeout(60)try:self.driver.get(url)self.driver.implicitly_wait(10)log.info("打开网页:%s" % url)except TimeoutException:raise TimeoutException("打开%s超时请检查网络或网址服务器" % url)@staticmethoddef element_locator(func, locator):"""元素定位器"""name, value = locatorreturn func(cm.LOCATE_MODE[name], value)def find_element(self, locator):"""寻找单个元素"""return WebPage.element_locator(lambda *args: self.wait.until(EC.presence_of_element_located(args)), locator)def find_elements(self, locator):"""查找多个相同的元素"""return WebPage.element_locator(lambda *args: self.wait.until(EC.presence_of_all_elements_located(args)), locator)def focus(self):"""聚焦元素"""self.driver.execute_script("window.scrollTo(0,document.body.scrollHeight)")def elements_num(self, locator):"""获取相同元素的个数"""number = len(self.find_elements(locator))log.info("相同元素:{}".format((locator, number)))return numberdef input_text(self, locator, txt):"""输入(输入前先清空)"""sleep(0.5)ele = self.find_element(locator)ele.clear()ele.send_keys(txt)log.info("输入文本:{}".format(txt))def is_click(self, locator):"""点击"""ele = self.find_element(locator)ele.click()sleep()log.info("点击元素:{}".format(locator))def is_exists(self, locator):"""元素是否存在(DOM)"""try:WebPage.element_locator(lambda *args: EC.presence_of_element_located(args)(self.driver), locator)return Trueexcept NoSuchElementException:return Falsedef alert_exists(self):"""判断弹框是否出现,并返回弹框的文字"""alert = EC.alert_is_present()(self.driver)if alert:text = alert.textlog.info("Alert弹窗提示为:%s" % text)alert.accept()return textelse:log.error("没有Alert弹窗提示!")def element_text(self, locator):"""获取当前的text"""_text = self.find_element(locator).textlog.info("获取文本:{}".format(_text))return _textdef get_attribute(self, locator, name):"""获取元素属性"""return self.find_element(locator).get_attribute(name)@propertydef get_source(self):"""获取页面源代码"""return self.driver.page_sourcedef refresh(self):"""刷新页面F5"""self.driver.refresh()self.driver.implicitly_wait(30)if __name__ == "__main__":pass

page_element更改

login.yaml

账号: "css==input[name=account]"
密码: "css==input[name=password]"
登录: "css==button#submit"
我的地盘: "xpath==//nav[@id='navbar']//span[text()=' 我的地盘']"
右上角名称: "css==.user-name"
退出登录: "xpath==//a[text()='退出']"

product.yaml

产品按钮: "xpath==//nav[@id='navbar']//a[text()='产品']"
添加产品: "xpath==//div[@id='pageActions']//a[text()=' 添加产品']"
产品名称: "css==#name"
产品代号: "css==#code"
保存产品: "css==#submit"
产品列表: "xpath==//ul[@class='nav nav-stacked nav-secondary scrollbar-hover']//a[1]"

page_object更改

loginpage.py

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
from page.webpage import WebPage
from common.readelement import Elementlogin = Element('login')class LoginPage(WebPage):"""登录类"""def username(self, name):"""用户名"""self.input_text(login['账号'], name)def password(self, pwd):"""密码"""self.input_text(login['密码'], pwd)def submit(self):"""登录"""self.is_click(login['登录'])def quit_login(self):"""退出登录"""self.is_click(login['右上角名称'])self.is_click(login['退出登录'])def login_success(self):"""验证登录"""return self.is_exists(login['我的地盘'])

productpage.py

#!/usr/bin/env python3
# -*- coding:utf-8 -*-
from page.webpage import WebPage, sleep
from common.readelement import Elementproduct = Element('product')class ProductPage(WebPage):"""产品类"""def click_product(self):"""点击产品"""self.is_click(product['产品按钮'])def add_product(self):"""添加产品"""self.is_click(product['添加产品'])def add_product_content(self, name, code):"""添加产品内容"""self.input_text(product['产品名称'], name)self.input_text(product['产品代号'], code)def save_product(self):"""保存产品"""self.focus()self.is_click(product['保存产品'])def product_list(self):"""产品列表"""return [i.get_attribute('title') for i in self.find_elements(product['产品列表'])]if __name__ == '__main__':a = product['产品列表'][1] + "[1]"print(a)

TestCase更改

test_login.py

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import pytest
from tools.times import sleep
from page_object.loginpage import LoginPageclass TestLogin:"""测试登录"""@pytest.mark.parametrize("name,pwd", [('admin', 'Admin123456'), ('test', 'test123')])def test_001(self, drivers, name, pwd):login = LoginPage(drivers)login.username(name)login.password(pwd)login.submit()sleep(3)res = login.alert_exists()if res:assert res == "登录失败,请检查您的用户名或密码是否填写正确。"elif login.login_success():login.quit_login()

test_product.py

#!/usr/bin/env python3
# -*- coding:utf-8 -*-
import pytest
import allure
from random import randint
from tools.times import sleep
from page_object.loginpage import LoginPage
from page_object.productpage import ProductPage@allure.feature("测试产品模块")
class TestProduct:@pytest.fixture(scope='class', autouse=True)def is_login(self, request, drivers):login = LoginPage(drivers)login.username('admin')login.password('Admin123456')login.submit()sleep(3)def logout():login.quit_login()request.addfinalizer(logout)@allure.story("测试添加产品")def test_001(self, drivers):"""搜索"""product = ProductPage(drivers)product.click_product()product.add_product()name, code = randint(100, 999), randint(100, 999)product.add_product_content(name, code)product.save_product()sleep(3)product.click_product()assert str(name) in product.product_list()if __name__ == '__main__':pytest.main(['TestCase/test_aproduct.py'])

测试结果

登录之后的测试用例:

图片

测试登录的用例

图片

最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走! 

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java---认识异常(详解)

还有大家来到权权的博客~欢迎大家对我的博客提出意见哦&#xff0c;有错误会及时改进的~点击进入我的博客主页 目录 一、异常的概念及体系结构1.1 异常的概念1.2 异常的体系结构1.3异常的分类 二、异常的处理2.1防御式编程2.2 异常的抛出2.3 异常的捕获2.3.1异常声明throws2.3.…

鸿蒙多线程开发——并发模型对比(Actor与内存共享)

1、概 述 并发是指在同一时间段内&#xff0c;能够处理多个任务的能力。为了提升应用的响应速度与帧率&#xff0c;以及防止耗时任务对主线程的干扰&#xff0c;HarmonyOS系统提供了异步并发和多线程并发两种处理策略。 异步并发&#xff1a;指异步代码在执行到一定程度后会被…

Axure是什么软件?全方位解读助力设计入门

在产品设计和开发领域&#xff0c;Axure是一款大名鼎鼎且功能强大的软件&#xff0c;它为专业人士和团队提供了卓越的设计支持&#xff0c;帮助他们将创意转化为实际可操作的产品原型。 一、Axure 的基本介绍 Axure是一款专业的原型设计工具&#xff0c;主要用于创建交互式的…

客户手机号收集小程序有什么用

客户手机号收集小程序具有多方面的重要作用&#xff0c;主要体现在以下几个领域&#xff1a; 商业营销与客户关系管理 精准营销&#xff1a;通过收集客户手机号&#xff0c;企业能够依据客户的消费行为、偏好等信息&#xff0c;进行精准的个性化营销。例如&#xff0c;电商企业…

Spring Boot集成SQL Server快速入门Demo

1.什么是SQL Server&#xff1f; SQL Server是由Microsoft开发和推广的以客户/服务器&#xff08;c/s&#xff09;模式访问、使用Transact-SQL语言的关系数据库管理系统&#xff08;DBMS&#xff09;&#xff0c;它最初是由Microsoft、Sybase和Ashton-Tate三家公司共同开发的&…

[CKS] Create/Read/Mount a Secret in K8S

最近准备花一周的时间准备CKS考试&#xff0c;在准备考试中发现有一个题目关于读取、创建以及挂载secret的题目。 ​ 专栏其他文章: [CKS] Create/Read/Mount a Secret in K8S-CSDN博客[CKS] Audit Log Policy-CSDN博客 -[CKS] 利用falco进行容器日志捕捉和安全监控-CSDN博客[C…

深入理解Java虚拟机:你真的了解JVM吗?

Java虚拟机(JVM) 是 Java 技术的核心,它帮助 Java 实现了一次编译,到处运行的梦想。然而,你真的理解 JVM 的工作原理吗?今天,我们就从 JVM 的内部架构、垃圾回收机制、性能调优等角度,深入探讨这个“神秘黑盒”。 1. JVM 的基本架构:探索虚拟机内部 JVM 是运行 Java …

大模型就业收入高吗?大模型入门到精通,收藏这篇就够了

目前&#xff0c;已经可以说人工智能&#xff08;AI&#xff09;是推动社会进步和产业升级的重要力量。 其中&#xff0c;AI大模型作为人工智能领域的核心技术之一&#xff0c;正引领着新一轮的技术革命。 2024年&#xff0c;AI大模型开发工程师无疑成为了IT行业中最炙手可热…

el-table 纵向垂直表头处理

项目中表格展示会遇到需要纵向垂直表头情况&#xff0c;下面&#xff0c;我们基于el-table组件来实现这种表格。 以下是这次需要用到的数据表格&#xff0c;已知左侧违章名称是固定的&#xff0c;而月份是不固定的&#xff0c;在后端返回数据格式已确定的情况下&#xff0c;需…

leetcode day10 动态规划篇 64+139

64 最小路径和 给定一个包含非负整数的 m x n 网格 grid &#xff0c;请找出一条从左上角到右下角的路径&#xff0c;使得路径上的数字总和为最小。 说明&#xff1a;每次只能向下或者向右移动一步。 m grid.lengthn grid[i].length1 < m, n < 2000 < grid[i][j]…

Flutter 小技巧之 Shader 实现酷炫的粒子动画

在之前的《不一样的思路实现炫酷 3D 翻页折叠动画》我们其实介绍过&#xff1a;如何使用 Shader 去实现一个 3D 的翻页效果&#xff0c;具体就是使用 Flutter 在 3.7 开始提供 Fragment Shader API &#xff0c;因为每个像素都会过 Fragment Shader &#xff0c;所以我们可以通…

SpringCloud框架学习(第二部分:Consul、LoadBalancer和openFeign)

目录 六、Consul服务注册和发现 1.基本介绍 2.下载运行 3.服务注册与发现 &#xff08;1&#xff09;支付服务provider8001注册进consul &#xff08;2&#xff09;修改订单服务cloud-consumer-order80 4.CAP &#xff08;1&#xff09;CAP理论 &#xff08;2&#x…

大数据学习12之HBase

1.基本概念 1.1简介 Apache HBase&#xff08;Hadoop DataBase&#xff09;是一个开源的、高可靠性、高性能、面向列&#xff08;这里指列族&#xff0c;非列式存储&#xff09;、可伸缩、实时读写的分布式数据库&#xff0c;其设计思想来源于 Google 的 BigTable 论文。利用 …

el-input 正则表达式校验输入框不能输入汉字

<el-form :model"data1" :rules"rules" ref"ruleForm" label-width"210px" class"demo-ruleForm"><el-form-item label"锯路&#xff1a;" prop"sawKref"><el-input class"inptWid…

linux rocky 9.4部署和管理docker harbor私有源

文章目录 Harbor简介安装Harbor技术细节1.安装系统(略),设置主机名和IP2.安装docker3.安装docker-compose4.安装Harbor私有源仓库5 测试登录1.本机登录2.客户端登录Harbor服务器配置docker源1. 下载镜像2.把镜像上传到Harbor私有仓库源3.客户端下载镜像,并且启动容器linux …

03WIFI与蓝牙1——基于全志V3S的Linux开发板教程笔记

1. Kernel支持 1&#xff09;配置 终端输入&#xff1a; make menuconfig使能如下部分&#xff1a; 2&#xff09;编译 保存并退出后编译内核&#xff1a; make licheepi_zero_defconfig make menuconfig #配置内核&#xff0c;有需要的话配置 make -j16 make -j16 modu…

iOS 18.2 重磅更新:6个大动作

根据外媒报道&#xff0c;iOS 18.2迎来重磅更新&#xff0c;将带来6个大动作&#xff0c;这是一次非常实用的更新。不过要注意的是&#xff0c;其中涉及到AI的功能&#xff0c;国行iPhone 暂时还不可用&#xff0c;只能等审核通过了。 1&#xff0c;Safari下载进度 过去通过S…

【单例模式】饿汉式与懒汉式以及线程安全

1. 单例模式介绍 饿汉式单例模式&#xff1a;还没有获取实例对象&#xff0c;实例对象就已经产生了。一定是线程安全的。 懒汉式单例模式&#xff1a;需要用的时候再构造实例。 应用场景&#xff1a;比如日志模块&#xff0c;数据库模块&#xff0c;开发的解析器模块。 2.饿…

一文了解 Tableau 2024.3 如何展现已发布数据源的数据模型

通过减少数据源的重复建设&#xff0c;提高数据透明度&#xff0c;可组合数据源将为企业带来更高的数据利用效率 在 TC24 用户大会上&#xff0c;Tableau 产品团队提出了一个非常重要的功能概念——可组合的数据源。这意味着你将很快能够对 Tableau 已发布的数据源进行连接、关…

Unity 网格模型及优化

一个模型中可以包含很多网格&#xff0c;一个模型可以由多个网格组成。在Unity3D中一个网格可以由多个子网格&#xff08;Sub-Mesh)组成。 在渲染引擎的时候&#xff0c;每个子网格都要匹配一个材质球来做渲染&#xff0c;实际上一个子网格本身就是一个个普通的模型&#xff0…