热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

测试框架之UI数据驱动、uiFrame框架

概述在实际工作中,自动化测试中使用到的部分数据改动时,需要修改的部分比较多,所以我们把这部分分离出来,就不需要一个一个修改模块中的信息(类似于Postman中的请求地址部分)一、数

概述

在实际工作中,自动化测试中使用到的部分数据改动时,需要修改的部分比较多,所以我们把这部分分离出来,就不需要一个一个修改模块中的信息(类似于Postman中的请求地址部分)


一、数据驱动

源代码展示

from selenium import webdriver
import unittest
import time
class SinaTest(unittest.TestCase):
def setUp(self) -> None:
self.driver
=webdriver.Chrome()
self.driver.maximize_window()
self.driver.get(
"https://mail.sina.com.cn/")
self.driver.implicitly_wait(
60)
def tearDown(self) -> None:
self.driver.quit()
def test_login_null(self):
self.driver.find_element_by_id(
"freename").send_keys('')
self.driver.find_element_by_id(
"freepassword").send_keys('')
self.driver.find_element_by_class_name(
"loginBtn").click()
time.sleep(
4)
divText
=self.driver.find_element_by_xpath("/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]")
# assertEqual比较两个对象是否相等==
self.assertEqual(divText.text,'请输入邮箱名')
def test_login_email_format(self):
# 登录:验证账户格式不规范的错误提示信息
self.driver.find_element_by_id("freename").send_keys("wrnryf")
self.driver.find_element_by_id(
"freepassword").send_keys("sfevg")
self.driver.find_element_by_class_name(
'loginBtn').click()
time.sleep(
4)
divText
=self.driver.find_element_by_xpath("/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]")
self.assertEqual(divText.text,
"您输入的邮箱名格式不正确")
if __name__ =="__main__":
unittest.main()

1、可以新建init模块,分离的部分为:

import unittest
from selenium import webdriver
class SinaTest(unittest.TestCase):
def setUp(self) -> None:
self.driver
=webdriver.Chrome()
self.driver.maximize_window()
self.driver.get(
"https://mail.sina.com.cn/")
self.driver.implicitly_wait(
60)
def tearDown(self) -> None:
self.driver.quit()

分离后源代码就可以继承新建的模块内容

# 新增导入库为之前的模块名称
from 测试框架.init import Init
class Sina(Init):
# 之前书写的测试用例
def test_login_null(self):

 


2、数据驱动分离

在自动化测试中,把测试中使用到的数据分离出来——JSON,YAML


2.1、设置json模式

新建相关文件名,如新浪邮箱的就可以写作:sina.json。以字典,格式:key:value的形式写入

{
"null": "请输入邮箱名",
"format": "您输入的邮箱名格式不正确"
}

在源代码中就需要导入json库,通过反序列化读取文件内容

import unittest
from selenium import webdriver
import jsondef readJson():
return json.load(open('sina.json',encoding='utf-8')) # windows中输出的内容含有汉字内容,就需要加上编码
import time
from 测试框架.init import Init
class Sina(Init):
def test_login_null(self):
self.driver.find_element_by_id(
"freename").send_keys('')
self.driver.find_element_by_id("freepassword").send_keys('')
self.driver.find_element_by_class_name(
'loginBtn').click()
time.sleep(
4)
divText
=self.driver.find_element_by_xpath("/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]")

# 断言中修改调用模式
self.assertEqual(divText.text,readJson()[
'null'])
def test_login_email_format(self):
self.driver.find_element_by_id(
"freename").send_keys("wrnryf")
self.driver.find_element_by_id(
"freepassword").send_keys("sfevg")
self.driver.find_element_by_class_name(
'loginBtn').click()
time.sleep(
4)
divText
=self.driver.find_element_by_xpath("/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]")
self.assertEqual(divText.text,readJson()[
"format"])

 

当页面输出Empty suite时,可以通过控制台卸载pytest库解决——命令为:pip3 uninstall pytest

 

 

 


2.2、yaml模式

优点:结构化明晰。

操作步骤:

1、使用前需要安装一个第三方的库:pip3 install pyyaml

2、以key:value形式书写,需要注意书写结构,冒号与value间一定加个空格,可以不用加引号

注意事项在JS中,java go语言中的null、true、false,需要写为:None、True、False

login:
null: 请输入邮箱名
format: 您输入的邮箱名格式不正确
error:
user: admin
passwd: admin

添加到源代码中就需要先调用yaml

import yaml
# with上下文读取文件,通过yaml.safe_load()读取文件
def readYmal():
with open(
"sina.yaml",encoding='utf-8') as f:
return yaml.safe_load(f)
# 因为之前书写的内容是结构化信息,所以在断言中需要按层级书写
self.assertEqual(divText.text,readYmal()['login'][None])
self.assertEqual(divText.text,readYmal()[
'login']["format"])

 


3、参数化应用

以字典形式写入,格式为key:value,value中的内容为列表形式

{
"login":
[
{
"username":"","password":"","result":"请输入邮箱名"},
{
"username":"wev","password":"wfg","result":"您输入的邮箱名格式不正确"},
{
"username":"seefg@sina.com","password":"srbv","result":"登录名或密码错误"}
]
}

import json
import unittest
from parameterized import parameterized,param
from selenium import webdriver
import time
from 测试框架.init import Init
def readJson():
return json.load(open('login_list.json',encoding='utf-8'))["login"]
# 将json格式中的列表进行循环并输出
for item in readJson():
print(item)
class Sina(Init):
# 方法一
#
参数化需要借助外部的parameterized的库中的expand方法,param指向的是每次循环的测试数据
@parameterized.expand(
[param(
"", "", "请输入邮箱名"),
param(
"asdwer", "123456", "您输入的邮箱名格式不正确"),
param(
"asd123@sina.com", "123456", "登录名或密码错误")])
# 方法二
@parameterized.expand([
param(readJson[0][
"username"],readJson[0]["password"],readJson[0]["result"]),
param(readJson[
1]["username"],readJson[1]["password"],readJson[1]["result"]),
param(readJson[
2]["username"],readJson[2]["password"],readJson[2]["result"])
])
def test_login(self,u,p,r):
# 登录:测试登录不同场景
self.driver.find_element_by_id("freename").send_keys(u)
self.driver.find_element_by_id(
"freepassword").send_keys(p)
self.driver.find_element_by_class_name(
'loginBtn').click()
time.sleep(
4)
divText
=self.driver.find_element_by_xpath("/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]")
self.assertEqual(divText.text,r)
if __name__ =="__main__":
unittest.main()

 


二、页面对象设计模式(page object)—— po模式


1、三种优势

1、创建可以跨多个测试用例共享的代码

2、减少重复代码的数量

3、如果用户界面发生了维护,我们只需要维护一个地方,这样修改及维护的成本相对而言是比较低的


2、操作创建项目

选择Attach

 

添加解释器

 

 


2、UI框架设计

base:基础层,主要编写底层定位元素的类

page:对象层,编写具体的业务逻辑,把页面每一个操作行为单独的写一个方法或者是函数

test:测试层,里面主要是测试模块

utils:工具类

common:公共类,里面编写公共使用到的方法

data:存储测试使用到测试数据

config:配置文件存储目录

report:测试报告目录


1、base基础层

from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support.expected_conditions import NoSuchElementException
import time as t
class webUI(object):
def __init__(self,driver):
self.driver=driver
# *args元组
def findElement(self,*args):
'''
单个元素定位的方式
:param args:
:return:它是一个元组,需要带上具体什么方式定位元素属性以及元素属性的值
'''
try:
return self.driver.find_element(*args)
except NoSuchElementException as e:
return e.args[0]
def findElements(self,*args,index):
'''
单个元素定位的方式
:param index:
:return:它是一个元组,需要带上具体什么方式定位元素属性以及元素属性的值
'''
try:
return self.driver.find_elements(*args)[index]
except NoSuchElementException as e:
return e.args[0]

2、page对象层

from selenium.webdriver.common.by import By
from base.base import webUI
import time as t
class Login(webUI):
username
=(By.ID,"freename")
password
=(By.ID,"freepassword")
login
=(By.CLASS_NAME,'loginBtn')
divText
=(By.XPATH,"/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[1]")
def inputUserName(self,username):
t.sleep(
3)
self.findElement(
*self.username).send_keys(username)
def inputUserPassword(self,password):
t.sleep(
3)
self.findElement(
*self.password).send_keys(password)
def clickLogin(self):
self.findElement(
*self.login).click()
t.sleep(
3)
def getDivText(self):
return self.findElement(*self.divText).text
def singLogin(self,username,password):
self.inputUserName(username
=username)
self.inputUserPassword(password
=password)
self.clickLogin()

3、test测试层

from selenium import webdriver
import unittest
from page.login import Login
import time as t
class LoginTest(unittest.TestCase,Login):
def setUp(self)->None:
self.driver
=webdriver.Chrome()
self.driver.maximize_window()
self.driver.get(
"https://mail.sina.com.cn/")
self.driver.implicitly_wait(
30)
def tearDown(self) -> None:
self.driver.quit()
def test_login_null(self):
# 登录验证:账户密码为空的错误提示信息
self.singLogin(username="",password='')
self.assertEqual(self.getDivText(),
"请输入邮箱名")
def test_login_format(self):
# 登录验证:邮箱名格式不正确
self.singLogin(username="sedsgnr",password='dgbfg')
self.assertEqual(self.getDivText(),
"您输入的邮箱名格式不正确")
def test_login_error(self):
# 登录验证:账户密码不匹配
self.singLogin(username="sfrb@sina.com",password='werh')
self.assertEqual(self.getDivText(),
"登录名或密码错误")

 


练习

def test_login_(self):
# 登录验证:账户正确输入,但密码不匹配
self.singLogin(username="wp2256520569@sina.com", password='werh')
self.assertEqual(self.getDivText(),
"登录名或密码错误")
def test_login_abnormal(self):
self.singLogin(username
="18292091170", password='pan.wang12345')
self.assertEqual(self.getDivText(),
"抱歉!登录失败,请稍候再试")
def test_login_yl(self):
self.singLogin(username
="wp2256520569@sina.com", password='')
self.assertEqual(self.getDivText(),
"请输入密码")

在最后一个测试用例中,因为提示出现在密码位置,所以需要在page中添加相关的xpath才能正常运行

divText=(By.XPATH,"/html/body/div[3]/div/div[2]/div/div/div[4]/div[1]/div[1]/div[1]/span[2]")

推荐阅读
  • Go GUIlxn/walk 学习3.菜单栏和工具栏的具体实现
    本文介绍了使用Go语言的GUI库lxn/walk实现菜单栏和工具栏的具体方法,包括消息窗口的产生、文件放置动作响应和提示框的应用。部分代码来自上一篇博客和lxn/walk官方示例。文章提供了学习GUI开发的实际案例和代码示例。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • Voicewo在线语音识别转换jQuery插件的特点和示例
    本文介绍了一款名为Voicewo的在线语音识别转换jQuery插件,该插件具有快速、架构、风格、扩展和兼容等特点,适合在互联网应用中使用。同时还提供了一个快速示例供开发人员参考。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • 本文介绍了在Windows环境下如何配置php+apache环境,包括下载php7和apache2.4、安装vc2015运行时环境、启动php7和apache2.4等步骤。希望对需要搭建php7环境的读者有一定的参考价值。摘要长度为169字。 ... [详细]
  • 深入理解CSS中的margin属性及其应用场景
    本文主要介绍了CSS中的margin属性及其应用场景,包括垂直外边距合并、padding的使用时机、行内替换元素与费替换元素的区别、margin的基线、盒子的物理大小、显示大小、逻辑大小等知识点。通过深入理解这些概念,读者可以更好地掌握margin的用法和原理。同时,文中提供了一些相关的文档和规范供读者参考。 ... [详细]
  • 基于dlib的人脸68特征点提取(眨眼张嘴检测)python版本
    文章目录引言开发环境和库流程设计张嘴和闭眼的检测引言(1)利用Dlib官方训练好的模型“shape_predictor_68_face_landmarks.dat”进行68个点标定 ... [详细]
  • 本文主要复习了数据库的一些知识点,包括环境变量设置、表之间的引用关系等。同时介绍了一些常用的数据库命令及其使用方法,如创建数据库、查看已存在的数据库、切换数据库、创建表等操作。通过本文的学习,可以加深对数据库的理解和应用能力。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 如何在HTML中获取鼠标的当前位置
    本文介绍了在HTML中获取鼠标当前位置的三种方法,分别是相对于屏幕的位置、相对于窗口的位置以及考虑了页面滚动因素的位置。通过这些方法可以准确获取鼠标的坐标信息。 ... [详细]
  • JavaScript和HTML之间的交互是经由过程事宜完成的。事宜:文档或浏览器窗口中发作的一些特定的交互霎时。能够运用侦听器(或处置惩罚递次来预订事宜),以便事宜发作时实行相应的 ... [详细]
  • 使用chrome编辑器实现网页截图功能的方法
    本文介绍了在chrome浏览器中使用编辑器实现网页截图功能的方法。通过在地址栏中输入特定命令,打开控制台并调用命令面板,用户可以方便地进行网页截图操作。 ... [详细]
  • 用Vue实现的Demo商品管理效果图及实现代码
    本文介绍了一个使用Vue实现的Demo商品管理的效果图及实现代码。 ... [详细]
  • 本文介绍了利用ARMA模型对平稳非白噪声序列进行建模的步骤及代码实现。首先对观察值序列进行样本自相关系数和样本偏自相关系数的计算,然后根据这些系数的性质选择适当的ARMA模型进行拟合,并估计模型中的位置参数。接着进行模型的有效性检验,如果不通过则重新选择模型再拟合,如果通过则进行模型优化。最后利用拟合模型预测序列的未来走势。文章还介绍了绘制时序图、平稳性检验、白噪声检验、确定ARMA阶数和预测未来走势的代码实现。 ... [详细]
author-avatar
O臭煊儿O
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有