drf开发常用工具函数总结(给请求封装通用参数、打印内置属性/方法、动态构建查找条件、yaml读取真正的环境变量、异常处理之自定义异常类和捕捉系统异常)

本文主要是介绍drf开发常用工具函数总结(给请求封装通用参数、打印内置属性/方法、动态构建查找条件、yaml读取真正的环境变量、异常处理之自定义异常类和捕捉系统异常),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

给请求封装通用参数

使用中间件

中间件可以在请求进入视图之前或响应返回给客户端之前执行代码。你可以创建一个自定义中间件来添加通用参数。

class CommonParamsMiddleware:def __init__(self, get_response):self.get_response = get_responsedef __call__(self, request):request.common_params = {'param1': 'value1','param2': 'value2',}response = self.get_response(request)return response

然后在你的settings.py中添加这个中间件:

MIDDLEWARE = [# ..'path.to.your.middleware.CommonParamsMiddleware',
]

在你的视图中,你可以通过request.common_params来访问这些参数。

使用自定义视图基类

如果你想要在视图层面添加通用参数,你可以创建一个自定义的视图基类,并在其他视图中继承它。

# views.py
from rest_framework.views import APIViewclass CommonParamsAPIView(APIView):def initialize_request(self, request, *args, **kwargs):request = super().initialize_request(request, *args, **kwargs)# 在这里添加你的通用参数request.common_params = {'param1': 'value1','param2': 'value2',}return request# 然后在你的其他视图中继承这个基类
class MyView(CommonParamsAPIView):def get(self, request, *args, **kwargs):# 使用通用参数common_params = request.common_params# ...

这样,任何继承CommonParamsAPIView的视图都会自动获得这些通用参数。

使用视图装饰器

如果你不想创建一个新的基类,你也可以使用装饰器来修改视图的行为。

# decorators.py
from functools import wrapsdef add_common_params(view_func):@wraps(view_func)def _decorated(view_instance, request, *args, **kwargs):# 在这里添加你的通用参数request.common_params = {'param1': 'value1','param2': 'value2',}return view_func(view_instance, request, *args, **kwargs)return _decorated# 在你的视图中使用装饰器
from rest_framework.decorators import api_view
from .decorators import add_common_params@api_view(['GET'])
@add_common_params
def my_view(request):# 使用通用参数common_params = request.common_params# ...

选择哪种方法取决于你的具体需求和偏好。中间件适用于全局的请求处理,自定义视图基类和装饰器则提供了更细粒度的控制。

打印所有内置属性/方法

def log_request(self, request):for attr in dir(request):if not attr.startswith('_'):# 检查属性是否存在,避免访问已经被移除的属性if hasattr(request, attr):try:value = getattr(request, attr, None)if not callable(value):print(f"{attr} : {value}")except Exception as e:print(f"Error accessing {attr}: {e}")

callable是一个内置函数,它用来检查一个对象是否是“可调用的”。一个对象如果是可调用的,意味着你可以像函数一样使用它,即可以在后面加上括号和参数来调用它。函数、方法、类以及实现了__call__方法的实例都是可调用的。

动态构建查找条件

def query_some(scope_type, scope_value):# 构建查询参数,其中 scope_type 是一个变量,它的值对应于模型中的字段名filter_kwargs = {scope_type: scope_value}objs = Project.objects.filter(**filter_kwargs).first()if not objs:return os.get("DEFAULT_RESULT")return objs.some_id

从yaml中读取真正的环境变量

# 必须参数
name: "MyNewSpace" 
platform_url: os.getenv("XXX")  #
app_code: os.getenv("YYY")# 非必需参数
desc: "Description of the new space."
def handle_env_var(value):# 尝试匹配 os.getenv("...") 格式并从环境变量中拿取真正的值if isinstance(value, str):match = re.match(r"os\.getenv\(['\"](.*?)['\"]\)", value)if match:return os.getenv(match.group(1))return valueelif isinstance(value, dict):for key, v in value.items():value[key] = handle_env_var(v)return valueelif isinstance(value, list):return [handle_env_var(v) for v in value]else:return value

动态构造序列化器内容:

class TASK_ID_KEY(Enum):TASK_ID = "task_tag"

class DemoQuerySerializer(serializers.Serializer):category = serializers.CharField(help_text="API对应的分类", max_length=255, default="test")# 动态添加字段def __init__(self, *args, **kwargs):super(DemoQuerySerializer, self).__init__(*args, **kwargs)self.fields[TASK_ID_KEY.TASK_ID.value] = serializers.CharField(help_text="task_tag_key", max_length=64)

异常处理

统一返回三段式处理函数

class ThirdPartResponse(Response):def __init__(self,result=True,message="",data=None,extra_data=None,status_code=status.HTTP_200_OK,code=None,):if message:message = "Error code: {}, {}".format(code, message)logger.error(message)standardized_data = {"result": result,"message": message,"data": data,}# 如果传入了extra_data,并且它是一个字典,则将其合并到输出字典中if extra_data and isinstance(extra_data, dict):standardized_data.update(extra_data)super().__init__(standardized_data, status_code)

自定义异常类

from rest_framework import statusclass TokenException(Exception):status_code = status.HTTP_400_BAD_REQUESTcode = 1100error = ""def __init__(self, error):self.error = errorclass TokenAPIError(TokenException):code = 1101error = "get access_token from apigw failed"def __init__(self, error):super().__init__(error)class TokenNotExist(TokenException):code = 1102error = "access_token does not exist in db"def __init__(self, error):super().__init__(error)class APIException(Exception):status_code = status.HTTP_200_OKcode = 1200error = ""def __init__(self, error):self.error = errorclass ThirdPartException(APIException):code = 1201error = "第三方请求错误"def __init__(self, error):super().__init__(error)

抛出和捕捉自定义异常

if not response["result"]:  msg = "原样返回上游平台响应:{}".format(str(response))  logger.error(msg)  raise ThirdPartyException(error=msg)
except IpHostFormatException as e:  return ThreePartResponse(status_code=e.status_code, message=e.error, code=e.code)  

捕捉系统异常

except Exception as e:  msg = "reformat_global_vars error: {}".format(str(e))  logger.error(msg)  return ThreePartResponse(message=msg)

正常返回三段式

return ThreePartResponse(data=serializer.data)

补充返回非三段式

return ThreePartResponse(data=data, extra_data=task_id_dict)

这篇关于drf开发常用工具函数总结(给请求封装通用参数、打印内置属性/方法、动态构建查找条件、yaml读取真正的环境变量、异常处理之自定义异常类和捕捉系统异常)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1067610

相关文章

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

【Altium】查找PCB上未连接的网络

【更多软件使用问题请点击亿道电子官方网站】 1、文档目标: PCB设计后期检查中找出没有连接的网络 应用场景:PCB设计后期,需要检查是否所有网络都已连接布线。虽然未连接的网络会有飞线显示,但是由于布线后期整板布线密度较高,虚连,断连的网络用肉眼难以轻易发现。用DRC检查也可以找出未连接的网络,如果PCB中DRC问题较多,查找起来就不是很方便。使用PCB Filter面板来达成目的相比DRC

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

十五.各设计模式总结与对比

1.各设计模式总结与对比 1.1.课程目标 1、 简要分析GoF 23种设计模式和设计原则,做整体认知。 2、 剖析Spirng的编程思想,启发思维,为之后深入学习Spring做铺垫。 3、 了解各设计模式之间的关联,解决设计模式混淆的问题。 1.2.内容定位 1、 掌握设计模式的"道" ,而不只是"术" 2、 道可道非常道,滴水石穿非一日之功,做好长期修炼的准备。 3、 不要为了

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

ABAP怎么把传入的参数刷新到内表里面呢?

1.在执行相关的功能操作之前,优先执行这一段代码,把输入的数据更新入内表里面 DATA: lo_guid TYPE REF TO cl_gui_alv_grid.CALL FUNCTION 'GET_GLOBALS_FROM_SLVC_FULLSCR'IMPORTINGe_grid = lo_guid.CALL METHOD lo_guid->check_changed_data.CALL M

ROS话题通信流程自定义数据格式

ROS话题通信流程自定义数据格式 需求流程实现步骤定义msg文件编辑配置文件编译 在 ROS 通信协议中,数据载体是一个较为重要组成部分,ROS 中通过 std_msgs 封装了一些原生的数据类型,比如:String、Int32、Int64、Char、Bool、Empty… 但是,这些数据一般只包含一个 data 字段,结构的单一意味着功能上的局限性,当传输一些复杂的数据,比如:

java中查看函数运行时间和cpu运行时间

android开发调查性能问题中有一个现象,函数的运行时间远低于cpu执行时间,因为函数运行期间线程可能包含等待操作。native层可以查看实际的cpu执行时间和函数执行时间。在java中如何实现? 借助AI得到了答案 import java.lang.management.ManagementFactory;import java.lang.management.Threa