前情:
在日常的接口測試工作中辨宠,模擬接口請求通常有兩種方法悯辙,
利用工具來模擬琳省,比如fiddler,postman躲撰,poster针贬,soapUI等
利用代碼來模擬,使用到一些網(wǎng)絡模塊拢蛋,比如HttpClient桦他,或者requests等
Fiddler是一個簡單的http協(xié)議調(diào)試代理工具,它界面友好谆棱,易于操作快压,是調(diào)試網(wǎng)絡,抓取請求的利器之一垃瞧。
Requests包是趁著目前如火如荼的爬蟲風潮而流行起來的Python的第三方包蔫劣,使用簡單,對于不需要對網(wǎng)絡底層模塊有太多了解又想實現(xiàn)接口測試的測試人員來說簡直不要太友好个从。
在接口測試中脉幢,接口通常是GET請求或者POST請求。以下是對接口測試中常見的四種Post請求數(shù)據(jù)方式進行一個詳細的講解:
post請求主體詳解
一個正常的post請求主要包括請求行嗦锐,請求頭鸵隧,請求主體
對于get請求來說沒有請求主體entity-body。對于post請求而言意推,不會對發(fā)送請求的數(shù)據(jù)格式進行限制,理論上你可以發(fā)任意數(shù)據(jù)珊蟀,但是服務器能不能處理就是另一回事了菊值。服務器收到數(shù)據(jù)后外驱,如何解析數(shù)據(jù)呢?它會以請求頭中的Content-Type設置的內(nèi)容來進行數(shù)據(jù)解析腻窒。確定好Content-Type的格式之后昵宇,請求主體的數(shù)據(jù)格式也就確定下來了。
Content-Type的格式有四種:分別是application/x-www-form-urlencoded(這也是默認格式)儿子、application/json瓦哎、text/xml以及multipart/form-data格式。
這些不同的post請求數(shù)據(jù)格式要通過HttpEntity來構(gòu)造柔逼,有必要簡單理一下HttpClient的HttpEntity對象蒋譬,因為所有的post請求數(shù)據(jù)均需要置于HttpEntity實體中進行發(fā)送。HttpEntity是一個接口愉适,實現(xiàn)這個接口的具體類有很多犯助,比較常用的是StringEntity、UrlEncodedFormEntity(繼承自StringEntity)维咸、MultipartEntity剂买。他們將在發(fā)送不同格式的post請求時被用到。接下來就詳細地介紹每一種數(shù)據(jù)格式對應的fiddler請求模擬和Requests請求模擬(Python實現(xiàn))的實現(xiàn)情況癌蓖。
(一)application/x-www-form-urlencoded數(shù)據(jù)格式
在W3C官網(wǎng)上明確對這種數(shù)據(jù)格式進行了定義:
This is the default content type. Forms submitted with this content type must be encoded as follows:Control names and values are escaped. Space characters are replaced by '+', and then reserved characters are escaped as described in [RFC1738], section 2.2: Non-alphanumeric characters are replaced by '%HH', a percent sign and two hexadecimal digits representing the ASCII code of the character. Line breaks are represented as "CR LF" pairs (i.e., '%0D%0A').The control names/values are listed in the order they appear in the document. The name is separated from the value by '=' and name/value pairs are separated from each other by '&'.
這是post請求最常見也是默認的數(shù)據(jù)提交格式瞬哼。它要求數(shù)據(jù)名稱(name)和數(shù)據(jù)值(value)之間以等號相連,與另一組name/value值之間用&相連租副。例如:parameter1=12345¶meter2=23456坐慰。將請求的內(nèi)容進行格式化了,其實這個方法同時簡化的客戶端發(fā)送附井,也簡化了服務器端獲取讨越,服務器通過getParameters(String name)即可獲取到傳送來的信息。這是最常見post提交數(shù)據(jù)的方式永毅,以form表單形式提交數(shù)據(jù)把跨。
(1)如果用fiddler模擬請求的話,請求頭和請求主體的內(nèi)容可以這樣構(gòu)造:
模擬請求之后沼死,從返回結(jié)果可以查看到我們的請求數(shù)據(jù):
(2)如果用Requests模擬post請求的話着逐,請求可以這樣構(gòu)造:
datas = {'parameter1':'12345','parameter2':'23456'}
r = requests.post('http://example.com',data=datas)
print(r.content)
print(r.status_code)
Reqeusts支持以form表單形式發(fā)送post請求,只需要將請求的參數(shù)構(gòu)造成一個字典意蛀,然后傳給requests.post()的data參數(shù)即可耸别。
**(二)application/json數(shù)據(jù)格式 **
application/json格式的請求頭是指用來告訴服務端post過去的消息主體是序列化后的 JSON 字符串。
(1)如果用fiddler模擬請求的話县钥,請求頭和請求主體的內(nèi)容可以這樣構(gòu)造:
模擬請求之后秀姐,從返回結(jié)果可以看到我們的請求數(shù)據(jù):
(2)如果用Requests模擬post請求的話,請求可以這樣構(gòu)造:
url = 'http://www.example/post'
s = json.dumps({'key1': 'value1', 'key2': 'value2'})
r = requests.post(url, data=s)
print (r.text)
這里我們可以發(fā)現(xiàn)Requests模擬post請求時若贮,請求頭格式為application/x-www-form-urlencoded與application/json的主要差別在于請求主體的構(gòu)造格式(前者是鍵值對省有,后者是JSON串),前者直接用字典傳入痒留,后者用json.dumps()函數(shù)將字典轉(zhuǎn)為JSON串即可。
(三)text/xml數(shù)據(jù)格式
(1)如果用fiddler模擬請求的話蠢沿,請求頭和請求主體的內(nèi)容可以這樣構(gòu)造:
模擬請求之后伸头,從返回結(jié)果可以看到我們的請求數(shù)據(jù):
(2)如果用Requests模擬post請求的話,請求可以這樣構(gòu)造:
xml = """my xml"""
headers = {'Content-Type': 'application/xml'}
requests.post('http://www.example.com', data=xml, headers=headers)
或者把xml作為一個文件來傳輸:
import requests
def request_ws(request):
with open(archivo_request,"r") as archivo:
request_data = archivo.read()
target_url = "http://127.0.0.1:8000/?wsdl"
headers = {'Content-type':'text/xml'}
data_response = requests.post(target_url, data=request_data, headers=headers)
(四)multipart/form-data數(shù)據(jù)格式
除了傳統(tǒng)的application/x-www-form-urlencoded表單舷蟀,我們另一個經(jīng)常用到的是上傳文件用的表單恤磷,這種表單的類型為multipart/form-data。
(1)如果用fiddler模擬請求的話野宜,請求頭和請求主體的內(nèi)容可以這樣構(gòu)造:
第一步扫步,先設置好請求頭格式,然后點擊upload file...
第二步速缨,上傳你的文件锌妻,這里我上傳一個png的圖片
這是fiddler根據(jù)我們上傳的文件自動調(diào)整生成的請求,在請求頭中看到旬牲,我們需要選擇一段數(shù)據(jù)作為“分割邊界”(boundary屬性)仿粹,這個“邊界數(shù)據(jù)”不能在內(nèi)容其他地方出現(xiàn),一般來說使用一段從概率上說“幾乎不可能”的數(shù)據(jù)即可原茅。每次post瀏覽器都會生成一個隨機的30-40位長度的隨機字符串吭历,瀏覽器一般不會遍歷這次post的所有數(shù)據(jù)找到一個不可能出現(xiàn)在數(shù)據(jù)中的字符串,一般都是隨機生成擂橘。選擇了這個邊界之后晌区,瀏覽器便把它放在Content-Type 里面?zhèn)鬟f給服務器,服務器根據(jù)此邊界解析數(shù)據(jù)通贞。下面的數(shù)據(jù)便根據(jù)boundary劃分段朗若,每一段便是一項數(shù)據(jù)。(每個field被分成小部分昌罩,而且包含一個value是"form-data"的"Content-Disposition"的頭部哭懈;一個"name"屬性對應field的ID等等,文件的話包括一個filename)模擬請求之后茎用,從返回結(jié)果可以看到我們的請求數(shù)據(jù):
(2)如果用Requests模擬post請求的話遣总,請求可以這樣構(gòu)造:
url = 'http://www.example.com'
files = {'file': open('C://Users//Someone//Desktop//2.png', 'rb')}
r = requests.post(url, files=files)
print(r.text)
綜上,就是接口測試中fiddler與requests模擬post接口四種請求數(shù)據(jù)的構(gòu)造方法轨功,總結(jié)起來有利于在以后的接口測試工組過程中可以及時查閱旭斥。
同時,Requests庫還可以處理多部分編碼(Multipart-Encoded)的文件古涧,比如xls文件垂券,甚至改寫content-Type的報頭。
比如:
url = 'http://httpbin.org/post'
files = {'file': ('report.xls', open('report.xls', 'rb'), 'application/vnd.ms-excel', {'Expires': '0'})}
r = requests.post(url, files=files)