当前位置: 首页 > news >正文

Pytest项目_day20(log日志)

Log日志

  • 优点:记录程序运行信息,方便定位问题

  • python日志模块logging,日志等级如下:

    • DEBUG
    • INFO(正常)
    • WARNING
    • ERROR(报错)
  • 示例代码如下:

import logging
import os.path
import timeroot_path = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
log_path = os.path.join(root_path, "log")class Logger:def __init__(self):# 定义日志位置和文件名self.logname = os.path.join(log_path, "{}.log".format(time.strftime("%Y-%m-%d")))# 定义一个日志容器self.logger = logging.getLogger("log")# 设置日志打印的级别# 让所有级别的日志都打印输出self.logger.setLevel(logging.DEBUG)# 创建日志输入的格式self.formater = logging.Formatter('[%(asctime)s][%(filename)s %(lineno)d][%(levelname)s]:%(message)s')# 创建日志处理器,用来存放日志文件self.filelogger = logging.FileHandler(self.logname, mode='a', encoding='utf8')# 创建日志处理器,在控制台打印self.console = logging.StreamHandler()# 设置控制台打印日志界别self.console.setLevel(logging.DEBUG)# 文件存放日志级别self.filelogger.setLevel(logging.DEBUG)# 文件存放日志格式self.filelogger.setFormatter(self.formater)# 控制台打印日志格式self.console.setFormatter(self.formater)# 将日志输出渠道添加到日志收集器中self.logger.addHandler(self.filelogger)self.logger.addHandler(self.console)logger = Logger().logger
if __name__ == '__main__':logger.debug("打印debug日志")logger.info("打印info日志")logger.warning("打印warning日志")logger.error("打印error日志")
  • 输出结果如下:
    在这里插入图片描述
    在这里插入图片描述
  • 我们可以在测试用例中引入日志,如下:
import pytestfrom utils.log_util import logger
from utils.yaml_util import func_yaml
from utils.read_data import read_data@pytest.mark.parametrize("data", read_data()["person"])
def test_person(data):logger.info("开始执行test_person测试用例")result = func_yaml(data)print(result)logger.info("test_person测试用例执行完毕")
  • 输出结果如下:
    在这里插入图片描述
    在这里插入图片描述

完整的get、post、delete、put请求优化

  • 我们需要将这些请求都加上日志,需要记录url和参数信息,为了简化操作,放在最后一个函数处
  • 代码如下:
import jsonimport requestsfrom utils.log_util import logger
from utils.read import get_dataapi_root_url = get_data.read_ini()["host"]["api_sit_url"]class RestClient:def __init__(self):self.api_root_url = api_root_urldef get(self, url, **kwargs):return self.request(self.api_root_url + url, "GET", **kwargs)def post(self, url, **kwargs):return self.request(self.api_root_url + url, "POST", **kwargs)def put(self, url, **kwargs):return self.request(self.api_root_url + url, "PUT", **kwargs)def delete(self, url, **kwargs):return self.request(self.api_root_url + url, "DELETE", **kwargs)def request(self, url, method, **kwargs):self.request_log(url, method, **kwargs)if method == "GET":return requests.get(url, **kwargs)if method == "POST":return requests.post(url, **kwargs)if method == "PUT":return requests.put(url, **kwargs)if method == "DELETE":return requests.delete(url, **kwargs)def request_log(self, url, method, **kwargs):logger.info("接口请求的地址>>>>{}".format(url))logger.info("接口请求的方法>>>>{}".format(method))data = dict(**kwargs).get("data")json_data = dict(**kwargs).get("json")params = dict(**kwargs).get("params")headers = dict(**kwargs).get("headers")if data is not None:logger.info("接口请求的data参数>>>>\n{}".format(json.dumps(data, indent=2)))if json_data is not None:logger.info("接口请求的json参数>>>>\n{}".format(json.dumps(json_data, indent=2)))if params is not None:logger.info("接口请求的params参数>>>>\n{}".format(json.dumps(params, indent=2)))if headers is not None:logger.info("接口请求的headers参数>>>>\n{}".format(json.dumps(headers, indent=2)))
  • 同时,我们使用conftest.py来设置前置和后置操作
import pytestfrom utils.log_util import logger@pytest.fixture(scope="function", autouse=True)
def func():logger.info("开始执行测试用例~")yieldlogger.info("测试用例执行完毕~")
  • 最后的效果如下:
    在这里插入图片描述

请求返回的日志优化

  • 我们可以在请求的返回函数中加上log日志记录
  • 还可以定义一个类,用于拼接状态码和body
    • 注意,此处可以使用python的类属性动态赋值特性,不需要在类中显式定义success和body,而是直接赋值,python会自动创建该类属性
  • 代码如下:
# response_util
import jsonfrom core.ResultBase import ResultResponse
from utils.log_util import loggerdef process_response(response):if response.status_code ==200 or response.status_code ==201:ResultResponse.success = TrueResultResponse.body = response.json()logger.info("接口返回的内容>>>>: "+json.dumps(response.json(), ensure_ascii=False))else:ResultResponse.success = Falselogger.info("接口状态码不是2开头,请检查")return ResultResponse# ResultBase
class ResultResponse:pass# Testcase
def test_mobile():print("测试手机归属地get请求")param = get_data.read_data()["mobile_belong_dir"]result = mobile_query(param)assert result.success == Trueassert result.body['status'] == 0assert result.body['msg'] == "ok"assert result.body['result']["shouji"] == "13906191369"assert result.body['result']["province"] == "浙江"assert result.body['result']["city"] == "杭州"assert result.body['result']["company"] == "中国移动"assert result.body['result']["areacode"] == "0571"
  • 输出结果如下:
    在这里插入图片描述

小优化

  • 我们可以使用requests.request()来简化代码,直接传入method,从而可以自动切换请求类型
  • 为了能够保持登陆状态,我们还可以使用session来进行请求,代码如下:
import jsonimport requestsfrom utils.log_util import logger
from utils.read import get_dataapi_root_url = get_data.read_ini()["host"]["api_sit_url"]class RestClient:def __init__(self):self.api_root_url = api_root_url# 新增self.session = requests.Session()def get(self, url, **kwargs):return self.request(url, "GET", **kwargs)def post(self, url, **kwargs):return self.request(url, "POST", **kwargs)def put(self, url, **kwargs):return self.request(url, "PUT", **kwargs)def delete(self, url, **kwargs):return self.request(url, "DELETE", **kwargs)def request(self, url, method, **kwargs):self.request_log(url, method, **kwargs)# if method == "GET":#     return requests.get(url, **kwargs)# if method == "POST":#     return requests.post(url, **kwargs)# if method == "PUT":#     return requests.put(url, **kwargs)# if method == "DELETE":#     return requests.delete(url, **kwargs)# 新增response = self.session.request(method, self.api_root_url + url, **kwargs)return responsedef request_log(self, url, method, **kwargs):logger.info("接口请求的地址>>>>{}".format(url))logger.info("接口请求的方法>>>>{}".format(method))data = dict(**kwargs).get("data")json_data = dict(**kwargs).get("json")params = dict(**kwargs).get("params")headers = dict(**kwargs).get("headers")if data is not None:logger.info("接口请求的data参数>>>>\n{}".format(json.dumps(data, indent=2)))if json_data is not None:logger.info("接口请求的json参数>>>>\n{}".format(json.dumps(json_data, indent=2)))if params is not None:logger.info("接口请求的params参数>>>>\n{}".format(json.dumps(params, indent=2)))if headers is not None:logger.info("接口请求的headers参数>>>>\n{}".format(json.dumps(headers, indent=2)))
http://www.lryc.cn/news/626817.html

相关文章:

  • PyTorch API 2
  • GPT-5 上线风波深度复盘:从口碑两极到策略调整,OpenAI 的变与不变
  • C++开发/Qt开发:单例模式介绍与应用
  • 拓扑排序判断环 P1347 排序题解
  • 第二十七天:游戏组队问题
  • 跨平台 RTSP/RTMP 播放器工程化实践:低延迟与高稳定性的挑战与突破
  • Redisson最新版本(3.50.0左右)启动时提示Netty的某些类找不到
  • pip 安装常见错误及实例化解决办法大全
  • Tomcat部署与HTTP协议详解
  • 凸问题-非凸问题-非凸模型
  • 第十四届“中国软件杯”大赛晋级现场总决赛名单公布
  • PyTorch API 6
  • 单片机通信协议核心关系梳理笔记(UART/USART/232/485/SPI/12C/LIN/BLE/WIFI)
  • Spring Boot 3.4.x 性能优化实战:用 Undertow 替换 Tomcat 全指南​
  • JavaScript 性能优化实战:从原理到落地的完整指南
  • 【OneAI】使用Rust构建的轻量AI网关
  • 【Axure高保真原型】拖拉拽画圆
  • JavaScript 性能优化实战(易懂版)
  • 实验8.20
  • LeetCode 刷题【47. 全排列 II】
  • 一种融合AI与OCR的施工许可证识别技术,提升工程监管效率,实现自动化、精准化处理。
  • 【解决方案】powershell自动连接夜神adb端口
  • 深入解析RAGFlow六阶段架构
  • 结合SAT-3D,运动+饮食双重养腰新方式
  • 十二,数据结构-链表
  • Linux用30秒部署Nginx+Tomcat+Mysql+Jdk1.8环境
  • 学习嵌入式的第二十二天——数据结构——双向链表
  • 为6G和超快光谱铺路,《Nature Communications》发布新型太赫兹光芯片,实现多通道信号操纵
  • AI 效应: GPT-6,“用户真正想要的是记忆”
  • 书籍推荐|《Computational Methods for Rational Drug Design》574页