flask后端实践02-全局Response返回和异常处理

全局Response返回和异常处理

github

前言

主要谈在Python中,

  1. 如何进行统一Response处理。
  2. 如何进行全局异常处理。

定义全局Response

BizResponse

根据Flask所需参数,定义status,message,header

返回的header,我们很少关心

返回的status,按照业务返回即可。一般成功返回200。错误根据服务端错误和客户端错误自行确定,项目初期,可以区分下即可。

返回的message 即我们接口的返回体,一般返回Json类型格式

这里是我们重点关心的地方,要保证有统一的返回结构体。

代码详细请看

app/model/biz_response.py

思路

我们定义统一的biz_response来处理Flask的所有返回参数。

其中,对于参数message 要求必须实现dict_msg()这个函数

dict_msg()函数负责将出参映射成为希望看到的样子

定义装饰器

装饰器的感觉和java的注解类似,实现 dict_response

python 复制代码
def dict_response(cls):
    def dict_msg(cls):
        return {key: value for key, value in cls.__dict__.items()}

    cls.dict_msg = dict_msg
    return cls

@dataclass
@dict_response
class TestResponse:
    name: str
    age: int

通过装饰器dict_response, 为TestResponse增加一个dict_msg的方法。

上述的方法也可以手动实现

python 复制代码
class TestResponse:
    name: str
    age: int
    def dict_msg(cls):
      return {"name":cls.name, "age":cls.age}

额外学习:鸭子类型

python没有Java中的interface的概念。

python本身是支持鸭子类型的。和go中鸭子类型一样。所以不需要像java中定义interface。

python 复制代码
class Animal:
    def bark(cls):
        print("im a animal")

class Dog:
    def bark(cls):
        print("wang wang")

animal : Animal = Dog()
animal.bark() # wang wang
python3 复制代码
#biz_response.py
class BizResponse:
    def dict_msg(cls):
        return {
            "Response": cls.message.dict_msg()
        }

@dataclass
class TestResponse:
    id: str
    
    def dict_msg(cls):
        # 手动改为大写开头的格式
        return {
            "Id": cls.id,
        }

所以在BizResponse中,直接使用message的dict_msg()方法,不用关心具体的类型。

同时这也意味着我们所有的返回Response,都应该实现这个方法。
详细示例请看

到此为止,我们就能完全统一我们的返回结构体。

使用单元测试

使用unittest进行单元测试

https://docs.python.org/zh-cn/3/library/unittest.html

使用go test 的类似写法,使用xxx_test 进行命名。

定义业务异常

代码详细请查看 app/common/error.py

python 复制代码
#app/common/error.py
class ErrorCode(Enum):
    InternalError = "InternalError"
    InvalidParameter = "InvalidParameter"


class BizException(Exception):
    def __init__(cls, code: ErrorCode, message: str):
        cls.code = code
        cls.message = message


class Error:
    def __init__(cls, msg: str, code: str):
        cls.message = msg
        cls.code = code

    def dict_msg(cls):
        return {
            "Error": {
                "Message": cls.message,
                "Code": cls.code
            }
        }

定义业务异常的目的也是为了更好地通过错误代码错误,并判断错误原因。

个人倾向于使用string定义错误码。方便可读性。

定义全局异常处理

python 复制代码
@app.errorhandler(Exception)
def error_handler(e):
    """
    全局异常捕获
    """
    logger.error("error=%s", e)
    logger.error("traceback=%s", traceback.format_exc())
    if isinstance(e, BizException):
        message = e.message
        code = e.code.value
    else:
        message = f"error={e}"
        code = ErrorCode.InternalError.value
    response = BizResponse.fail(Error(message, code))
    return jsonify(response.dict_msg()), response.status, response.header
相关推荐
2402_8575893623 分钟前
Spring Boot新闻推荐系统设计与实现
java·spring boot·后端
子墨77724 分钟前
Flask之Hello world 详解
python
繁依Fanyi26 分钟前
旅游心动盲盒:开启个性化旅行新体验
java·服务器·python·算法·eclipse·tomcat·旅游
J老熊31 分钟前
Spring Cloud Netflix Eureka 注册中心讲解和案例示范
java·后端·spring·spring cloud·面试·eureka·系统架构
计算机编程-吉哥32 分钟前
计算机毕业设计 基于Python的个性化旅游线路推荐系统的设计与实现 Python+Django+Vue 前后端分离 附源码 讲解 文档
python·django·毕业设计·课程设计·毕业论文·计算机毕业设计选题·个性化旅游线路推荐系统
Benaso34 分钟前
Rust 快速入门(一)
开发语言·后端·rust
sco528235 分钟前
SpringBoot 集成 Ehcache 实现本地缓存
java·spring boot·后端
罔闻_spider36 分钟前
爬虫prc技术----小红书爬取解决xs
爬虫·python·算法·机器学习·自然语言处理·中文分词
python机器学习ML40 分钟前
机器学习K近邻算法——python详细代码解析(sklearn)(1)
python·机器学习·近邻算法·knn
原机小子1 小时前
在线教育的未来:SpringBoot技术实现
java·spring boot·后端