首页 > Python资料 博客日记
通过钩子函数+Traceid实现Flask链路追踪
2024-06-02 15:00:03Python资料围观200次
这篇文章介绍了通过钩子函数+Traceid实现Flask链路追踪,分享给大家做个参考,收藏Python资料网收获更多编程知识
背景
在flask web中我们通常需要一个traceid作为调用参数传递给全链路各个调用函数
- 需要针对一次请求创建一个唯一的traceid:这里用uuid去简化代替
- 我们需要保证traceid不被污染,在每个请求期间存在,在请求结束销毁且线程独立:这里通过flask中的g对象来存储线程内的数据
- 由于我们使用g对象来存储,那么当接口中发起新的请求时候,新请求会创建新的g对象,此时g对象为空,我们需要让traceid可以在多个请求中共享数据:这里通过请求头中增加traceid来传递
实现
首先定义二个主要函数
- 定义一个请求开始的时候需要调用的函数,用于初始化traceid或者获取上一个请求中的traceid以及其他一些相关请求参数
- 定义一个请求结束的时候需要调用的函数,用于请求结束的日志响应报文收尾记录。
import requests from flask import request, g import time from flask import Flask def get_uuid(): import uuid return str(uuid.uuid4()).replace('-', '') def trace_add_log_record(event_des='',msg_dict={},remarks=''): #trace_links_index每次调用+1 request.trace_links_index = request.trace_links_index + 1 logs = { 'traceid': g.traceid, 'trace_index': request.trace_links_index, 'event_des': event_des, 'msg_dict': msg_dict, 'remarks': remarks } print(logs) def trace_start_log_record_handler(): # 获取traceid,如果存在则使用,否则生成一个 if "traceid" in request.headers: g.traceid = request.headers['traceid'] else: g.traceid = get_uuid() # 初始化trace_links_index request.trace_links_index = 0 # 记录开始时间 request.start_time = time.time() log_msg = { 'headers': request.headers, 'url': request.url, 'method': request.method, "request_data": request.args if request.method == "GET" else request.get_json(), 'ip': request.headers.get("X-Real-IP") or request.remote_addr, 'start_time': request.start_time } # 记录日志 trace_add_log_record(event_des='start', msg_dict=log_msg) def trace_end_log_record_handler(reponse): # 记录结束时间 request.end_time = time.time() # 记录traceid到响应头 reponse.headers.add('traceid', g.traceid) log_msg = { "end_time" : request.end_time, "cost_time": request.end_time - request.start_time, "status_code": reponse.status_code, "headers": reponse.headers, "response_data": reponse.data.decode('utf-8') } # 记录日志 trace_add_log_record(event_des='end', msg_dict=log_msg)
接着,我们通过钩子函数去触发我们上述所写的两个重要函数
""" 写钩子函数在app中注册,还有一种写法 @app.before_request def before_request2(): print('before_request2') @app.after_request def after_request1(response): print('after_request1') return response """ #写钩子函数在app中注册 def register_handler(response): def before_request(): trace_start_log_record_handler() def after_request(response): trace_end_log_record_handler(reponse=response) return response # 注册钩子函数 response.before_request(before_request) response.after_request(after_request)
最后写测试接口进行测试
app = Flask(__name__) @app.route('/') def hello_world(): return 'Hello, World!' @app.route('/test') def test(): name = request.args.get('name') hello_world = requests.get('http://127.0.0.1:5000/', headers={'traceid': g.traceid}) return name + hello_world.text @app.route('/test2', methods=['POST']) def test2(): name = request.get_json().get('name') hello_world = requests.get('http://127.0.0.1:5000/', headers={'traceid': g.traceid}) return name + hello_world.text if __name__ == '__main__': register_handler(app) app.run(debug=True)
这样我们简单了完成了通过traceid把链路串联起来了
{'traceid': 'a5637579351c477a80090a88f5347088', 'trace_index': 1, 'event_des': 'start', 'msg_dict': {'headers': EnvironHeaders([('User-Agent', 'Apifox/1.0.0 (https://apifox.com)'), ('Content-Type', 'application/json'), ('Accept', '*/*'), ('Host', '127.0.0.1:5000'), ('Accept-Encoding', 'gzip, deflate, br'), ('Connection', 'keep-alive')]), 'url': 'http://127.0.0.1:5000/test?name=yetangjian', 'method': 'GET', 'request_data': ImmutableMultiDict([('name', 'yetangjian')]), 'ip': '127.0.0.1', 'start_time': 1717311086.757312}, 'remarks': ''} {'traceid': 'a5637579351c477a80090a88f5347088', 'trace_index': 1, 'event_des': 'start', 'msg_dict': {'headers': EnvironHeaders([('Host', '127.0.0.1:5000'), ('User-Agent', 'python-requests/2.25.1'), ('Accept-Encoding', 'gzip, deflate'), ('Accept', '*/*'), ('Connection', 'keep-alive'), ('Traceid', 'a5637579351c477a80090a88f5347088')]), 'url': 'http://127.0.0.1:5000/', 'method': 'GET', 'request_data': ImmutableMultiDict([]), 'ip': '127.0.0.1', 'start_time': 1717311086.7663064}, 'remarks': ''} {'traceid': 'a5637579351c477a80090a88f5347088', 'trace_index': 2, 'event_des': 'end', 'msg_dict': {'end_time': 1717311086.7663064, 'cost_time': 0.0, 'status_code': 200, 'headers': Headers([('Content-Type', 'text/html; charset=utf-8'), ('Content-Length', '13'), ('traceid', 'a5637579351c477a80090a88f5347088')]), 'response_data': 'Hello, World!'}, 'remarks': ''} {'traceid': 'a5637579351c477a80090a88f5347088', 'trace_index': 2, 'event_des': 'end', 'msg_dict': {'end_time': 1717311086.7683115, 'cost_time': 0.010999441146850586, 'status_code': 200, 'headers': Headers([('Content-Type', 'text/html; charset=utf-8'), ('Content-Length', '23'), ('traceid', 'a5637579351c477a80090a88f5347088')]), 'response_data': 'yetangjianHello, World!'}, 'remarks': ''}
版权声明:本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!
标签:
相关文章
最新发布
- 光流法结合深度学习神经网络的原理及应用(完整代码都有Python opencv)
- Python 图像处理进阶:特征提取与图像分类
- 大数据可视化分析-基于python的电影数据分析及可视化系统_9532dr50
- 【Python】入门(运算、输出、数据类型)
- 【Python】第一弹---解锁编程新世界:深入理解计算机基础与Python入门指南
- 华为OD机试E卷 --第k个排列 --24年OD统一考试(Java & JS & Python & C & C++)
- Python已安装包在import时报错未找到的解决方法
- 【Python】自动化神器PyAutoGUI —告别手动操作,一键模拟鼠标键盘,玩转微信及各种软件自动化
- Pycharm连接SQL Sever(详细教程)
- Python编程练习题及解析(49题)
点击排行
- 版本匹配指南:Numpy版本和Python版本的对应关系
- 版本匹配指南:PyTorch版本、torchvision 版本和Python版本的对应关系
- Python 可视化 web 神器:streamlit、Gradio、dash、nicegui;低代码 Python Web 框架:PyWebIO
- 相关性分析——Pearson相关系数+热力图(附data和Python完整代码)
- Anaconda版本和Python版本对应关系(持续更新...)
- Python与PyTorch的版本对应
- Windows上安装 Python 环境并配置环境变量 (超详细教程)
- Python pyinstaller打包exe最完整教程