介面自動化測試

IT小學生蔡坨坨發表於2021-01-10

介面自動化測試

(1)介面自動化測試的意義、前後端分離思想

介面自動化測試的優缺點:

優點:

  1. 測試複用性。

  2. 維護成本相對UI自動化低一些。

    為什麼UI自動化維護成本更高?
    因為前端頁面變化太快,而且UI自動化比較耗時(比如等待頁面元素的載入、新增等待時間、定位元素、操作元素、模擬頁面動作這些都需要時間)
    
    為什麼介面自動化維護成本較低?
    因為介面較穩定,介面的響應時間基本上都是秒級、毫秒級別的,速度快,並且介面自動化本身也可以做一些有關聯的操作、全流程的操作(比如:註冊 --> 登入 --> 修改個人資訊)。
    
  3. 迴歸方便。

  4. 可以執行更多更繁瑣的測試。自動化的一個明顯的好處是可以在較少的時間內執行更多的測試。

    優點1、優點3、優點4是介面自動化和UI自動化公有的優點。
    

缺點:

  1. 不能完全取代手工測試。(自動化永遠不能替代手工測試,只是提高測試效率)
  2. 手工測試比自動化測試發現的缺陷更多,自動化測試不容易發現新的BUG。

GET請求和POST請求的區別:

  1. GET請求一般是從後臺伺服器上獲取資料用於前端頁面的展示(例如:看到列表頁面等),POST請求是向伺服器傳送資料(登入、註冊、上傳檔案、釋出文章)。什麼時候用GET,什麼時候用POST取決於開發。無論用POST請求還是GET請求,都能完成對資料的增刪改查,分不同的請求方式更多的是一種約定。
  2. GET請求的請求引數是拼接在url後面的,只能以文字的形式傳遞引數,請求引數會顯示在位址列,資料長度受限於url的長度,傳遞的資料量小(4KB左右,不同瀏覽器會有差異),POST請求的請求引數是放在request body裡面,傳遞資料量大(預設8M),對資料長度也沒有要求。GET請求可以在瀏覽器中直接訪問,而POST請求只能藉助工具完成(比如:postman、jmeter)。
  3. GET請求速度快,安全性不高;POST請求一般用於像登入這種安全性要求高的場合,請求不會被快取,也不會保留在瀏覽器的歷史記錄中。
以前:get 查詢;post 新增;put 編輯;delete 刪除
現在:get 查詢;post 新增 + 編輯 + 刪除
或者:純post走天下

前後端分離

開發模式

以前老的方式:

  • 產品經理 / 領導 / 客戶提出需求(提出文字需求)

  • UI做出設計圖

  • 前端工程師做html頁面(使用者能看到的頁面)

  • 後端工程師將html頁面套成jsp頁面(前後端強依賴,後端必須要等到前端的html頁面做好才能套jsp。如果html發生變更,就很麻煩,開發效率低)

    比如雲商系統:http://101.133.169.100/yuns/index.php

  • 整合出現問題

  • 前端返工

  • 後端返工

  • 二次整合

  • 整合成功

  • 交付

新的方式:

  • 產品經理 / 領導 / 客戶提出需求(提出文字需求)
  • UI做出設計圖
  • 前後端約定介面 & 資料 & 引數
  • 前後端並行開發(無強依賴,可前後端並行開發,如果需求變更,只要介面 & 引數不變,就不用兩邊都修改程式碼,開發效率高)
  • 前後端整合
  • 前端頁面調整
  • 整合成功
  • 交付

?通過F12開啟瀏覽器開發者工具進行抓包,返回資料是json格式的就是前後端分離,返回時html頁面就是沒有前後端分離。

微服務的概念:

將大模組切分成小模組。減少程式碼的耦合度,從而降低模組與模組之間的影響。原先是一個jar包裡面包含所有模組,改一個模組就有可能影響其他模組,現在是將一個一個的模組都打成一個一個的jar包,模組與模組之間的互動通過介面,哪個模組出了問題,只需要修改那個模組的jar包,避免因為修改一個模組的程式碼導致其他模組出錯。

(2)Python requests框架講解

介面自動化requests環境搭建

介面自動化核心庫:requests

安裝requests庫的方法:

方法一:

命令列安裝,開啟cmd或者終端,輸入以下命令:

pip install requests -i https://pypi.douban.com/simple/

image-20210107214728817

方法二:
在pycharm中安裝,settings --> Project --> Project Interpreter --> 點選“+”號 --> 輸入request安裝

image-20210108203030662

測試環境是否ok

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/7 21:48

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!")

響應超時timeout

import requests

# V部落:http://182.92.178.83:8081/index.html
# 文章列表
url_v_article = "http://182.92.178.83:8081/article/all"
v_headers = {
    "Cookie": "studentUserName=ctt01; Hm_lvt_cd8218cd51f800ed2b73e5751cb3f4f9=1609742724,1609762306,1609841170,1609860946; adminUserName=admin; JSESSIONID=9D1FF19F333C5E25DBA60769E9F5248E"}
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是不定長的,根據定義的字典傳參
    # timeout超時,單位為秒
    # 通過設定超時時間,告訴requests在經過多久後停止等待響應
    result = requests.get(url_v_article, headers=v_headers, params=article_params, timeout=30)
    print(result.json())

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格式化:http://www.bejson.com/jsonviewernew/

url線上編碼轉換:https://www.w3cschool.cn/tools/index?name=urlencode_decode

(3)get、post、put、delete請求方式的自動化實現

GET請求方式

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/7 21:48

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請求方式

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/9 22:51

import requests

url_v_login = "http://182.92.178.83: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://182.92.178.83:8081/admin/category/", headers=headers, data=payload)

DELETE請求方式

# 刪除欄目
result = self.requests.delete("http://182.92.178.83:8081/admin/category/" + “2010”, headers=headers)
print(result.json())  # {'status': 'success', 'msg': '刪除成功!'}
self.assertEqual("刪除成功!", result.json()["msg"])

(4)介面自動化測試過程中cookie的處理

手動傳入cookie的值(每次通過瀏覽器F12抓包,然後複製request header裡面的cookie)

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/7 22:25

import requests

# V部落查詢欄目
url_v_category = "http://182.92.178.83: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())

image-20210108222858329

cookie自動獲取

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/9 22:51

import requests

url_v_login = "http://182.92.178.83: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指令碼

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/9 23:26

import requests


def get_cookie(username, password):
    """通過考試系統學生登入獲取單個cookie"""

    url_login = "http://182.92.178.83: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
# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/9 23:23

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
......

(5)不同型別請求引數的處理

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/7 22:25

import requests

# 文章列表
url_v_article = "http://182.92.178.83: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())

image-20210108223334792

(6)結合Python+Requests+Unittest框架做介面自動化測試

unittest框架結構:

介面自動化測試

程式碼地址:https://github.com/itcaituotuo/unittest_api

if _name_ == '__main__':

if __name__ == '__main__'的意思是:

  • 當.py檔案被直接執行時,if __name__ == '__main__'下的程式碼塊將被執行;
  • 當.py檔案以模組形式被匯入時,if __name__ == '__main__'下的程式碼塊不被執行。

(7)介面自動化測試過程中高階斷言

閉環斷言(新增 --> 查詢 --> 修改 --> 查詢 --> 刪除 -->查詢)

    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://182.92.178.83:8081/article/", headers=headers, data=payload)
        # ②查詢文章
        url_v_article = "http://182.92.178.83: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://182.92.178.83:8081/article/", headers=headers, data=payload)
        # 編輯完,查詢文章
        url_v_article = "http://182.92.178.83: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://182.92.178.83: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://182.92.178.83:8081/article/dustbin", headers=headers, data=payload)
        print(result.json())
        act = result.json()["msg"]
        self.assertEqual(act, "刪除成功!")

(8)通過HTMLTestRunner.py生成視覺化HTML測試報告

HTMLTestRunner.py百度網盤連結:

連結:https://pan.baidu.com/s/1Oeh-p0d7sAJ2i7qEvycaVw
提取碼:p20c

# -*- coding:utf-8 -*-
# 作者:IT小學生蔡坨坨
# 時間:2021/1/10 13:45

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總結:

  • postman:介面功能測試

  • JMeter:介面效能測試

  • requests:介面自動化

  • ?三個的共同特點:都能完成介面功能測試。

相關文章