接口测试(六)—— 接⼝对象封装的思想和实现

一、接⼝对象封装

1、解决的问题

  • 代码冗余度⾼(有⼤量重复代码)
  • 代码耦合度⾼
  • 代码维护成本⾼

2、核⼼思想:代码分层

  • 分层思想:
    • 将 普通⽅法实现的 ,分为 接⼝对象层测试脚本层
  • 接⼝对象层:
    • 对 接⼝ 进⾏封装。封装好之后,给 测试⽤例层 调⽤!
    • ⾯向对象 类 封装 实现。
  • 测试⽤例层:
    • 调⽤ 接⼝对象层 封装的⽅法,拿到 响应结果,断⾔进⾏接⼝测试!
    • 借助 unittest 框架实现

二、封装Tpshop商城

1、普通⽅式实现

import unittest
import requests

class TestTpshopLogin(unittest.TestCase):
    # 测试 登录成功
    def test01_login_ok(self):
        # 创建 session 实例
        session = requests.Session()

        # 使⽤实例,调⽤get 发送获取验证码请求
        session.get(url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=verify&r=0.21519623710645064")

        # 使⽤实例,调⽤post 发送登录请求
        resp = session.post(
            url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=do_login&t=0.7094195931397276",
            data={"username": "13012345678", "password": "123456", "verify_code":
"8888"})

        print("响应结果 =", resp.json())

        # 断⾔:
        self.assertEqual(200, resp.status_code)
        self.assertEqual(1, resp.json().get("status"))
        self.assertEqual("登陆成功", resp.json().get("msg"))

    # 测试 ⼿机号不存在
    def test02_tel_not_exists(self):
        # 创建 session 实例
        session = requests.Session()

        # 使⽤实例,调⽤get 发送获取验证码请求
        session.get(url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=verify&r=0.21519623710645064")

        # 使⽤实例,调⽤post 发送登录请求
        resp = session.post(
            url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=do_login&t=0.7094195931397276",
            data={"username": "13847834701", "password": "123456", "verify_code":
"8888"})
        print("响应结果 =", resp.json())

        # 断⾔:
        self.assertEqual(200, resp.status_code)
        self.assertEqual(-1, resp.json().get("status"))
        self.assertEqual("账号不存在!", resp.json().get("msg"))

    # 测试 密码错误
    def test03_pwd_err(self):
        # 创建 session 实例
        session = requests.Session()

        # 使⽤实例,调⽤get 发送获取验证码请求
        session.get(url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=verify&r=0.21519623710645064")

        # 使⽤实例,调⽤post 发送登录请求
        resp = session.post(
                url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=do_login&t=0.7094195931397276",
                data={"username": "13012345678", "password": "123456890",
"verify_code": "8888"})

        print("响应结果 =", resp.json())

        # 断⾔:
        self.assertEqual(200, resp.status_code)
        self.assertEqual(-2, resp.json().get("status"))
        self.assertEqual("密码错误!", resp.json().get("msg"))

2、登录接⼝对象层

封装思想:

  • 将 动态变化的数据,设计到⽅法的参数。
  • 将 固定不变的,直接写成⽅法实现。
  • 将 响应结果,通过返回值传出。

分析:

封装实现:
 

class TpshopLoginApi(object):
    # 发送验证码请求
    @classmethod
    def get_verify(cls, session):
        session.get(url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=verify&r=0.21519623710645064")

    # 发送登录请求
    @classmethod
    def login(cls, session, login_data):
        resp = session.post(
                url="http://tpshop-test.itheima.net/index.phpm=Home&c=User&a=do_login&t=0.7094195931397276",
                data=login_data)
        return resp

3、登录接⼝测试⽤例层

3.1 优化前:

import unittest
import requests
from tpshop_login_api import TpshopLoginApi

class TestTpshopLogin(unittest.TestCase):

    # 测试 登录成功
    def test01_login_ok(self):
        # 创建 session实例
        s = requests.Session()
        # ⽤实例,调⽤⾃⼰封装的 获取验证码 接⼝
        TpshopLoginApi.get_verify(s)
        # 调⽤ ⾃⼰封装的接⼝,登录
        abc = {"username": "13012345678", "password": "123456", "verify_code":
"8888"}
        resp = TpshopLoginApi.login(s, abc)
        print(resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(1, resp.json().get("status"))
        self.assertEqual("登陆成功", resp.json().get("msg"))

    # 测试 ⼿机号不存在
    def test02_tel_not_exists(self):
        # 创建 session实例
        s = requests.Session()
        # ⽤实例,调⽤⾃⼰封装的 获取验证码 接⼝
        TpshopLoginApi.get_verify(s)
        # 调⽤ ⾃⼰封装的接⼝,登录
        abc = {"username": "13048932745", "password": "123456", "verify_code":
"8888"}
        resp = TpshopLoginApi.login(s, abc)
        print(resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(-1, resp.json().get("status"))
        self.assertIn("账号不存在", resp.json().get("msg"))

        # 测试 密码错误
        def test03_pwd_err(self):
            # 创建 session实例
            s = requests.Session()
            # ⽤实例,调⽤⾃⼰封装的 获取验证码 接⼝
            TpshopLoginApi.get_verify(s)
            # 调⽤ ⾃⼰封装的接⼝,登录
            abc = {"username": "13012345678", "password": "123456789", "verify_code":
"8888"}
            resp = TpshopLoginApi.login(s, abc)
            print(resp.json())

            # 断⾔
            self.assertEqual(200, resp.status_code)
            self.assertEqual(-2, resp.json().get("status"))
            self.assertIn("密码错误", resp.json().get("msg"))

3.2 优化后

import unittest
import requests
from tpshop_login_api import TpshopLoginApi

# 封装 通⽤ 的 断⾔函数
def common_assert(self, resp, status_code, status, msg):
    self.assertEqual(status_code, resp.status_code)
    self.assertEqual(status, resp.json().get("status"))
    self.assertIn(msg, resp.json().get("msg"))

class TestTpshopLogin(unittest.TestCase):
    # 添加类属性
    session = None

    @classmethod
    def setUpClass(cls) -> None: # 在 类中 所 有 测试⽅法执⾏之前,⾃动执⾏1次。
        cls.session = requests.Session()

    def setUp(self) -> None: # 在 每个 测试⽅法执⾏之前,⾃动执⾏1次。
        # 调⽤ ⾃⼰封装的接⼝,获取验证码
        TpshopLoginApi.get_verify(self.session)

    # 测试 登录成功
    def test01_login_ok(self):
        # 调⽤ ⾃⼰封装的接⼝,登录
        data = {"username": "13012345678", "password": "123456", "verify_code":"8888"}
        resp = TpshopLoginApi.login(self.session, data)
        # 断⾔
        common_assert(self, resp, 200, 1, "登陆成功")

    # 测试 ⼿机号不存在
    def test02_tel_not_exists(self):
        # 调⽤ ⾃⼰封装的接⼝,登录
        data = {"username": "13048392845", "password": "123456", "verify_code":"8888"}
        resp = TpshopLoginApi.login(self.session, data)
        # 断⾔
        common_assert(self, resp, 200, -1, "账号不存在")

    # 测试 密码错误
    def test03_pwd_err(self):
        # 调⽤ ⾃⼰封装的接⼝,登录
        data = {"username": "13012345678", "password": "123456890", "verify_code":"8888"}
        resp = TpshopLoginApi.login(self.session, data)
        # 断⾔
        common_assert(self, resp, 200, -2, "密码错误")

三、封装iHRM登录

登录接⼝

1、普通⽅式实现

import unittest
import requests

class TestIhrmLogin(unittest.TestCase):
    # 登陆成功
    def test01_login_success(self):
        resp = requests.post(url="http://ihrm-test.itheima.net/api/sys/login",
                    json={"mobile": "13800000002", "password": "123456"})
        print(resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(True, resp.json().get("success"))
        self.assertEqual(10000, resp.json().get("code"))
        self.assertIn("操作成功", resp.json().get("message")

2、登录接⼝对象层

  • 思路:
    • 动态变化的,写⼊参数
    • 固定不变,直接写到⽅法实现内
    • 响应结果,通过返回值 return
  • 分析:
  • 封装实现:
    # 接⼝对象层
    import requests
    
    class IhrmLoginApi(object):
        @classmethod
        def login(cls, json_data):
            resp = requests.post(url="http://ihrm-test.itheima.net/api/sys/login",
                        json=json_data)
            return resp

3、登录接⼝测试⽤例层

import unittest
from ihrm_login_api import IhrmLoginApi

# 定义测试类
class TestIhrmLogin(unittest.TestCase):
    # 测试⽅法 - 登录成功
    def test01_login_success(self):
        # 调⽤ ⾃⼰封装 login
        login_data = {"mobile": "13800000002", "password": "123456"}
        resp = IhrmLoginApi.login(login_data)
        print("登录成功:", resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(True, resp.json().get("success"))
        self.assertEqual(10000, resp.json().get("code"))
        self.assertIn("操作成功", resp.json().get("message"))

    # 测试⽅法 - ⼿机号未注册
    def test02_mobile_not_register(self):
        # 调⽤ ⾃⼰封装 login
        login_data = {"mobile": "1384780932", "password": "123456"}
        resp = IhrmLoginApi.login(login_data)
        print("⼿机号未注册:", resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(False, resp.json().get("success"))
        self.assertEqual(20001, resp.json().get("code"))
        self.assertIn("⽤户名或密码错误", resp.json().get("message"))

    # 测试⽅法 - 密码错误
    def test03_pwd_error(self):
        # 调⽤ ⾃⼰封装 login
        login_data = {"mobile": "13800000002", "password": "890"}
        resp = IhrmLoginApi.login(login_data)
        print("密码错误:", resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(False, resp.json().get("success"))
        self.assertEqual(20001, resp.json().get("code"))
        self.assertIn("⽤户名或密码错误", resp.json().get("message"))

    # 测试⽅法 - ⼿机号为空
    def test04_mobile_is_none(self):
        # 调⽤ ⾃⼰封装 login
        login_data = {"mobile": None, "password": "123456"}
        resp = IhrmLoginApi.login(login_data)
        print("⼿机号为空:", resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(False, resp.json().get("success"))
        self.assertEqual(20001, resp.json().get("code"))
        self.assertIn("⽤户名或密码错误", resp.json().get("message"))

    # 测试⽅法 - 多参
    def test12_more_params(self):
        # 调⽤ ⾃⼰封装 login
        login_data = {"mobile": "13800000002", "password": "123456", "abc":"123"}
        resp = IhrmLoginApi.login(login_data)
        print("⼿机号为空:", resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(True, resp.json().get("success"))
        self.assertEqual(10000, resp.json().get("code"))
        self.assertIn("操作成功", resp.json().get("message"))

    # 测试⽅法 - ⽆参
    def test14_none_params(self):
        # 调⽤ ⾃⼰封装 login
        login_data = None
        resp = IhrmLoginApi.login(login_data)
        print("⼿机号为空:", resp.json())

        # 断⾔
        self.assertEqual(200, resp.status_code)
        self.assertEqual(False, resp.json().get("success"))
        self.assertEqual(99999, resp.json().get("code"))
        self.assertIn("抱歉,系统繁忙,请稍后重试", resp.json().get("message"))

4、封装断⾔⽅法

1. 创建 ⽂件 assert_util.py
2. 在 ⽂件内,定义 common_assert() 函数
3. 直接粘贴 unittest框架中的断⾔代码,修改参数。
4. 回到 unittest框架 实现的 测试脚本中, 调⽤该函数,实现断⾔,传递 实际参数。

# 定义 断⾔ 函数
# def common_assert():
# self.assertEqual(200, resp.status_code) # self / resp 报错
# self.assertEqual(True, resp.json().get("success"))
# self.assertEqual(10000, resp.json().get("code"))
# self.assertIn("操作成功", resp.json().get("message"))

def common_assert(self, resp, status_code, success, code, message):
    self.assertEqual(status_code, resp.status_code)
    self.assertEqual(success, resp.json().get("success"))
    self.assertEqual(code, resp.json().get("code"))
    self.assertIn(message, resp.json().get("message"))

# 调⽤演示
    common_assert(self, resp, 200, True, 10000, "操作成功")
    common_assert(self, resp, 200, False, 20001, "⽤户名或密码错误")
    common_assert(self, resp, 200, False, 99999, "抱歉,系统繁忙,请稍后重试")

三、Tpshop商城参数化

1、准备⼯作

分析 tpshop 商城测试⽤例:

提取每个测试⽤例 使⽤的 测试数据断⾔数据

封装函数,将 数据 转换为 元组列表

2、参数化步骤

1. 导包 from parameterized import parameterized
2. 在 通⽤测试⽅法,上⼀⾏,添加 @parameterized.expand()
3. 给 expand() 传⼊ [(),(),()](调⽤ 转换 [{},{},{}] --> [(),(),()] 的函数)
4. 修改 通⽤测试⽅法,添加形参,个数、顺序,与 [{},{},{}] 中 { } 内的所有 key 完全⼀⼀对应。
5. 在 通⽤测试⽅法内,使⽤形参。

from parameterized import parameterized

class TestTpshopLogin(unittest.TestCase):
    # 添加类属性
    session = None

    @classmethod
    def setUpClass(cls) -> None:
        cls.session = requests.Session()

    def setUp(self) -> None:
        # 调⽤ ⾃⼰封装的接⼝,获取验证码
        TpshopLoginApi.get_verify(self.session)

    # 测试 tpshop 登录
    @parameterized.expand(read_json_data())
    def test_tpshop_login(self, req_body, status_code, status, msg):
        resp = TpshopLoginApi.login(self.session, req_body)
        common_assert(self, resp, status_code, status, msg)

Python 参数化 参考:软件测试 —— Python(七)之UnitTest框架与测试报告 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

心上学事上练

感谢您的支持,让我们一起成长

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值