luffy后台配置之二次封装
封装logger
项目都要记录日志
- 日志都可以打印到控制台
- 日志可以写到日志文件中
- 日志存到某个库中
- 所有项目日志统一管理 sentry:django写的服务,收集日志的,可以展示 开源的
项目中print都调整为logger.info()
以后项目上线,只需要调整日志级别,低级别的日志就不打印了,于是日志输出不用删掉
操作步骤
- django中集成日志步骤,django使用的就是python内置的日志模块
第一步:在配置文件中加入日志配置 ---大字典
# dev.py
LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'formatters': { 'verbose': { 'format': '%(levelname)s %(asctime)s %(module)s %(lineno)d %(message)s' }, 'simple': { 'format': '%(levelname)s %(module)s %(lineno)d %(message)s' }, }, 'filters': { 'require_debug_true': { '()': 'django.utils.log.RequireDebugTrue', }, }, 'handlers': { 'console': { # 实际开发建议使用WARNING 'level': 'INFO', 'filters': ['require_debug_true'], 'class': 'logging.StreamHandler', 'formatter': 'simple' }, 'file': { # 实际开发建议使用ERROR 'level': 'WARNING', 'class': 'logging.handlers.RotatingFileHandler', # 日志位置,日志文件名,日志保存目录必须手动创建,注:这里的文件路径要注意BASE_DIR代表的是小luffyapi 'filename': os.path.join(os.path.dirname(BASE_DIR), "logs", "luffy.log"), # 日志文件的最大值,这里我们设置300M 'maxBytes': 300 * 1024 * 1024, # 日志文件的数量,设置最大日志数量为10 'backupCount': 10, # 日志格式:详细格式 'formatter': 'verbose', # 文件内容编码 'encoding': 'utf-8' }, }, # 日志对象 'loggers': { 'django': { 'handlers': ['console', 'file'], 'propagate': True, # 是否让日志信息继续冒泡给其他的日志处理系统 }, } }
第二步:在utils新建 common_logger.py ,得到日志对象
import logging logger = logging.getLogger('django')
第三步:在任意想用日志的地方,导入使用即可
from utils.common_logger import logger logger.info('info级别的日志') logger.error('error级别的日志')
示例:
# home下的view.py
from django.http import JsonResponse from utils.common_logger import logger def test_logger(request): logger.info('info级别的日志') logger.error('error级别的日志') return JsonResponse({'info':'日志测试'},json_dumps_params={'ensure_ascii': False})
# urls.py
from home import views urlpatterns = [ path('test_logger/',views.test_logger), ]
使用步骤:
第一步:在utils中新建 common_exceptions.py,写个异常处理函数
from rest_framework.views import exception_handler as drf_exception_handler from rest_framework.response import Response from utils.common_logger import logger # 只要走到这个函数中,一定是出异常了,所以要记录日志 def exception_handler(exc, context): # 1 记录日志 : 哪个ip地址,用户id是多少,访问哪个路径,执行哪个视图函数,出了什么错 request = context.get('request') view = context.get('view') ip = request.META.get('REMOTE_ADDR') user_id = request.user.pk path = request.get_full_path() response = drf_exception_handler(exc, context) if response: logger.warning('drf出了异常,异常是:%s' % str(exc)) # drf的异常已经处理了--->直接取detail 会有点小小的问题,碰到再解决 res = Response({'code': 999, 'msg': response.data.get('detail', '服务器异常,请联系系统管理员')}) else: # djagno的异常,咱们要处理 logger.error('用户【%s】,ip地址为【%s】,访问地址为【%s】,执行视图函数为【%s】,出错是【%s】' % (user_id, ip, path, str(view), str(exc))) res = Response({'code': 888, 'msg': '服务器异常,请联系系统管理员'}) return res
第二步:配置配置文件
- 以后只要出了异常,都会走咱们的函数
REST_FRAMEWORK = { 'EXCEPTION_HANDLER': 'utils.exception.exception_handler', }
from rest_framework.views import APIView from rest_framework.response import Response from rest_framework.exceptions import APIException class TestException(APIView): def get(self,request): raise APIException('drf抛出异常') return Response('ok')
# urls.py
from home import views urlpatterns = [ path('test_exception/',views.TestException.as_view()) ]
# return APIResponse()----------前端收到-->{code:100,msg:成功} # return APIResponse(token=xsdse,username=lqz)----------前端收到-->{code:100,msg:成功,token:xsdse,username:lqz} # return APIResponse(data=[{},{},{}])----------前端收到-->{code:100,msg:成功,data:[{},{},{}]} # return APIResponse(code=101,msg=失败)----------前端收到-->{code:101,msg:失败} # return APIResponse(headers={'xx':'xx'})----------前端收到-->{code:100,msg:成功},但是响应头中有xx=xx # return APIResponse(status=201)----------前端收到-->{code:100,msg:成功},但是响应状态码是201
对Response进行封装,封装后,code,msg可以不传,不传就用默认的
使用步骤
第一步:在utils下新建common_response.py,封装APIResponse
from rest_framework.response import Response class APIResponse(Response): def __init__(self, code=100, msg='成功', status=None, headers=None, **kwargs): data = {'code': code, 'msg': msg} if kwargs: # 有值,说明,传了 除了code msg status headers 以外的,咱们都要返回给前端,放到这个data中 data.update(kwargs) # Response(data=data,status=status,headers=headers) super().__init__(data=data, status=status, headers=headers) # 不要return,可以这样做 self.data=data self.status=status
第二步:导入使用
- 视图函数的方法,返回时,都使用咱们自己的
示例
from rest_framework.response import Response from utils.common_response import APIResponse class TestAPIResponse(APIView): def get(self,request): return APIResponse(token='asdsadfsaf',username='xxx',addr='cfsafsaf')