之前的文章中已經對如何錄制 web 的請求進行了詳細的描述,敬請參閱:JMeter學習-004-WEB腳本入門實戰
同時,我們的手機應用(例如:京東、天貓、唯品會、攜程、易迅 等等 App)所發出的請求,也可進行錄制,只需要將手機連接的無線網絡代理到本機(以 iPhone 為例,依次點擊 【設置】/【無線局域網】/【連接網絡信息,即網絡右側詳細信息符號】/【HTTP代理】/手動】,填寫 服務器IP、端口 對應填寫 JMeter 機器的 IP、JMeter HTTP 代理端口 即可),並將對應的端口設置為 JMeter 中設置的端口后,即可對相應的操作進行錄制操作。非常簡單,在此不再贅述。
當我們執行腳本的時候,需要依據一定的條件去判斷腳本是否執行成功,例如響應信息(響應頭、主體、響應數據、響應代碼、大小等)、響應時間。JMeter 中也響應的給我們提供了相應的斷言,大多數通用的斷言 JMeter 均已給出,例如:Response Assertion、Size Assertion、MD5Hex Assertion等等。JMeter 提供的斷言列表如下所示:
以下就以易迅網App的一個簡單的類目搜索,對其添加響應斷言和大小斷言,來簡單的演示一下 JMeter 中的斷言。具體的操作步驟如下:
- 獲取類目 手機數碼-手機通訊-蘋果 的 HTTP 請求(通過 JMeter HTTP代理服務器 錄制即可獲得,在此略)
- 添加如下所示 Response Assertion(前四)、Size Assertion(后一):
- 01 - 判斷類目搜索結果狀態碼
- 02 - 判斷類目搜索結果響應數據包含蘋果
- 03 - 判斷類目搜索結果中不包含信息
- 04 - 判斷類目搜索結果響應頭信息中包含 Server: nginx
- 05 - 判斷響應結果大小(body)
- 添加查看結果樹
添加響應斷言:
右鍵單擊 HTTP請求(手機數碼-手機通訊-蘋果),依次選擇:【添加/斷言/響應斷言】,如下圖所示:
響應斷言添加頁面如下所示:
要測試的響應字段:取樣區域,也可簡單理解為取樣的數據源。單選 與 復選按鈕可同時選擇。
模式匹配規則:判斷的關系依據。單選 與 復選按鈕可同時選擇,選擇了復選框后,為前面關系條件的否。例如前面選擇了 包括,同時勾選了 否,則實際判斷關系依據為:不包括。
要測試的模式:可簡單理解為判斷的值。
PS:一個斷言僅可判斷一種類型的斷言,無法混雜判斷,即 要測試的響應字段、模式匹配規則、要測試的模式 在此斷言中都是相同,若模式匹配規則不同,則需新增響應斷言。
最終的腳本樣式如下所示:
腳本源碼為:
<?xml version="1.0" encoding="UTF-8"?> <jmeterTestPlan version="1.2" properties="2.8" jmeter="2.13 r1665067"> <hashTree> <TestPlan guiclass="TestPlanGui" testclass="TestPlan" testname="接口自動化測試用例 - 非登錄態" enabled="true"> <stringProp name="TestPlan.comments"></stringProp> <boolProp name="TestPlan.functional_mode">false</boolProp> <boolProp name="TestPlan.serialize_threadgroups">false</boolProp> <elementProp name="TestPlan.user_defined_variables" elementType="Arguments" guiclass="ArgumentsPanel" testclass="Arguments" testname="用戶定義的變量" enabled="true"> <collectionProp name="Arguments.arguments"/> </elementProp> <stringProp name="TestPlan.user_define_classpath"></stringProp> </TestPlan> <hashTree> <ThreadGroup guiclass="ThreadGroupGui" testclass="ThreadGroup" testname="001-類目搜索驗證" enabled="true"> <stringProp name="ThreadGroup.on_sample_error">continue</stringProp> <elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="循環控制器" enabled="true"> <boolProp name="LoopController.continue_forever">false</boolProp> <stringProp name="LoopController.loops">1</stringProp> </elementProp> <stringProp name="ThreadGroup.num_threads">1</stringProp> <stringProp name="ThreadGroup.ramp_time">1</stringProp> <longProp name="ThreadGroup.start_time">1419564228000</longProp> <longProp name="ThreadGroup.end_time">1419564228000</longProp> <boolProp name="ThreadGroup.scheduler">false</boolProp> <stringProp name="ThreadGroup.duration"></stringProp> <stringProp name="ThreadGroup.delay"></stringProp> </ThreadGroup> <hashTree> <HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="手機數碼-手機通訊-蘋果" enabled="true"> <elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" enabled="true"> <collectionProp name="Arguments.arguments"> <elementProp name="districtId" elementType="HTTPArgument"> <boolProp name="HTTPArgument.always_encode">false</boolProp> <stringProp name="Argument.name">districtId</stringProp> <stringProp name="Argument.value">29357</stringProp> <stringProp name="Argument.metadata">=</stringProp> <boolProp name="HTTPArgument.use_equals">true</boolProp> </elementProp> <elementProp name="exAppTag" elementType="HTTPArgument"> <boolProp name="HTTPArgument.always_encode">false</boolProp> <stringProp name="Argument.name">exAppTag</stringProp> <stringProp name="Argument.value">2045191607</stringProp> <stringProp name="Argument.metadata">=</stringProp> <boolProp name="HTTPArgument.use_equals">true</boolProp> </elementProp> </collectionProp> </elementProp> <stringProp name="HTTPSampler.domain">mb.51buy.com</stringProp> <stringProp name="HTTPSampler.port"></stringProp> <stringProp name="HTTPSampler.connect_timeout"></stringProp> <stringProp name="HTTPSampler.response_timeout"></stringProp> <stringProp name="HTTPSampler.protocol">http</stringProp> <stringProp name="HTTPSampler.contentEncoding">GB2312</stringProp> <stringProp name="HTTPSampler.path">/json.php?mod=Search&act=page&p=1&path=706188t706189&districtId=29357&areacode=1&dtype=list%7Cpage%7Cclasses&appSource=android&appVersion=45</stringProp> <stringProp name="HTTPSampler.method">POST</stringProp> <boolProp name="HTTPSampler.follow_redirects">true</boolProp> <boolProp name="HTTPSampler.auto_redirects">false</boolProp> <boolProp name="HTTPSampler.use_keepalive">true</boolProp> <boolProp name="HTTPSampler.DO_MULTIPART_POST">false</boolProp> <boolProp name="HTTPSampler.monitor">false</boolProp> <stringProp name="HTTPSampler.embedded_url_re"></stringProp> </HTTPSamplerProxy> <hashTree> <HeaderManager guiclass="HeaderPanel" testclass="HeaderManager" testname="HTTP信息頭管理器" enabled="true"> <collectionProp name="HeaderManager.headers"> <elementProp name="Charset" elementType="Header"> <stringProp name="Header.name">Charset</stringProp> <stringProp name="Header.value">UTF-8</stringProp> </elementProp> <elementProp name="Content-Type" elementType="Header"> <stringProp name="Header.name">Content-Type</stringProp> <stringProp name="Header.value">application/x-www-form-urlencoded</stringProp> </elementProp> <elementProp name="Accept-Encoding" elementType="Header"> <stringProp name="Header.name">Accept-Encoding</stringProp> <stringProp name="Header.value">gzip</stringProp> </elementProp> <elementProp name="User-Agent" elementType="Header"> <stringProp name="Header.name">User-Agent</stringProp> <stringProp name="Header.value">Dalvik/1.6.0 (Linux; U; Android 4.4.2; GT-I9502 Build/KOT49H)</stringProp> </elementProp> </collectionProp> </HeaderManager> <hashTree/> <ResponseAssertion guiclass="AssertionGui" testclass="ResponseAssertion" testname="01 - 判斷類目搜索結果狀態碼" enabled="true"> <collectionProp name="Asserion.test_strings"> <stringProp name="71131476">"errno":0</stringProp> </collectionProp> <stringProp name="Assertion.test_field">Assertion.response_data</stringProp> <boolProp name="Assertion.assume_success">false</boolProp> <intProp name="Assertion.test_type">2</intProp> </ResponseAssertion> <hashTree/> <ResponseAssertion guiclass="AssertionGui" testclass="ResponseAssertion" testname="02 - 判斷類目搜索結果響應數據包含蘋果" enabled="true"> <collectionProp name="Asserion.test_strings"> <stringProp name="1065923">蘋果</stringProp> </collectionProp> <stringProp name="Assertion.test_field">Assertion.response_data</stringProp> <boolProp name="Assertion.assume_success">false</boolProp> <intProp name="Assertion.test_type">2</intProp> </ResponseAssertion> <hashTree/> <ResponseAssertion guiclass="AssertionGui" testclass="ResponseAssertion" testname="03 - 判斷類目搜索結果中不包含信息" enabled="true"> <collectionProp name="Asserion.test_strings"> <stringProp name="1425729940">我的博客地址: http://www.cnblogs.com/fengpingfan/</stringProp> </collectionProp> <stringProp name="Assertion.test_field">Assertion.response_headers</stringProp> <boolProp name="Assertion.assume_success">false</boolProp> <intProp name="Assertion.test_type">6</intProp> </ResponseAssertion> <hashTree/> <ResponseAssertion guiclass="AssertionGui" testclass="ResponseAssertion" testname="04 - 判斷類目搜索結果響應頭信息中包含 Server: nginx" enabled="true"> <collectionProp name="Asserion.test_strings"> <stringProp name="1066132337">Server: nginx</stringProp> </collectionProp> <stringProp name="Assertion.test_field">Assertion.response_headers</stringProp> <boolProp name="Assertion.assume_success">false</boolProp> <intProp name="Assertion.test_type">2</intProp> </ResponseAssertion> <hashTree/> <SizeAssertion guiclass="SizeAssertionGui" testclass="SizeAssertion" testname="05 - 判斷響應結果大小(body)" enabled="true"> <stringProp name="Assertion.test_field">SizeAssertion.response_data</stringProp> <stringProp name="SizeAssertion.size">4000</stringProp> <intProp name="SizeAssertion.operator">3</intProp> </SizeAssertion> <hashTree/> </hashTree> </hashTree> <ResultCollector guiclass="AssertionVisualizer" testclass="ResultCollector" testname="斷言結果" enabled="true"> <boolProp name="ResultCollector.error_logging">false</boolProp> <objProp> <name>saveConfig</name> <value class="SampleSaveConfiguration"> <time>true</time> <latency>true</latency> <timestamp>true</timestamp> <success>true</success> <label>true</label> <code>true</code> <message>true</message> <threadName>true</threadName> <dataType>true</dataType> <encoding>false</encoding> <assertions>true</assertions> <subresults>true</subresults> <responseData>false</responseData> <samplerData>false</samplerData> <xml>false</xml> <fieldNames>false</fieldNames> <responseHeaders>false</responseHeaders> <requestHeaders>false</requestHeaders> <responseDataOnError>false</responseDataOnError> <saveAssertionResultsFailureMessage>false</saveAssertionResultsFailureMessage> <assertionsResultsToSave>0</assertionsResultsToSave> <bytes>true</bytes> <threadCounts>true</threadCounts> </value> </objProp> <stringProp name="filename"></stringProp> </ResultCollector> <hashTree/> <ResultCollector guiclass="ViewResultsFullVisualizer" testclass="ResultCollector" testname="察看結果樹" enabled="true"> <boolProp name="ResultCollector.error_logging">false</boolProp> <objProp> <name>saveConfig</name> <value class="SampleSaveConfiguration"> <time>true</time> <latency>true</latency> <timestamp>true</timestamp> <success>true</success> <label>true</label> <code>true</code> <message>true</message> <threadName>true</threadName> <dataType>true</dataType> <encoding>false</encoding> <assertions>true</assertions> <subresults>true</subresults> <responseData>false</responseData> <samplerData>false</samplerData> <xml>false</xml> <fieldNames>false</fieldNames> <responseHeaders>false</responseHeaders> <requestHeaders>false</requestHeaders> <responseDataOnError>false</responseDataOnError> <saveAssertionResultsFailureMessage>false</saveAssertionResultsFailureMessage> <assertionsResultsToSave>0</assertionsResultsToSave> <bytes>true</bytes> <threadCounts>true</threadCounts> </value> </objProp> <stringProp name="filename"></stringProp> </ResultCollector> <hashTree/> </hashTree> </hashTree> </jmeterTestPlan>
腳本執行結果如下所示:
那么當斷言失敗的時候是什么樣子呢?我們依據下兩圖進行相應的修改:
在此執行的結果如下圖所示,通過點擊對應的斷言,可查看斷言的詳細信息:
當然您也可以添加斷言結果(右鍵單擊測試計划、線程組、HTTP請求,以此選擇【添加/監聽器/斷言結果】添加即可,注意斷言結果的作用域,避免出現干擾),通過斷言結果更加直觀的查看所有的失敗斷言匯總信息,如下所示:
我們可以利用 JMeter 提供的各種斷言,進行請求響應結果的判定,其他在本文中沒有用到的斷言,各位小主們可先自行研究探索。同時,在實際的自動化應用過程中,JMeter 的斷言難免出現不滿足我們測試場景的時候,那么就需要我們針對不同的場景進行擴展了,通過已有斷言 BeanShell 和 BSF 去進行擴展,也許可以解決我們的問題。另外,JMeter 是用 Java 語言編寫的開源測試工具,當然也具有 Java 良好的擴展性,我們可以通過開發 JMeter 插件對其進行符合我們當前業務的定制擴展,以大大提升適應性、易用性。(PS:相關的簡單實戰示例,后續陸續更新,敬請期待!)
至此, JMeter學習-006-JMeter 斷言實例之一 - 響應斷言 順利完結,希望此文能夠給初學 JMeter 的您一份參考。
最后,非常感謝親的駐足,希望此文能對親有所幫助。熱烈歡迎親一起探討,共同進步。非常感謝! ^_^