UI自动化测试示例:python+pytest+selenium+allure

重点应用是封装、参数化:

比如在lib文件夹下,要存储封装好的方法和必要的环境变量(指网址等)

1.cfg.py:封装网址和对应的页面

SMP_ADDRESS = 'http://127.0.0.1:8234'SMP_URL_LOGIN        = f'{SMP_ADDRESS}/login.html'
SMP_URL_DEVICE_MODEL = f'{SMP_ADDRESS}/index.html#/devicemodel'
SMP_URL_SERVICE_RULE = f'{SMP_ADDRESS}/index.html#/svcrule'

二、登录部分 

2.1 lib.py-登录模块的封装

主要就是输入账号名和密码进行测试。 

from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import Select
import time
from lib.cfg import *class SMP_UI:def __init__(self):options = webdriver.ChromeOptions()#排除一些不需要的日志输出。options.add_experimental_option( 'excludeSwitches', ['enable-logging'])self.wd = webdriver.Chrome(options=options)self.wd.implicitly_wait(3) #隐式等待时间为5秒,等待元素加载完成#登录模块测试def login(self, username, password):self.wd.get(SMP_URL_LOGIN) #打开登录网址#time.sleep(1)#sleep方便观察if username is not None:self.wd.find_element(By.ID, 'username').send_keys(username)if password is not None:self.wd.find_element(By.ID, 'password').send_keys(password)self.wd.find_element(By.ID, 'loginBtn').click()#点击登录按钮smpUI = SMP_UI()

2.2自动化测试脚本代码-登录

from selenium.webdriver.common.by import By
import time,pytest
from lib.webUI_smp import smpUIdef test_SMP_login_001():#正确账号和密码smpUI.login('byhy','sdfsdf' )nav = smpUI.wd.find_elements(By.TAG_NAME, 'nav') #登陆后就可以查看nav了,有信息代表登陆成功assert nav != []@pytest.fixture
def clearAlert():yieldtry:smpUI.wd.switch_to.alert.accept()except Exception as e:print(e)@pytest.mark.parametrize('username, password, expectedalert', [(None, 'sdfsdf', '请输入用户名'),#空用户名('byhy', None, '请输入密码'),#空密码('byhy', 'sdfsd',   '登录失败: 用户名或者密码错误'),#少输入密码('byhy', 'sdfsdff', '登录失败: 用户名或者密码错误'),#多输入密码('byh', 'sdfsdf',   '登录失败: 用户名不存在'),#少输入用户名('byhyy', 'sdfsdf', '登录失败: 用户名不存在'),#多输入用户名])
def test_SMP_login_002(username, password, expectedalert, clearAlert):smpUI.login(username, password)smpUI.wd.implicitly_wait(5)#隐式等待alert = smpUI.wd.switch_to.alert.textassert alert == expectedalert #对弹出框进行断言对比

三、商品部分

3.1 lib.py-商品服务模块的封装

主要就是添加商品、删除商品等,输出第一页的第一条的值(检查返回值) 

    #添加设备模块检测def add_device_model(self, devType, name, desc):# 实例化选择框做为对象select = Select(smpUI.wd.find_element(By.ID, "device-type"))# 选择对应列表中的种类select.select_by_visible_text(devType)#清除填充框后填-设备型号ele = smpUI.wd.find_element(By.ID, 'device-model')ele.clear()ele.send_keys(name)#清除填充框后填-型号描述ele = smpUI.wd.find_element(By.ID, 'device-model-desc')ele.clear()ele.send_keys(desc)#点提交smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-submit-btn-div .btn').click()# time.sleep(1)#得到第一页的设备信息(注:一页五条),是显示第一页的第一个数据def get_first_page_device_models(self):time.sleep(1)self.wd.implicitly_wait(0)values = self.wd.find_elements(By.CSS_SELECTOR,'.field-value')self.wd.implicitly_wait(10)deviceModels = []for idx, value in enumerate(values):if (idx+1) % 3 == 0:deviceModels.append([values[idx-2].text, values[idx-1].text, values[idx].text])return deviceModels#删除第一个项目def del_first_item(self) -> bool:self.wd.implicitly_wait(0)delBtn = self.wd.find_elements(By.CSS_SELECTOR,'.result-list-item:first-child .result-list-item-btn-bar span:first-child')self.wd.implicitly_wait(10)if not delBtn:return FalsedelBtn[0].click()self.wd.switch_to.alert.accept()return True

3.2自动化测试脚本代码-商品服务

之间登录商品增删改查的界面。注意编完数据后要删除

from selenium.webdriver.common.by import By
import time,pytest
from lib.webUI_smp import smpUI
from selenium.webdriver.support.ui import Select
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from lib.cfg import *@pytest.fixture(scope='module')
def inDeviceModelMgr():print('inDeviceModelMgr setup')smpUI.login('byhy','sdfsdf' ) #直接登录smpUI.wd.get(SMP_URL_DEVICE_MODEL)#登录后转到SMP_URL_DEVICE_MODEL这个主界面yield@pytest.fixture()
def delAddedDeviceModel():yieldprint('删除添加的设备型号')smpUI.del_first_item()@pytest.mark.parametrize('type, device, desc', [("存储柜", 'hebut-cn-box-01', '河北工业大学菜鸟柜子01柜-10大20中40小'),#添加设备-存储柜测试("存储柜", '河'*100, '河北工业大学菜鸟柜子02柜-10大20中40小'),#测试名称最大长度
("电瓶车充电站", 'hebut-dpccdz-es-01', '河北工业大学北辰充电站01站-220v电瓶车充电站'),#添加设备-电瓶车充电站
("洗车站", 'hebut-xcz-washcar-01', '河北工业大学洗车站东站'),#添加设备-洗车站测试
("汽车充电站", 'hebut-powercar-01', '河北工业大学7千瓦汽车充电站01'),#添加设备-汽车充电站])
#添加设备-存储柜测试
def test_SMP_device_model_001(type,device,desc,inDeviceModelMgr, delAddedDeviceModel):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_device_model(type,device,desc)dms = smpUI.get_first_page_device_models()assert  dms == [[type,device,desc    ]]smpUI.wd.implicitly_wait(3)def test_SMP_device_model_501(inDeviceModelMgr, delAddedDeviceModel):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_device_model("存储柜", 'daixiugai-01', '待修改测试数据-01')smpUI.wd.implicitly_wait(3)# 点击修改按钮changeBtn = smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[2]/span[2]').click()# if changeBtn.text == '修改':#     changeBtn.click()# 实例化选择框做为对象select = Select(smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[1]/div[1]/select'))# 选择对应列表中的种类select.select_by_visible_text("洗车站")# 清除填充框后填-设备型号ele = smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[1]/div[2]/input')ele.clear()ele.send_keys('修改后的型号1')# 清除填充框后填-型号描述ele = smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[1]/div[3]/input')ele.clear()ele.send_keys('修改后的型号描述')# 点提交smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[2]/span[1]').click()dms = smpUI.get_first_page_device_models()assert  dms == [["洗车站",'修改后的型号1','修改后的型号描述' ]]smpUI.wd.implicitly_wait(3)def test_SMP_device_model_601(inDeviceModelMgr, delAddedDeviceModel):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_device_model("存储柜", 'daixshanchu-01', '待删除测试数据-01')time.sleep(1)smpUI.del_first_item()dms = smpUI.get_first_page_device_models()assert ['存储柜', 'daixshanchu-01', '待删除测试数据-01'] not in dmssmpUI.wd.implicitly_wait(3)

四、业务规则部分

4.1 lib.py-业务规则模块的封装

    def add_svc_rule(self, ruleName:str, ruleType:str,minFee:str,estFee:str, feeRate=None, desc:str=None):"""添加业务规则:param ruleName: 规则名称:param ruleType: 规则类型,只能是:后付费-上报业务量、预付费-下发费用、预付费-下发业务量:param minFee: 最小费用,不需要时,填写空字符串:param estFee: 预计费用,不需要时,填写空字符串:param desc: 描述:param feeRate: 费率, 如果ruleType是预付费-下发费用: 不用填写预付费-下发业务量: 格式为 ['千瓦时', '1'], 元素分别是 单位、单价后付费-上报业务量: 格式为 [['10L', '小时','1'],['20L', '小时','2'],], 每个元素里面分别是 : 业务码、单位、单价:return:"""#名称输入ele = self.wd.find_element(By.CSS_SELECTOR, '.add-one-form > .field:nth-child(1) >input')ele.clear()ele.send_keys(ruleName)select = Select(self.wd.find_element(By.CSS_SELECTOR, ".add-one-form select"))select.select_by_visible_text(ruleType)if ruleType != '后付费-上报业务量': #预付费-下发业务量操作ele = self.wd.find_element(By.CSS_SELECTOR, '.add-one-form > .field:nth-child(3) >input')ele.clear()if minFee:ele.send_keys(minFee)#输入最小消费ele =  self.wd.find_element(By.CSS_SELECTOR, '.add-one-form > .field:nth-child(4) >input')ele.clear()if estFee:ele.send_keys(estFee)#输入预估消费# 三者都有描述, 用xpath而不用 .field:nth-child 因为后付费-上报业务量 次序会变if desc:ele =  self.wd.find_element(By.XPATH,"//*[@class='add-one-submit-btn-div']/preceding-sibling::*[1]/input")ele.clear()ele.send_keys(desc)#输入对应描述# 费率填写if ruleType == '预付费-下发费用':# 没有费率设置passelif ruleType == '后付费-上报业务量':# 先删除上次添加遗留的费率self.wd.implicitly_wait(3)  # 先修改短等待时间while True:eles = self.wd.find_elements(By.CSS_SELECTOR, '.fee-rate span:last-child')if eles:eles[0].click() #删除之前的费率time.sleep(0.5)else:breakself.wd.implicitly_wait(3)  # 再改回来for one in feeRate:#遍历每个费率块self.wd.find_element(By.CSS_SELECTOR,'.fee-rate-list button').click()#锚定整个框entry = self.wd.find_element(By.CSS_SELECTOR,'div.fee-rate:nth-last-child(2)')# 输入业务码entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(1)').send_keys(one[0])# 输入计费单位entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(2)').send_keys(one[1])# 输入单位价格entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(3)').send_keys(one[2])elif ruleType == '预付费-下发业务量':#锚定整个框entry = self.wd.find_element(By.CSS_SELECTOR, 'div.fee-rate')# 输入计费单位entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(1)').send_keys(feeRate[0])# 输入单位价格entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(2)').send_keys(feeRate[1])else:raise Exception('ruleType 参数值错误')# 确定提交self.wd.find_element(By.CSS_SELECTOR, '.add-one-submit-btn-div .btn').click()self.wd.implicitly_wait(3)#业务部分查询首页def get_first_page_svc_rules(self):time.sleep(1)self.wd.implicitly_wait(0)items = self.wd.find_elements(By.CSS_SELECTOR,'.result-list-item-info')self.wd.implicitly_wait(10)rules = []for item in items:nameValueList = item.find_elements(By.CSS_SELECTOR, '.field>.field-name, .field>.field-value')itemInfo = []for idx, _ in enumerate(nameValueList):if (idx+1) % 2 == 0:nameEle,valueEle = nameValueList[idx-1], nameValueList[idx]if nameEle.text == '规则内容':ruleContent = {}sfns = valueEle.find_elements(By.CSS_SELECTOR,'.sub-field-name')for sfnEle in sfns:sfn = sfnEle.textsfvEle = sfnEle.find_element(By.XPATH, "following-sibling::*[1]")if sfn == '费率':ruleContent[sfn] = sfvEle.textelse:ruleContent[sfn] = sfvEle.textitemInfo.append(ruleContent)else:itemInfo.append(valueEle.text)rules.append(itemInfo)return rules#删除第一个项目def del_first_item(self) -> bool:self.wd.implicitly_wait(0)delBtn = self.wd.find_elements(By.CSS_SELECTOR,'.result-list-item:first-child .result-list-item-btn-bar span:first-child')self.wd.implicitly_wait(10)if not delBtn:return FalsedelBtn[0].click()self.wd.switch_to.alert.accept()return True

4.2自动化测试脚本代码-业务规则

注意事项跟3.2的商品的自动化测试脚本类似

from selenium.webdriver.common.by import By
import time,pytest
from lib.webUI_smp import smpUI
from lib.cfg import *@pytest.fixture(scope='module')
def inServiceRuleMgr():print('** inServiceRuleMgr setup **')smpUI.login('byhy','sdfsdf' )smpUI.wd.get(SMP_URL_SERVICE_RULE)yield@pytest.fixture()
def delAddedServiceRule():yieldprint('** 删除添加的业务规则')smpUI.del_first_item()@pytest.mark.parametrize('svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv', [("全国-电瓶车充电费率1", "预付费-下发业务量", "0.1","2","全国-电瓶车充电费率1的描述",['千瓦时', '1'],None),#添加预付费-下发业务量])
def test_SMP_service_rule_001(svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv,inServiceRuleMgr, delAddedServiceRule):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR,'.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_svc_rule(svcname, svctype,paymin,pay, jifeidanwei,svcdesc)dms = smpUI.get_first_page_svc_rules()assert dms[0][:3] == [svcname, svctype, {'最小消费': paymin, '预估消费': pay, '费率': f'单位:{jifeidanwei[0]} \n单价:{jifeidanwei[1]}'}]@pytest.mark.parametrize('svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv', [("南京-洗车机费率1", "预付费-下发费用", "2","10","南京-洗车机费率1的描述",None,None),#预付费-下发费用])
def test_SMP_service_rule_101(svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv,inServiceRuleMgr, delAddedServiceRule):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR,'.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_svc_rule(svcname, svctype,paymin,pay,svcdesc)dms = smpUI.get_first_page_svc_rules()assert dms[0][:3] == [svcname, svctype, {'最小消费': paymin, '预估消费': pay}]@pytest.mark.parametrize('svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv', [("南京-存储柜费率1", "预付费-下发业务量", None,None,None,None,[['100L','小时','2']]),#添加预付费-下发业务量,['50L','小时','1'],['10L','小时','0.5']])
def test_SMP_service_rule_201(svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv,inServiceRuleMgr, delAddedServiceRule):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR,'.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_svc_rule(svcname, svctype, paymin, pay,  svcdesc, feilv)dms = smpUI.get_first_page_svc_rules()assert dms[0][:3] == [svcname, svctype, {'业务码': f'单位:{feilv[0]} \n计费单位:{feilv[1]} \n单位价格:{feilv[2]}'}]

还有部分没粘上去,万变不离其宗,还要接口自动化测试,这里先不写,下次再写。 

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

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

相关文章

【微服务】springboot远程docker进行debug调试使用详解

目录 一、前言 二、线上问题常用解决方案 2.1 微服务线上运行中常见的问题 2.2 微服务线上问题解决方案 2.3 远程debug概述 2.3.1 远程debug原理 2.3.2 远程debug优势 三、实验环境准备 3.1 搭建springboot工程 3.1.1 工程结构 3.1.2 引入基础依赖 3.1.3 添加配置文…

400行程序写一个实时操作系统(九):替换FreeRTOS的内存管理算法

前言 通过前面几章,笔者带领大家完成了内存管理算法的编写。 我们完成的内存管理算法,被称为小内存管理算法。我们也可以将它作为一个库,在后续的嵌入式开发中,使用我们自己编写的malloc,不仅效率会更高,…

机器学习笔记-2

文章目录 一、Linear model二、How to represent this function三、Function with unknown parameter四、ReLU总结、A fancy name 一、Linear model 线性模型过于简单,有很大限制,我们需要更多复杂模式 蓝色是线性模型,线性模型无法去表示…

如何匿名浏览网站,保护在线隐私?

在现如今的网络世界,在线隐私已不复存在。你总是被跟踪,即使你使用隐身模式也无济于事。隐身模式会阻止浏览器保存你的浏览历史记录。但它并不能阻止你的互联网服务提供商 (ISP)、雇主、学校、图书馆或你访问的网站看到你在网上做什么。 更有不法分子在未…

Lumerical学习——资源管理和运行模拟

一、资源管理(Resource Manager) 在模拟计算前必须对计算资源进行配置。采用资源管理器可以完成这项任务。单击主工具条的“资源(Resources)”按钮(见上图)就可以打开资源管理器。通常每个计算机只需设置一…

大型生物制药企业如何实现安全又高效地跨网域数据传输?

大型生物制药企业由于组织结构庞大、业务覆盖研发、生产及销售,因此内部会再细分为多个管理单位,包括研发部门、生产部门、质量控制部门、供应链管理部门及营销部和日常业务支撑部门等。在物理区域划分上,大型生物制药企业会设立实验室、研发…

摇人摇人, JD内推岗位(社招+校招)

摇人摇人, 有找工作的家人们看过来啊~ 虚位以待, 快到碗里来 算法开发工程师岗 京东云 北京|T7, 5-10年 岗位职责: 参与基于RAG知识库平台和ChatBI产品打造和商业化落地,进行相关技术:包括OCR、文档拆分、意图理解、多轮对话、NL2SQL、Embed…

mysql用户管理(user表列信息介绍,本质,管理操作),数据库的权限管理(权限列表,权限操作)

目录 用户管理 介绍 user表 介绍 列信息 Host User *_priv authentication_string 用户管理的本质 操作 创建用户 删除用户 修改用户信息 修改密码 自己修改 root用户修改指定用户的密码 数据库的权限 权限列表 给用户授权 查看权限 回收权限 刷新权限 …

Linux性能调优,还可以从这些方面入手

linux是目前最常用的操作系统,下面是一些常见的 Linux 系统调优技巧,在进行系统调优时,需要根据具体的系统负载和应用需求进行调整,并进行充分的测试和监控,以确保系统的稳定性和性能。同时,调优过程中要谨…

万界星空科技:智能称重打标系统

万界星空科技的称重系统是其为制造业,特别是线缆、漆包线、食品等行业提供的重要解决方案之一。以下是对该系统的详细介绍: 一、系统概述 万界星空科技称重系统是集成在其MES(制造执行系统)中的一个功能模块,专门用于…

基于springboot实习管理系统

作者:计算机学长阿伟 开发技术:SpringBoot、SSM、Vue、MySQL、ElementUI等,“文末源码”。 系统展示 【2024最新】基于JavaSpringBootVueMySQL的,前后端分离。 开发语言:Java数据库:MySQL技术:…

React Agent 自定义实现

目录 背景 langchin 中的 agent langchin 中 agent 的问题 langchain 的 agent 案例 自定义 React Agent 大模型 工具定义 问题设定 问题改写,挖掘潜在意图 React Prompt 下一步规划 问题总结 代码 背景 之前使用过 langchian 中的 agent 去实现过一些…

2020年计算机网络408真题解析

第一题: 解析:OSI参考模型网络协议的三要素 网络协议的三要素:语法 ,语义,同步(时序) 语法:定义收发双方所交换信息的格式 语法:定义收发双方所要完成的操作 网页的加载 …

深入理解队列(Queue)的实现(纯小白进)

目录: 前言一、 什么是队列?1.1、 队列的特性1.2、 队列的图解 二、 队列的详细实现2.1、 队列不同的实现方式2.2、 队列结构体2.3、 队列的初始化2.4、 入队列2.5、 出队列2.6、 获取对头元素2.7、 获取队尾元素2.8、 队列的判空2.9、 队列有效的元素个数2.10、 队…

Kind部署的K8s证书过期后的解决方案

证书通常有效期为1年,一年后服务将不可用解决方案就是更新证书 1. 找到 Kind 集群的控制平面容器名称,容器名称不一定是这个 docker ps --filter "namekind-control-plane"2. 进入 Kind 控制平面的容器: docker exec -it kind-control-plane…

洛谷入门刷题Day5(想刷水题结果被水题刷了)

P1304 哥德巴赫猜想 题目描述 输入一个偶数 N N N,验证 4 ∼ N 4\sim N 4∼N 所有偶数是否符合哥德巴赫猜想:任一大于 2 2 2 的偶数都可写成两个质数之和。如果一个数不止一种分法,则输出第一个加数相比其他分法最小的方案。例如 10 10…

论文笔记:Ontology-enhanced Prompt-tuning for Few-shot Learning

论文来源:WWW 2022 论文地址:https://arxiv.org/pdf/2201.11332.pdfhttps://arxiv.org/pdf/2201.11332.pdf 论文代码:暂未公开 笔记仅供参考,撰写不易,请勿恶意转载抄袭! Abstract 小样本学习旨在基于…

Linux——传输层协议

目录 一再谈端口号 1端口号范围划分 2两个问题 3理解进程与端口号的关系 二UDP协议 1格式 2特点 3进一步理解 3.1关于UDP报头 3.2关于报文 4基于UDP的应用层协议 三TCP协议 1格式 2TCP基本通信 2.1关于可靠性 2.2TCP通信模式 3超时重传 4连接管理 4.1建立…

数据挖掘学习笔记:朴素贝叶斯 | Python复现

数据挖掘学习笔记:朴素贝叶斯 机器学习系列(四):朴素贝叶斯(华强买瓜版) - yyxy的文章 - 知乎 十分钟,让你再也忘不掉贝叶斯分类 - VoidHaruhi的文章 - 知乎 《机器学习》(西瓜书&am…

基于Arduino的简易收音机

DIY FM收音机:使用Arduino和Si4703模块打造 引言 在本项目中,我们将使用Arduino Nano和Si4703 FM调谐模块来构建一个功能完备的FM收音机接收器。这个易于跟随的指南非常适合想要深入无线电频率和无线通信世界的业余爱好者和电子爱好者。 Si4703模块是…