封装Request类

回顾

上次我们完善了登录注册相关页面,还有后端接口,算是从0到1完成了一个功能的编码工作。可能前端部分会讲的比较粗糙,因为第一可能是笔者造诣不够,第二也跟我们直接从现有的框架进行改造有关,很多东西不是从0写到1,而是从11.1,但是后面不一样:

后面的页面都是咱们自己写,自己用,从0写到1

安装requests库

终端输入pip3 install requests并执行。

构思后端Request相关接口

要知道,咱们这是一个接口测试平台(后续可能会集成UI自动化,但是这个笔者也还没有想好)。一个接口自动化平台,最核心的当然是对api的请求操作,所以咱们刻不容缓,加快进度,趁热打铁,来点干货吧。用requests来协助我们完成接口自动化请求。

我们新建一个文件: middleware/HttpClient.py

  1. import datetime
  2. import requests
  3. class Request(object):
  4. def __init__(self, url, session=False, **kwargs):
  5. self.url = url
  6. self.session = session
  7. self.kwargs = kwargs
  8. if self.session:
  9. self.client = requests.session()
  10. return
  11. self.client = requests
  12. def get(self):
  13. return self.request("GET")
  14. @staticmethod
  15. def get_elapsed(timer: datetime.timedelta):
  16. if timer.seconds > 0:
  17. return f"{timer.seconds}.{timer.microseconds // 1000}s"
  18. return f"{timer.microseconds // 100}ms"
  19. def request(self, method: str):
  20. status_code = 0
  21. elapsed = "-1ms"
  22. try:
  23. if method.upper() == "GET":
  24. response = self.client.get(self.url, **self.kwargs)
  25. elif method.upper() == 'POST':
  26. response = self.client.post(self.url, **self.kwargs)
  27. else:
  28. response = self.client.request(method, self.url, **self.kwargs)
  29. status_code = response.status_code
  30. if status_code != 200:
  31. return Request.response(False, status_code)
  32. elapsed = Request.get_elapsed(response.elapsed)
  33. data = response.json()
  34. return Request.response(True, 200, data, response.headers, response.request.headers, elapsed=elapsed)
  35. except Exception as e:
  36. return Request.response(False, status_code, msg=str(e), elapsed=elapsed)
  37. def post(self):
  38. return self.request("POST")
  39. @staticmethod
  40. def response(status, status_code=200, response=None, response_header=None,
  41. request_header=None, elapsed=None, msg="success"):
  42. request_header = {k: v for k, v in request_header.items()}
  43. response_header = {k: v for k, v in response_header.items()}
  44. return {
  45. "status": status, "response": response, "status_code": status_code,
  46. "response_header": response_header, "request_header": request_header,
  47. "msg": msg, "elapsed": elapsed,
  48. }

如果我是产品经理的话,那么postman就是我的原型图:

测试平台系列(11) 封装Request类 - 图1

讲解一下各个方法,首先这是一个Request请求类,他拥有核心方法: request,目前咱们暂时先只做到支持json类的请求,后续补全form, file等类型的请求。

其实这个类做的事情很简单,就是把requests相关的方法剥离了出来,封装了一层。

其中构造函数提供了一些选项,包括请求的信息,url,是否以session的方式请求等等,kwargs涵盖了requests原生的参数,只要你想传,你都可以传进来。

get_elapsed是根据postman为参照,对请求时间做的一个处理,如果大于1s的请求响应时间,那我们以为单位显示,否则以毫秒为单位显示。

response是构造返回结构对象,对本次请求的数据进行整理。

request就封装了requests库的核心操作,基本上属于原生处理,并且判断了http状态码。

编写request接口

新建controllers/request/http.py

  1. from flask import Blueprint
  2. from flask import jsonify
  3. from flask import request
  4. from app.middleware.HttpClient import Request
  5. req = Blueprint("request", __name__, url_prefix="/request")
  6. @req.route("/http", methods=['POST'])
  7. def http_request():
  8. data = request.get_json()
  9. method = data.get("method")
  10. if not method:
  11. return jsonify(dict(code=101, msg="请求方式不能为空"))
  12. url = data.get("url")
  13. if not url:
  14. return jsonify(dict(code=101, msg="请求地址不能为空"))
  15. body = data.get("body")
  16. headers = data.get("headers")
  17. r = Request(url, data=body, headers=headers)
  18. response = r.request(method)
  19. return jsonify(dict(code=0, data=response, msg="操作成功"))

其实和登录/注册接口都很相似,基本上就是创立了一个blueprint,前缀是/request,后续就是引入刚才的request类,进行http请求,最后返回response。

调整run.py

我们新建了一个蓝图,需要去run.py进行注册:

测试平台系列(11) 封装Request类 - 图2

  1. from app.controllers.request.http import req
  2. pity.register_blueprint(req)

测试平台系列(11) 封装Request类 - 图3

本节内容就到这里了,下一节咱们编写属于自己的第一个组件: postman。又是前端内容了,咱们做一个接近postman的页面即可,不需要多高大上,做东西要抓住用户的使用习惯。毕竟咱们不是🍎,它的理念是让用户去适应它。

后端代码地址: https://github.com/wuranxu/pity

前端代码地址: https://github.com/wuranxu/pityWeb