使用pytest+selenium+allure实现web页面自动化测试

测试文件

  • base 基本方法
  • data 测试数据
  • page web页面相关操作
  • image 测试截图
  • log 日志文件
  • report 测试报告文件
  • temp 临时文件
  • tool 文件读取,发邮件文件
  • TestCases 测试用例

在page下的__init__.py文件下配置

import os
import time
from selenium.webdriver.common.by import By# 项目路径
PROJECT_PATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
# 图片路径
IMAGE_PATH = os.path.join(PROJECT_PATH, 'image')
# 测试数据路径
DATA_PATH = os.path.join(PROJECT_PATH, 'data')
# 日志路径
LOG_PATH = os.path.join(PROJECT_PATH, 'log')
# 测试报告
REPORT_PATH = os.path.join(PROJECT_PATH, 'report', 'html')
# 临时文件
TEMP_PATH = os.path.join(PROJECT_PATH, 'temp')
# 请求url
url = "http://cal.apple886.com/""""  邮件信息  """
email_host = "smtp.qq.com"  # SMTP服务器
email_sender = '2022204437@qq.com'  # 发件人
email_license = 'jxyvidactsaiicja'  # 授权码
email_receivers = ['2022204437@qq.com']  # 收件人(可以群发)""" 以下为测试数据配置文件 """
image_path = os.path.join(IMAGE_PATH, '%serror.png' % time.strftime("%Y%m%d%H%M%S"))"""  以下为测试数据配置文件   """
add_data = os.path.join(DATA_PATH, 'add.json')
multi_data = os.path.join(DATA_PATH, 'multi.json')
subtr_data = os.path.join(DATA_PATH, 'subtr.json')"""  以下为logger配置文件   """
log_filename = os.path.join(LOG_PATH, '%sdemo.log' % time.strftime("%Y%m%d%H%M%S"))
log_when = "midnight"
log_interval = 1
log_backupCount = 30
log_encoding = "utf-8"
log_filemode = "w"
log_format = "%(asctime)s -[%(name)s]- %(levelname)s---[%(filename)s [%(funcName)s:%(lineno)s]] - %(message)s"
log_datefmt = "%Y-%m-%d %H:%M:%S"""""   以下为计算机配置数据   """
computer_add = By.XPATH, '//*[@id="simpleAdd"]'
computer_subtr = By.XPATH, '//*[@id="simpleSubtr"]'
computer_multi = By.XPATH, '//*[@id="simpleMulti"]'
computer_divi = By.XPATH, '//*[@id="simpleDivi"]'
computer_clear = By.XPATH, '//*[@id="simpleClearAllBtn"]'
computer_equal = By.XPATH, '//*[@id="simpleEqual"]'
computer_value = By.XPATH, '//*[@id="resultIpt"]'

在base下创建一个webpage.py文件

from selenium.webdriver.support.wait import WebDriverWait
from web.counter.base.logger import GetLogger
from web.counter import pagelog = GetLogger().get_logger()class WebPage:def __init__(self, driver):log.info("初始化driver{}".format(driver))self.driver = driver# 查找元素def base_find_element(self, loc, timeout=30, poll_frequency=0.5):log.info("正在查找元素{}".format(loc))return WebDriverWait(self.driver, timeout=timeout, poll_frequency=poll_frequency).until(lambda x: x.find_element(*loc))# 点击def base_click(self, loc):log.info("点击元素{}".format(loc))self.base_find_element(loc).click()# 输入def base_input(self, loc, key):log.info("输入元素{}输入值为{}".format(loc,key))self.base_find_element(loc).clear()self.base_find_element(loc).send_keys(key)# 获取文本def base_get_test(self, loc):log.info("获取文本{}".format(loc))return self.base_find_element(loc).text# 获取属性值def base_get_value(self, loc):log.info("获取{}属性值".format(loc))return self.base_find_element(loc).get_attribute('value')# 截图def base_get_image(self):log.error("错误!!!错误!!!错误!!!")self.driver.get_screenshot_as_file(page.image_path)

在base下创建一个driver.py文件

from selenium import webdriver
from web.counter import pageclass GetDriver:driver = None@classmethoddef get_driver(cls):if cls.driver is None:cls.driver = webdriver.Edge()cls.driver.maximize_window()cls.driver.get(page.url)return cls.driver@classmethoddef quit_driver(cls):if cls.driver:cls.driver.quit()cls.driver = None

在base下创建一个logger.py文件

import logging.handlers
from web.counter import pageclass GetLogger:logger = None@classmethoddef get_logger(cls):if cls.logger is None:# 获取日志器cls.logger = logging.getLogger()# 获取日志器级别cls.logger.setLevel(logging.INFO)# 获取处理器 控制台sh = logging.StreamHandler()# 获取处理器 文件-以以时间分隔th = logging.handlers.TimedRotatingFileHandler(filename=page.log_filename,when=page.log_when,interval=page.log_interval,backupCount=page.log_backupCount,encoding=page.log_encoding)fm = logging.Formatter(page.log_format)sh.setFormatter(fm)th.setFormatter(fm)cls.logger.addHandler(th)cls.logger.addHandler(sh)return cls.logger

在page下创建一个add_page.py文件

from selenium.webdriver.common.by import Byfrom web.counter.base.webpage import WebPage
from web.counter import pageclass AddPage(WebPage):# 点击数字def page_click_num(self, num):for n in str(num):loc = By.XPATH, '//*[@id="simple{}"]'.format(n)self.base_click(loc)# 点击加号def page_click_add(self):self.base_click(page.computer_add)# 点击等于def page_click_equal(self):self.base_click(page.computer_equal)# 获取结果def page_get_value(self):return self.base_get_value(page.computer_value)# 清屏def page_click_clear(self):self.base_click(page.computer_clear)# 截图def page_get_image(self):self.base_get_image()# 组装加法def page_add(self, a, b):self.page_click_num(a)self.page_click_add()self.page_click_num(b)self.page_click_equal()

在TestCases 下创建一个test_add.py文件

from time import sleep
import pytest
from web.counter.page.add_page import AddPage
from web.counter.base.driver import GetDriver
from web.counter.tool.read_json import GetJson
from web.counter.base.logger import GetLogger
from web.counter import pagelog = GetLogger().loggerclass TestAdd:def setup_class(self):self.computer = AddPage(GetDriver().get_driver())def teardown_class(self):GetDriver().quit_driver()@pytest.mark.parametrize("a, b, result", GetJson().get_data(page.add_data))def test_add_computer(self, a, b, result):self.computer.page_click_clear()self.computer.page_add(a, b)try:assert self.computer.page_get_value() == str(result)except:log.error("错误")sleep(1)self.computer.page_get_image()raise

在tool下创建一个read_json.py文件

import jsonclass GetJson:data_lists = Nonedata_dic = None# 读取json文件@classmethoddef read_json(cls, json_path):with open(json_path, "r", encoding="utf-8") as f:return json.load(f)# 获取测试数据@classmethoddef get_data(cls, json_path):cls.data_dic = cls.read_json(json_path)cls.data_lists = list()for data in cls.data_dic.values():cls.data_lists.append((data.get("a"), data.get("b"), data.get("result")))return cls.data_lists

在tool下创建一个send_emails.py文件

import time
import zipfile
import os
from email.mime.application import MIMEApplication
from web.counter import page
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipartclass SendEmails:def __init__(self):self.mime = MIMEMultipart()self.allure_report = Noneself.report_path = page.REPORT_PATHself.temp_path = page.TEMP_PATHdef send_email(self):self.mime['Subject'] = '%sxxx测试报告' % time.strftime('%Y年%m月%d日 %H:%M:%S ')content = 'hello, this is email content.'textApart = MIMEText(content)zip_Apart = MIMEApplication(open(SendEmails().make_zip, 'rb').read())zip_Apart.add_header('Content-Disposition', 'attachment', filename='allure_report.zip')self.mime.attach(textApart)self.mime.attach(zip_Apart)try:server = smtplib.SMTP(page.email_host)server.login(page.email_sender, page.email_license)server.sendmail(page.email_sender, page.email_receivers, self.mime.as_string())print('发送成功')server.quit()except smtplib.SMTPException as e:print('error:', e)  # 打印错误# 打包目录为zip文件(实际未压缩)@propertydef make_zip(self):if not os.path.exists(self.temp_path):os.makedirs(self.temp_path)self.allure_report = os.path.join(self.temp_path, 'allure_report.zip')zip_file = zipfile.ZipFile(self.allure_report, 'w')for present, dirs, files in os.walk(self.report_path):"""present:现在的目录dirs:该目录下包含的子目录files:该目录下包含的文件"""for file in files:pathfile = os.path.join(present, file)relpath = os.path.relpath(pathfile, page.PROJECT_PATH)  # 相对路径# ZIP_STORED:实际上并未压缩。zip_file.write(pathfile, relpath, zipfile.ZIP_STORED)zip_file.close()return self.allure_reportif __name__ == '__main__':report = page.REPORT_PATHtemp = page.TEMP_PATHprint(report)print(temp)SendEmails().send_email()print(SendEmails().allure_report)print("压缩包创建完成")# import zmail
# from web.counter import page
#
#
# def send_report():
#     """发送报告"""
#
#     with open(page.REPORT_PATH, encoding='utf-8') as f:
#         content_html = f.read()
#     try:
#         mail = {
#             'from': '2022204437@qq.com',
#             'subject': '最新的测试报告邮件',
#             'content_html': content_html,
#             'attachments': [page.REPORT_PATH, ]
#         }
#         server = zmail.server(*page.EMAIL_INFO.values())
#         server.send_mail(page.ADDRESSEE, mail)
#         print("测试邮件发送成功!")
#     except Exception as e:
#         print("Error: 无法发送邮件,{}!", format(e))
#
#
# if __name__ == "__main__":
#     '''请先在config/conf.py文件设置QQ邮箱的账号和密码'''
#     send_report()

web自动化大体就是这样,然后测试数据可以自己写

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

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

相关文章

通过U盘:将电脑进行重装电脑

目录 一.老毛桃制作winPE镜像 1.制作准备 2.具体制作 下载老毛桃工具 插入U盘 选择制作模式 正式配置U盘 安装提醒 安装成功 具体操作 二.使用ultrasio制作U盘 1.具体思路 2.图片操作 三.硬盘安装系统 具体操作 示例图 ​编辑 一.老毛桃制作winPE镜像 1.制作准…

eNSP错误40,原因三:windows10自带虚拟化软件Hyper-V

问题描述 Hyper-V软件与VirtualBox不兼容,一旦开启Hyper-V的话eNSP的路由器就会无法开启,显示ERROR 40 原理 大家注意看hypervisor的两种类型: 左边的是开启hypervisor的Type-1,hypervisor在启用的时候,宿主机也相…

Centos7在安装Graylog时新安装MongoDB报错端口不监听服务不启动无法运行启动失败

由于虚拟机服务器上需要安装Graylog需要安装MongoDB,尝试官网下载安装包,和yum安装均无法正常启动,折腾了好几天,重装了十几次,网上搜索了很多很多资料,均无法正常运行,百度上搜索各种文档&…

Maven将Jar包打入本地仓库

Maven将Jar包打入本地仓库 Maven将Jar包打入本地仓库嘚吧嘚下载Maven配置Maven新建MAVEN_HOME编辑Path验证Maven配置 Jar包打入Maven仓库 Maven将Jar包打入本地仓库 嘚吧嘚 最近项目用到一个Jar包,不能从远程仓库拉取,只有一个Jar包,所以需…

ssm420基于JavaEE的企业人事管理信息系统的设计与实现论文

摘 要 现代经济快节奏发展以及不断完善升级的信息化技术,让传统数据信息的管理升级为软件存储,归纳,集中处理数据信息的管理方式。本企业人事管理信息系统就是在这样的大环境下诞生,其可以帮助管理者在短时间内处理完毕庞大的数据…

文件夹数据同步工具 Sync Folders Pro mac支持选项

Sync Folders Pro for Mac 是一款功能强大的文件夹同步工具,旨在帮助用户在 Mac 计算机和移动设备之间创建双向同步。这款软件支持各种文件系统和设备,如 iPhone,iPad,iPod,Android 等。通过这款软件,用户可…

软件测试面试会问哪些问题?

软件测试面试,一般会被问到下面这九类问题。 1、基础问题 2、Linux命令 3、数据库 4、功能测试 5、Python基础 6、接口测试 7、自动化测试 8、性能测试 9、人事问题 接下来,以上9类问题,我都会分别结合2个案例,附上答案&#xff0…

FL Studio21.2上市更新发布中文有版本

FL Studio 21.2 已经上市,Image-Line 正在开展黑色星期五促销活动,让更多人能够加入 FL Studio,创作出最棒的音乐! FL Studio 21 Win-安装包下载如下: https://wm.makeding.com/iclk/?zoneid55981 FL Studio 21 Mac-安装包下载…

关于时区处理策略

前端会通过 App-Id 请求头附带 客户端时区 信息 前端传入的如果是 字符串,会自动根据 请求的客户端时区 解析为对应的 日期 如果前端传入的是时间戳,则无需额外解析转换 如果是 商户后台、管理后台 都统一基于 商户所在国家的时区(总台目前…

2023 英特尔On技术创新大会直播 |AI科技创新的引路者

英特尔大会 前言英特尔人工智能英特尔创新技术基于英特尔架构的科学计算总结 前言 英特尔技术创新大会是一个令人激动和启发的盛会。在这次大会上,我有幸观看了许多令人瞩目的科技创新和前沿技术的展示。这些展示不仅展示了英特尔作为科技巨头的实力,更…

30. MVC设计模式

JavaEE 开发流程 ↓MVC的概念 MVC是Model-View-Controller的简称,即模型-视图-控制器。 MVC是一种设计模式,它把应用程序分成三个核心模块:模型、视图、控制器,它们各自处理自己的任务。 模型(model) 模型是应用程序的主体部分…

系列十二(面试)、Java中的GC回收类型有哪些?

一、Java中的GC回收类型 1.1、概述 Java中的GC回收类型主要包含以下几种,即:UseSerialGC、UseParallelGC、UseConcMarkSweepGC、UseParNewGC、UseParallelOldGC、UseG1GC。 1.2、源码