技术标签: 技术分享 python 软件测试 lua 开发语言
优点:
自动化的一个明显的好处是可以在较少的时间内运行更多的测试。为什么UI自动化维护成本更高? 因为前端页面变化太快,而且UI自动化比较耗时(比如等待页面元素的加载、添加等待时间、定位元素、操作元素、模拟页面动作这些都需要时间) 为什么接口自动化维护成本较低? 因为接口较稳定,接口的响应时间基本上都是秒级、毫秒级别的,速度快,并且接口自动化本身也可以做一些有关联的操作、全流程的操作(比如:注册 --> 登录 --> 修改个人信息)
优点1、优点3、优点4是接口自动化和UI自动化公有的优点。
缺点:
GET请求和POST请求的区别:
以前:get 查询;post 新增;put 编辑;delete 删除
现在:get 查询;post 新增 + 编辑 + 删除
或者:纯post走天下
前后端分离
开发模式
以前老的方式:
新的方式:
通过F12打开浏览器开发者工具进行抓包,返回数据是json格式的就是前后端分离,返回时html页面就是没有前后端分离。
微服务的概念:
将大模块切分成小模块。减少代码的耦合度,从而降低模块与模块之间的影响。原先是一个jar包里面包含所有模块,改一个模块就有可能影响其他模块,现在是将一个一个的模块都打成一个一个的jar包,模块与模块之间的交互通过接口,哪个模块出了问题,只需要修改那个模块的jar包,避免因为修改一个模块的代码导致其他模块出错。
接口自动化requests环境搭建
接口自动化核心库:requests
安装requests库的方法:
方法一:
命令行安装,打开cmd或者终端,输入以下命令:
pip install requests -i https://pypi.douban.com/simple/
方法二:
在pycharm中安装,settings --> Project --> Project Interpreter --> 点击“+”号 --> 输入request安装
测试环境是否ok
import requests
url_toutiao = "https://www.ixigua.com/tlb/comment/article/v5/tab_comments/?tab_index=0&count=10&group_id=6914830518563373581&item_id=6914830518563373581&aid=1768"
# 方式一:
# result_toutiao = requests.get(url_toutiao)
# 方式二:
result_toutiao = requests.get(url=url_toutiao)
# 方式三:
# result_toutiao = requests.get(
# "https://www.ixigua.com/tlb/comment/article/v5/tab_comments/?tab_index=0&count=1&group_id=6914830518563373581&item_id=6914830518563373581&aid=1768")
# print(result_toutiao.json())
# print(type(result_toutiao.json())) # <class 'dict'>
result = result_toutiao.json()
print(result)
expect_result = "华晨金杯汽车花朵朵"
actual_result = result["data"][0]["comment"]["user_name"]
print(actual_result)
if expect_result == actual_result:
print("pass!")
else:
print("failed!")
JSON、URL、text、encoding、status_code、encoding、cookies
print(result.json()) # 响应结果以json的形式打印输出
print(result.url) # 打印url地址
print(result.text) # 以文本格式打印服务器响应的内容
print(result.status_code) # 响应状态码
print(result.encoding) # 编码格式
print(result.cookies) # cookie
JSON(JavaScript Object Notation, JS 对象简谱) 是一种轻量级的数据交换格式。它基于 ECMAScript (欧洲计算机协会制定的js规范)的一个子集,采用完全独立于编程语言的文本格式来存储和表示数据。简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。 易于人阅读和编写,同时也易于机器解析和生成,并有效地提升网络传输效率。
JSON格式在Python里面相当于字典类型。
JSON格式化:JSON在线视图查看器(Online JSON Viewer)
url在线编码转换:URL在线编码转换工具 - 编码转换工具 - W3Cschool
GET请求方式
import requests
url_toutiao = "https://www.ixigua.com/tlb/comment/article/v5/tab_comments/?tab_index=0&count=10&group_id=6914830518563373581&item_id=6914830518563373581&aid=1768"
# 方式一:
# result_toutiao = requests.get(url_toutiao)
# 方式二:
result_toutiao = requests.get(url=url_toutiao)
# 方式三:
# result_toutiao = requests.get(
# "https://www.ixigua.com/tlb/comment/article/v5/tab_comments/?tab_index=0&count=1&group_id=6914830518563373581&item_id=6914830518563373581&aid=1768")
# print(result_toutiao.json())
# print(type(result_toutiao.json())) # <class 'dict'>
result = result_toutiao.json()
print(result)
expect_result = "华晨金杯汽车花朵朵"
actual_result = result["data"][0]["comment"]["user_name"]
print(actual_result)
if expect_result == actual_result:
print("pass!")
else:
print("failed!")
运行结果:
{'message': 'success', 'err_no': 0, 'data': [{'comment': {'id': 6914864825282215951, 'id_str': '6914864825282215951', 'text': '藁城出国打工的人很多,重点检查藁城区!', 'content_rich_span': '{"links":[]}', 'user_id': 940799526971408, 'user_name': '华晨金杯汽车花朵朵',}, 'post_count': 0, 'stick_toast': 1, 'stable': True}
华晨金杯汽车花朵朵
pass!
POST请求方式
import requests
url_v_login = "http://[服务器ip]:8081/login"
# 定义参数,字典格式
payload = {'username': 'sang', 'password': '123'}
# Content-Type: application/json --> json
# Content-Type: application/x-www-form-urlencoded --> data
result = requests.post(url_v_login, data=payload)
# 将返回结果转为json格式
result_json = result.json()
print(result_json) # {'status': 'success', 'msg': '登录成功'}
# 获取RequestsCookieJar
result_cookie = result.cookies
print(result_cookie, type(result_cookie)) # RequestsCookieJar
# 将RequestsCookieJar转化为字典格式
result_cookie_dic = requests.utils.dict_from_cookiejar(result_cookie)
print(result_cookie_dic) # {'JSESSIONID': 'D042C5FE4CFF337806D545B0001E7197'}
# 获取SESSION
final_cookie = "JSESSIONID=" + result_cookie_dic["JSESSIONID"] # SJSESSIONID=D042C5FE4CFF337806D545B0001E7197
print(final_cookie)
PUT请求方式
# V部落_编辑栏目
# 定义请求头,自动获取cookie的方法详情请看下文
headers = {"Cookie": "VBlog(self.requests).get_cookie()"}
new_now_time = time.strftime("%Y%m%d%H%M%S", time.localtime(time.time()))
new_category_name = "更新栏目" + new_now_time
payload = {"id": 2010, "cateName": new_category_name}
self.requests.put("http://[服务器ip]:8081/admin/category/", headers=headers, data=payload)
DELETE请求方式
# 删除栏目
result = self.requests.delete("http://[服务器ip]:8081/admin/category/" + “2010”, headers=headers)
print(result.json()) # {'status': 'success', 'msg': '删除成功!'}
self.assertEqual("删除成功!", result.json()["msg"])
手动传入cookie的值(每次通过浏览器F12抓包,然后复制request header里面的cookie)
import requests
# V部落查询栏目
url_v_category = "http://[服务器ip]:8081/admin/category/all"
# 定制请求头
# 如果你想为请求添加HTTP头部,只要简单地传递一个字典给headers参数就可以了
v_headers = {
"cookie": "studentUserName=ctt01; Hm_lvt_cd8218cd51f800ed2b73e5751cb3f4f9=1609742724,1609762306,1609841170,1609860946; adminUserName=admin; JSESSIONID=9D1FF19F333C5E25DBA60769E9F5248E"}
result = requests.get(url_v_category, headers=v_headers)
# 打印json格式的响应结果
print(result.json())
cookie自动获取
import requests
url_v_login = "http://[服务器ip]:8081/login"
# 定义参数,字典格式
payload = {'username': 'sang', 'password': '123'}
# Content-Type: application/json --> json
# Content-Type: application/x-www-form-urlencoded --> data
result = requests.post(url_v_login, data=payload)
# 将返回结果转为json格式
result_json = result.json()
print(result_json) # {'status': 'success', 'msg': '登录成功'}
# 获取RequestsCookieJar
result_cookie = result.cookies
print(result_cookie, type(result_cookie)) # RequestsCookieJar
# 将RequestsCookieJar转化为字典格式
result_cookie_dic = requests.utils.dict_from_cookiejar(result_cookie)
print(result_cookie_dic) # {'JSESSIONID': 'D042C5FE4CFF337806D545B0001E7197'}
# 获取SESSION
final_cookie = "JSESSIONID=" + result_cookie_dic["JSESSIONID"] # SJSESSIONID=D042C5FE4CFF337806D545B0001E7197
print(final_cookie)
批量获取cookie脚本
import requests
def get_cookie(username, password):
"""通过考试系统学生登录获取单个cookie"""
url_login = "http://[服务器ip]:8088/api/user/login"
payload = {"userName": username, "password": password, "remember": False}
result = requests.post(url_login, json=payload)
# result_json = result.json()
# print(result_json)
# 获取RequestsCookieJar
result_cookie = result.cookies
# print(result_cookie, type(result_cookie)) # RequestsCookieJar
# 将RequestsCookieJar转化为字典格式
result_cookie_dic = requests.utils.dict_from_cookiejar(result_cookie)
# print(result_cookie_dic) # {'SESSION': 'YzFkM2IzN2QtZWY1OC00Nzc4LTgyOWYtNjg5OGRiZDZlM2E4'}
# 获取SESSION
final_cookie = "SESSION=" + result_cookie_dic["SESSION"] # SESSION=Mzc2...
return final_cookie
from test01.demo04_student_login import get_cookie
import os
def get_batch_cookies():
"""批量获取cookie"""
# 获取cookie之前,先将cookies.csv文件内容清空
# with open(r"D:\Desktop\Testman_Study\API_auto\file\cookies.csv", "w") as cookies_info:
# cookies_info.write("")
# 或者将文件删除
os.remove(r"D:\Desktop\Testman_Study\API_auto\file\cookies.csv")
# 读取csv文件
with open(r"D:\Desktop\Testman_Study\API_auto\file\register.csv", "r") as user_info:
for user in user_info:
user_list = user.strip().split(",")
# 调用获取单个cookies的方法,传入注册好的用户名和密码
cookies = get_cookie(user_list[0], user_list[1])
# 将cookie追加写入文件
with open(r"D:\Desktop\Testman_Study\API_auto\file\cookies.csv", "a") as cookies_info:
cookies_info.write(cookies + "\n")
# 调用方法
get_batch_cookies()
register.csv(前提是这些账号和密码都是已经注册过的,可以直接登录)
poopoo001,123456,1
poopoo002,123457,2
poopoo003,123458,3
poopoo004,123459,4
......
cookies.csv
SESSION=ZmE3YmU4ZDctNDExZS00MDdhLWE0YjEtMjAyZjQxOTMxYmUx
SESSION=YjdkNTZhNTUtNGFmMi00MjVkLWEyNjctOTNiMmRmOTY1YTdm
SESSION=ZTJmMTYzMWEtZjUzOS00NTlhLWI0OWQtMzBmN2RkYmU4YmRi
SESSION=YTM0ZGRhOTctZjk5Ni00OWZhLTg1YTItZjUyMTMwZGE2MjVi
......
import requests
# 文章列表
url_v_article = "http://[服务器ip]:8081/article/all"
v_headers = {
"Cookie": "studentUserName=ctt01; Hm_lvt_cd8218cd51f800ed2b73e5751cb3f4f9=1609742724,1609762306,1609841170,1609860946; adminUserName=admin; JSESSIONID=9D1FF19F333C5E25DBA60769E9F5248E"}
# 自定义url参数,定义一个字典,将参数拆分,再将字典传递给params变量即可
article_params = {"state": 1, # -1:全部文章 1:已发表 0:回收站 2:草稿箱
"page": 1, # 显示第1页
"count": 6, # 每页显示6条
"keywords": "" # 包含的关键字
}
keywords = ["大橘猫", "跑男", "牙"]
for keyword in keywords:
article_params["keywords"] = keyword
# headers和params是不定长的,根据定义的字典传参
result = requests.get(url_v_article, headers=v_headers, params=article_params)
print(result.json())
unittest框架结构:
if _name_ == '__main__':
if __name__ == '__main__'的意思是:
闭环断言(新增 --> 查询 --> 修改 --> 查询 --> 删除 -->查询)
def test_article(self):
# ①V部落_新增文章
now_time = time.strftime("%Y%m%d%H%M%S", time.localtime(time.time()))
title = "蔡坨坨" + now_time
payload = {"id": -1, "title": title, "mdContent": "文章内容", "state": 1, "htmlContent": "<p>文章内容</p>",
"dynamicTags": "", "cid": 62}
headers = {"Cookie": VBlog(self.requests).get_cookie()}
result = self.requests.post("http://[服务器ip]:8081/article/", headers=headers, data=payload)
# ②查询文章
url_v_article = "http://[服务器ip]:8081/article/all"
article_params = {"state": 1, # -1:全部文章 1:已发表 0:回收站 2:草稿箱
"page": 1, # 显示第1页
"count": 6, # 每页显示6条
"keywords": title # 包含的关键字title
}
result = requests.get(url_v_article, headers=headers, params=article_params, timeout=30)
print(result.json()) # 响应结果以json的形式打印输出
ls = result.json()["articles"]
act = 123
# 查到新增的文章,说明新增成功
for l in range(0, len(ls)):
if ls[l]["title"] == title:
act = "ok"
article_id = ls[l]["id"]
self.assertEqual("ok", act)
# ③编辑文章
now_time = time.strftime("%Y%m%d%H%M%S", time.localtime(time.time()))
title = "修改文章" + now_time
payload = {"id": article_id, "title": title, "mdContent": "修改内容", "state": 1, "htmlContent": "<p>修改内容</p>",
"dynamicTags": "", "cid": 62}
headers = {"Cookie": VBlog(self.requests).get_cookie()}
self.requests.post("http://[服务器ip]:8081/article/", headers=headers, data=payload)
# 编辑完,查询文章
url_v_article = "http://[服务器ip]:8081/article/all"
article_params = {"state": 1, # -1:全部文章 1:已发表 0:回收站 2:草稿箱
"page": 1, # 显示第1页
"count": 6, # 每页显示6条
"keywords": title # 包含的关键字title
}
result = requests.get(url_v_article, headers=headers, params=article_params, timeout=30)
print(result.json()) # 响应结果以json的形式打印输出
ls = result.json()["articles"]
act = 123
# 查到修改过的文章,说明编辑成功
for l in range(0, len(ls)):
if ls[l]["title"] == title:
act = "ok"
article_id = ls[l]["id"]
self.assertEqual("ok", act)
# ④查看文章详情
article_id = str(article_id)
result = self.requests.get("http://[服务器ip]:8081/article/" + article_id, headers=headers)
print(result.json())
if result.json()["title"] == title:
act = "ok"
self.assertEqual(act, "ok")
# ⑤删除文章
payload = {'aids': article_id, 'state': 1}
result = self.requests.put("http://[服务器ip]:8081/article/dustbin", headers=headers, data=payload)
print(result.json())
act = result.json()["msg"]
self.assertEqual(act, "删除成功!")
HTMLTestRunner.py百度网盘链接:
from reports import HTMLTestRunner
from case.exam_case.teacher_case import TeacherCase
import unittest
import os
import time
# 创建测试套件
suite = unittest.TestSuite()
# 添加测试用例,根据添加顺序执行
# 添加单个测试用例
# suite.addTest(TeacherCase("test_001_admin_login"))
# 添加多个测试用例
suite.addTests([TeacherCase("test_001_admin_login"),
TeacherCase("test_002_insert_paper"),
TeacherCase("test_003_select_paper"),
])
# 定义测试报告的存放的路径
path = r"D:\Desktop\Testman_Study\unittest_exam_system\reports"
# 判断路径是否存在
if not os.path.exists(path):
# 如果不存在,则创建一个
os.makedirs(path)
else:
pass
# 定义一个时间戳用于测试报告命名
now_time = time.strftime("%Y-%m-%d-%H-%M-%S", time.localtime(time.time()))
reports_path = path + "\\" + now_time + "(exam_report).html"
reports_title = u"考试系统&V部落——测试报告"
desc = u"考试系统&V部落——接口自动化测试报告"
# 二进制写
fp = open(reports_path, "wb")
runner = HTMLTestRunner.HTMLTestRunner(stream=fp, title=reports_title, description=desc)
# 运行
runner.run(suite)
postman、JMeter、requests总结:
感谢每一个认真阅读我文章的人!!!
作为一位过来人也是希望大家少走一些弯路,如果你不想再体验一次学习时找不到资料,没人解答问题,坚持几天便放弃的感受的话,在这里我给大家分享一些自动化测试的学习资源,希望能给你前进的路上带来帮助。
我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。
视频文档获取方式:
这份文档和视频资料,对于想从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!以上均可以分享,点下方进群即可自行领取。
文章浏览阅读1.6k次。安装配置gi、安装数据库软件、dbca建库见下:http://blog.csdn.net/kadwf123/article/details/784299611、检查集群节点及状态:[root@rac2 ~]# olsnodes -srac1 Activerac2 Activerac3 Activerac4 Active[root@rac2 ~]_12c查看crs状态
文章浏览阅读1.3w次,点赞45次,收藏99次。我个人用的是anaconda3的一个python集成环境,自带jupyter notebook,但在我打开jupyter notebook界面后,却找不到对应的虚拟环境,原来是jupyter notebook只是通用于下载anaconda时自带的环境,其他环境要想使用必须手动下载一些库:1.首先进入到自己创建的虚拟环境(pytorch是虚拟环境的名字)activate pytorch2.在该环境下下载这个库conda install ipykernelconda install nb__jupyter没有pytorch环境
文章浏览阅读5.2k次,点赞19次,收藏28次。选择scoop纯属意外,也是无奈,因为电脑用户被锁了管理员权限,所有exe安装程序都无法安装,只可以用绿色软件,最后被我发现scoop,省去了到处下载XXX绿色版的烦恼,当然scoop里需要管理员权限的软件也跟我无缘了(譬如everything)。推荐添加dorado这个bucket镜像,里面很多中文软件,但是部分国外的软件下载地址在github,可能无法下载。以上两个是官方bucket的国内镜像,所有软件建议优先从这里下载。上面可以看到很多bucket以及软件数。如果官网登陆不了可以试一下以下方式。_scoop-cn
文章浏览阅读4.5k次,点赞2次,收藏3次。首先要有一个color-picker组件 <el-color-picker v-model="headcolor"></el-color-picker>在data里面data() { return {headcolor: ’ #278add ’ //这里可以选择一个默认的颜色} }然后在你想要改变颜色的地方用v-bind绑定就好了,例如:这里的:sty..._vue el-color-picker
文章浏览阅读640次。基于芯片日益增长的问题,所以内核开发者们引入了新的方法,就是在内核中只保留函数,而数据则不包含,由用户(应用程序员)自己把数据按照规定的格式编写,并放在约定的地方,为了不占用过多的内存,还要求数据以根精简的方式编写。boot启动时,传参给内核,告诉内核设备树文件和kernel的位置,内核启动时根据地址去找到设备树文件,再利用专用的编译器去反编译dtb文件,将dtb还原成数据结构,以供驱动的函数去调用。firmware是三星的一个固件的设备信息,因为找不到固件,所以内核启动不成功。_exynos 4412 刷机
文章浏览阅读2w次,点赞24次,收藏42次。Linux系统配置jdkLinux学习教程,Linux入门教程(超详细)_linux配置jdk
文章浏览阅读3.3k次,点赞5次,收藏19次。xlabel('\delta');ylabel('AUC');具体符号的对照表参照下图:_matlab微米怎么输入
文章浏览阅读119次。顺序读写指的是按照文件中数据的顺序进行读取或写入。对于文本文件,可以使用fgets、fputs、fscanf、fprintf等函数进行顺序读写。在C语言中,对文件的操作通常涉及文件的打开、读写以及关闭。文件的打开使用fopen函数,而关闭则使用fclose函数。在C语言中,可以使用fread和fwrite函数进行二进制读写。 Biaoge 于2024-03-09 23:51发布 阅读量:7 ️文章类型:【 C语言程序设计 】在C语言中,用于打开文件的函数是____,用于关闭文件的函数是____。
文章浏览阅读3.4k次,点赞2次,收藏13次。跟随鼠标移动的粒子以grid(SOP)为partical(SOP)的资源模板,调整后连接【Geo组合+point spirit(MAT)】,在连接【feedback组合】适当调整。影响粒子动态的节点【metaball(SOP)+force(SOP)】添加mouse in(CHOP)鼠标位置到metaball的坐标,实现鼠标影响。..._touchdesigner怎么让一个模型跟着鼠标移动
文章浏览阅读178次。项目运行环境配置:Jdk1.8 + Tomcat7.0 + Mysql + HBuilderX(Webstorm也行)+ Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。项目技术:Springboot + mybatis + Maven +mysql5.7或8.0+html+css+js等等组成,B/S模式 + Maven管理等等。环境需要1.运行环境:最好是java jdk 1.8,我们在这个平台上运行的。其他版本理论上也可以。_基于java技术的停车场管理系统实现与设计
文章浏览阅读3.5k次。前言对于MediaPlayer播放器的源码分析内容相对来说比较多,会从Java-&amp;gt;Jni-&amp;gt;C/C++慢慢分析,后面会慢慢更新。另外,博客只作为自己学习记录的一种方式,对于其他的不过多的评论。MediaPlayerDemopublic class MainActivity extends AppCompatActivity implements SurfaceHolder.Cal..._android多媒体播放源码分析 时序图
文章浏览阅读2.4k次,点赞41次,收藏13次。java 数据结构与算法 ——快速排序法_快速排序法