基于Python的接口測(cè)試框架實(shí)例
背景
最近公司在做消息推送,那么自然就會(huì)產(chǎn)生很多接口,測(cè)試的過(guò)程中需要調(diào)用接口,我就突然覺(jué)得是不是可以自己寫(xiě)一個(gè)測(cè)試框架?
說(shuō)干就干,由于現(xiàn)有的接口測(cè)試工具Jmeter、SoupUI等學(xué)習(xí)周期有點(diǎn)長(zhǎng),干脆自己寫(xiě)一個(gè)吧,不求人,所有功能自己都能一清二楚。
當(dāng)然,寫(xiě)工具造輪子只是學(xué)習(xí)的一種方式,現(xiàn)成成熟的工具肯定比我們自己的寫(xiě)的好用。
開(kāi)發(fā)環(huán)境
-------------------------------------------------------------
操作系統(tǒng):Mac OS X EI Caption
Python版本:2.7
IDE:Pycharm
-------------------------------------------------------------
分析
接口是基于HTTP協(xié)議的,那么說(shuō)白了,就是發(fā)起HTTP請(qǐng)求就行了,對(duì)于Python來(lái)說(shuō)簡(jiǎn)直就是小菜一碟。直接使用requests就可以很輕松的完成任務(wù)。
架構(gòu)
整個(gè)框架是比較小的,涉及的東西也比較少,只要分清楚幾個(gè)模塊的功能就行了。

上面是一個(gè)接口測(cè)試的完整流程。只要一步一步的走下來(lái)就行了,并不是很難。
數(shù)據(jù)源
數(shù)據(jù)源我使用的是JSON來(lái)保存,當(dāng)然,比較廣泛的是使用Excel來(lái)保存,用JSON來(lái)保存是因?yàn)镴SON用起來(lái)比較方便,懶得去讀取Excel了,Python對(duì)JSON的支持是非常友好的。當(dāng)然這個(gè)就看個(gè)人喜好了。
{
"TestId": "testcase004",
"Method": "post",
"Title": "單獨(dú)推送消息",
"Desc": "單獨(dú)推送消息",
"Url": "http://xxx.xxx.xxx.xx",
"InputArg": {
"action": "44803",
"account": "1865998xxxx",
"uniqueid": "00D7C889-06A0-426E-BAB1-5741A1192038",
"title": "測(cè)試測(cè)試",
"summary": "豆豆豆",
"message": "12345",
"msgtype": "25",
"menuid": "203"
},
"Result": {
"errorno": "0"
}
}
示例如上面代碼所示,可以根據(jù)個(gè)人的業(yè)務(wù)需要進(jìn)行調(diào)整。
發(fā)送請(qǐng)求
發(fā)送請(qǐng)求就很簡(jiǎn)單了,用requests模塊,然后從JSON中讀取發(fā)送的參數(shù),post、get或者其他。由于要生成測(cè)試報(bào)告,那么發(fā)送的數(shù)據(jù)需要做一下記錄,我選擇用txt文本來(lái)作為記錄的容器。
f = file("case.json")
testData = json.load(f)
f.close()
def sendData(testData, num):
payload = {}
# 從json中獲取發(fā)送參數(shù)
for x in testData[num]['InputArg'].items():
payload[x[0]] = x[1]
with open('leftside.txt', 'a+') as f:
f.write(testData[num]['TestId'])
f.write('-')
f.write(testData[num]['Title'])
f.write('\n')
# 發(fā)送請(qǐng)求
data = requests.get(testData[num]['Url'], params=payload)
r = data.json()
接受返回
由于我們是需要生成測(cè)試報(bào)告的,那么返回的數(shù)據(jù)我們先需要進(jìn)行一次存儲(chǔ),可以選擇用數(shù)據(jù)庫(kù)存儲(chǔ),但是我覺(jué)得數(shù)據(jù)庫(kù)存儲(chǔ)太麻煩了,只要用txt文本作為存儲(chǔ)容器即可。
with open('rightside.txt', 'a+') as rs:
rs.write('發(fā)送數(shù)據(jù)')
rs.write('|')
rs.write('標(biāo)題:'+testData[num]['Title'])
rs.write('|')
rs.write('發(fā)送方式:'+testData[num]['Method'])
rs.write('|')
rs.write('案例描述:'+testData[num]['Desc'])
rs.write('|')
rs.write('發(fā)送地址:'+testData[num]['Url'])
rs.write('|')
rs.write('發(fā)送參數(shù):'+str(payload).decode("unicode-escape").encode("utf-8").replace("u\'","\'"))
rs.write('|')
rs.write(testData[num]['TestId'])
rs.write('\n')
結(jié)果判定
結(jié)果判定我使用的是全等于判定。因?yàn)槲覀兊慕涌谥恍枰@樣處理就行了,如果有需要,可以寫(xiě)成正則判定。
with open('result.txt', 'a+') as rst:
rst.write('返回?cái)?shù)據(jù)')
rst.write('|')
for x, y in r.items():
rst.write(' : '.join([x, y]))
rst.write('|')
# 寫(xiě)測(cè)試結(jié)果
try:
if cmp(r, testData[num]['Result']) == 0:
rst.write('pass')
else:
rst.write('fail')
except Exception:
rst.write('no except result')
rst.write('\n')
我這里結(jié)果有3種,成功、失敗或者沒(méi)結(jié)果。結(jié)果的設(shè)置就看自己的定義了。
生成測(cè)試報(bào)告
測(cè)試報(bào)告是一個(gè)重頭戲,由于我發(fā)送數(shù)據(jù)、返回?cái)?shù)據(jù)和結(jié)果都是用txt文本存儲(chǔ),那么每次使用a+模式新增,會(huì)讓結(jié)果越來(lái)越多,而且檢查起來(lái)非常蛋疼。
我的處理方式是每次測(cè)試完畢之后,用Python讀取txt文本中的數(shù)據(jù),然后使用Django動(dòng)態(tài)生成一個(gè)結(jié)果,然后再使用requests抓取這個(gè)網(wǎng)頁(yè),保存在Report文件夾中。
網(wǎng)頁(yè)報(bào)告
Django的方法我就不多說(shuō)了,博客中已經(jīng)有一整個(gè)系列文章了。我們需要在views文件中打開(kāi)之前記錄的3個(gè)txt文件,然后做一些數(shù)據(jù)處理,返回給前端,前端用Bootstrap來(lái)渲染,就能生成一個(gè)比較漂亮的測(cè)試報(bào)告。
def index(request):
rightside = []
result = []
rst_data = []
leftside = []
passed = 0
fail = 0
noresult = 0
with open(os.getcwd() + '/PortTest/leftside.txt') as ls:
for x in ls.readlines():
lf_data = {
'code': x.strip().split('-')[0],
'title': x.strip().split('-')[1]
}
leftside.append(lf_data)
with open(os.getcwd() + '/PortTest/rightside.txt') as rs:
for x in rs.readlines():
row = x.strip().split('|')
rs_data = {
"fssj": row[0],
"csbt": row[1],
"fsfs": row[2],
"alms": row[3],
"fsdz": row[4],
"fscs": row[5],
'testid': row[6]
}
rightside.append(rs_data)
with open(os.getcwd() + '/PortTest/result.txt') as rst:
for x in rst.readlines():
row = x.strip().split('|')
if row[len(row)-1] == 'fail':
fail += 1
elif row[len(row)-1] == 'pass':
passed += 1
elif row[len(row)-1] == 'no except result':
noresult += 1
rs_data = []
for y in row:
rs_data.append(y)
result.append(rs_data)
for a, b in zip(rightside, result):
data = {
"sendData": a,
"dealData": b,
"result": b[len(b)-1]
}
rst_data.append(data)
return render(request, 'PortTest/index.html', {"leftside": leftside,
"rst_data": rst_data,
"pass": passed,
"fail": fail,
"noresult": noresult})
基本上都是一些很基礎(chǔ)的知識(shí),字符串分割等等。這里的數(shù)據(jù)處理為了方便,在獲取數(shù)據(jù)存儲(chǔ)的時(shí)候就要按照一定的格式來(lái)存儲(chǔ),views的方法就很容易做處理。
前端代碼如下:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<link rel="stylesheet">
<script src="http://jb51.net/jquery/2.0.0/jquery.min.js"></script>
<script src="http://jb51.net/bootstrap/3.0.3/js/bootstrap.min.js"></script>
</head>
<body>
<div class="container">
<div class="row">
<div class="page-header">
<h1>接口測(cè)試報(bào)告
<small>Design By Sven</small>
</h1>
</div>
</div>
<div class="row">
<div class="col-md-4">
<h3>測(cè)試通過(guò) <span class="label label-success">{{ pass }}</span></h3>
</div>
<div class="col-md-4">
<h3>測(cè)試失敗 <span class="label label-danger">{{ fail }}</span></h3>
</div>
<div class="col-md-4">
<h3>無(wú)結(jié)果 <span class="label label-warning">{{ noresult }}</span></h3>
</div>
</div>
<p></p>
<div class="row">
<div class="col-md-3">
<ul class="list-group">
{% for ls in leftside %}
<li class="list-group-item"><a href="#{{ ls.code }}">{{ ls.code }} - {{ ls.title }}</a></li>
{% endfor %}
</ul>
</div>
<div class="col-md-9">
{{ x.result }}
{% for x in rst_data %}
<div class="panel-group" id="accordion">
{% if x.result == 'pass' %}
<div class="panel panel-success">
{% elif x.result == 'fail' %}
<div class="panel panel-danger">
{% elif x.result == 'no except result' %}
<div class="panel panel-warning">
{% endif %}
<div class="panel-heading">
<h4 class="panel-title">
<a data-toggle="collapse" href="#{{ x.sendData.testid }}">
{{ x.sendData.testid }} - {{ x.sendData.csbt }}
</a>
</h4>
</div>
<div id="{{ x.sendData.testid }}" class="panel-collapse collapse">
<div class="panel-body">
<b>{{ x.sendData.fssj }}</b><br>
{{ x.sendData.csbt }}<br>
{{ x.sendData.fsfs }}<br>
{{ x.sendData.alms }}<br>
{{ x.sendData.fsdz }}<br>
{{ x.sendData.fscs }}
<hr>
{% for v in x.dealData %}
{{ v }}<br>
{% endfor %}
</div>
</div>
</div>
</div>
<p></p>
{% endfor %}
</div>
</div>
</div>
<script>
$(function () {
$(window).scroll(function () {
if ($(this).scrollTop() != 0) {
$("#toTop").fadeIn();
} else {
$("#toTop").fadeOut();
}
});
$("body").append("<div id=\"toTop\" style=\"border:1px solid #444;background:#333;color:#fff;text-align:center;padding:10px 13px 7px 13px;position:fixed;bottom:10px;right:10px;cursor:pointer;display:none;font-family:verdana;font-size:22px;\">^</div>");
$("#toTop").click(function () {
$("body,html").animate({scrollTop: 0}, 800);
});
});
</script>
</body>
</html>
測(cè)試報(bào)告效果圖

最后
用Python寫(xiě)一個(gè)工具很容易,主要還是要能更方便地滿足實(shí)際工作中的使用需要為目的。如果要做完整的接口測(cè)試,還是盡量使用已經(jīng)成熟的工具。
PS:簡(jiǎn)單的造輪子也是學(xué)習(xí)原理的一個(gè)絕佳的方法。
以上這篇基于Python的接口測(cè)試框架實(shí)例就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- 利用Python如何實(shí)現(xiàn)數(shù)據(jù)驅(qū)動(dòng)的接口自動(dòng)化測(cè)試
- python實(shí)現(xiàn)接口并發(fā)測(cè)試腳本
- python接口自動(dòng)化測(cè)試之接口數(shù)據(jù)依賴的實(shí)現(xiàn)方法
- Python接口測(cè)試get請(qǐng)求過(guò)程詳解
- Python腳本完成post接口測(cè)試的實(shí)例
- 用Python進(jìn)行websocket接口測(cè)試
- python利用requests庫(kù)進(jìn)行接口測(cè)試的方法詳解
- 對(duì)python自動(dòng)生成接口測(cè)試的示例講解
- python編寫(xiě)接口測(cè)試文檔(以豆瓣搜索為例)
相關(guān)文章
Python Django實(shí)現(xiàn)layui風(fēng)格+django分頁(yè)功能的例子
今天小編就為大家分享一篇Python Django實(shí)現(xiàn)layui風(fēng)格+django分頁(yè)功能的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-08-08
Python并發(fā)之多進(jìn)程的方法實(shí)例代碼
這篇文章主要介紹了Python并發(fā)之多進(jìn)程的方法實(shí)例代碼,文中也提到了進(jìn)程與線程的共同點(diǎn),需要的朋友跟隨腳本之家小編一起看看吧2018-08-08
python中關(guān)于requests里的timeout()用法
這篇文章主要介紹了python中關(guān)于requests里的timeout()用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
簡(jiǎn)單了解Django ContentType內(nèi)置組件
這篇文章主要介紹了簡(jiǎn)單了解Django ContentType內(nèi)置組件,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-07-07
python單機(jī)五子棋的代碼實(shí)現(xiàn)示例
五子棋是經(jīng)典的棋牌類游戲,很多人都玩過(guò),那么如何用Python實(shí)現(xiàn)五子棋呢,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10
Python datetime時(shí)間格式化去掉前導(dǎo)0
Python datetime時(shí)間格式化去掉前導(dǎo)0,在format string的%與flag之間,添加一個(gè)“-”即可2014-07-07
python使用openpyxl庫(kù)讀取Excel文件數(shù)據(jù)
openpyxl是一個(gè)功能強(qiáng)大的庫(kù),可以輕松地實(shí)現(xiàn)Excel文件的讀寫(xiě)操作,本文將介紹如何使用openpyxl庫(kù)讀取Excel文件中的數(shù)據(jù),感興趣的小伙伴可以了解下2023-11-11

