fastapi的简单实战,且用uvicorn将日志同时输出到控制台和日志文件中

简单描述

  • fastapi的简单实战,且用uvicorn将日志同时输出到控制台和日志文件中

main.py

import signal
import sys
from contextlib import asynccontextmanagerfrom fastapi import FastAPI
import uvicorn
from fastapi.staticfiles import StaticFilesfrom settings import settings
from routers.xxx import xxx_router
from common.logging import logger@asynccontextmanager
async def lifespan(app: FastAPI):# Code to run on startuplogger.info("Starting up...")# You can initialize resources here (e.g., database connections)yield# Code to run on shutdownlogger.info("Shutting down...")# Clean up resources hereapp = FastAPI(lifespan=lifespan,debug=settings.debug,title="xxx平台",description='xxx',version='1.0.0',docs_url='/docs',redoc_url='/redoc',)app.mount("/static", StaticFiles(directory=settings.static_dir), name="static")app.include_router(xxx_router, prefix="/api/xxx", tags=["xxx"])@app.get("/")
def read_root():return {"data": "welcome to xxx center"}def signal_handler(sig, frame):print(f'Received Ctrl+C! {sig} exiting...')# 在这里执行任何必要的清理工作sys.exit()def run_http():signal.signal(signal.SIGINT, signal_handler)signal.signal(signal.SIGTERM, signal_handler)uvicorn.run(app="main:app", host=settings.server_ip, port=settings.server_port,log_config="uvicorn_config.json",log_level=settings.log_level, workers=settings.workers)if __name__ == "__main__":run_http()

settings/settings.py

import json
import os
import sys# 获取可执行文件所在的目录路径
if getattr(sys, 'frozen', False):# 如果程序是被打包成了单一文件,这个条件是TrueBASE_DIR = os.path.dirname(sys.executable)
else:# 如果程序是直接运行的.py文件,这个条件是TrueBASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))class BaseConfig(object):def __init__(self):self.config_path = self.get_config_path()self.config = self.get_config()@staticmethoddef get_config_path():config_path = os.path.join(BASE_DIR, 'config.json')if os.path.exists(config_path):return config_pathprint(f'config file {config_path} not exists')def get_config(self):try:with open(self.config_path, 'r', encoding="utf-8") as f:return json.load(f)except FileNotFoundError:print(f'config file {self.config_path} not exists')except json.decoder.JSONDecodeError:print(f'config file {self.config_path} format error')class ProjectConfig(BaseConfig):"""项目中的配置信息"""def get_server(self):"""后台服务配置信息"""try:return self.config["server"]["ip"], self.config["server"]["port"]except Exception as e:print(f'http server section not exists: {e}')def get_debug_mode(self):"""是否开启debug模式"""try:return True if self.config.get("debug_mode") == "true" else Falseexcept Exception as e:print(f'debug_mode section not exists: {e}')def get_workers(self):"""开启进程数"""try:return self.config["workers"]except Exception as e:print(f'process workers section not exists: {e}')def get_log(self):"""日志配置信息"""try:return self.config["logfile"]["level"], self.config["logfile"]["dir"], self.config["logfile"]["max_age"]except Exception as e:print(f'log file section not exists: {e}')def get_mysql(self):"""mysql数据库配置信息"""try:return self.config["mysql"]["default"]except Exception as e:print(f'mysql section not exists: {e}')def get_identify(self):"""识别服务配置信息"""try:return self.config["identify"]["ip"], self.config["identify"]["port"], self.config["identify"]["api_path"]except Exception as e:print(f'identify server section not exists: {e}')def get_white_list(self):"""请求白名单列表"""try:return self.config["white_list"]except Exception as e:print(f'white list section not exists: {e}')project_config = ProjectConfig()# 后台服务
server_ip, server_port = project_config.get_server()# 是否开启debug
debug = project_config.get_debug_mode()# 启动进程数
workers = project_config.get_workers()# 日志
log_level, log_dir, log_max_age = project_config.get_log()
log_dir = os.path.join(BASE_DIR, log_dir)
if not os.path.exists(log_dir):os.makedirs(log_dir)# mysql数据库
mysql_url = project_config.get_mysql()# 识别服务
identify_ip, identify_port, identify_api_path = project_config.get_identify()# 访问白名单列表
white_list = project_config.get_white_list()# 访问静态文件系统目录
static_dir = os.path.join(BASE_DIR, 'myfiles')
if not os.path.exists(static_dir):os.makedirs(static_dir)

common/logging.py

import os
import logging
from logging.handlers import RotatingFileHandlerfrom uvicorn.config import LOG_LEVELSfrom settings import settings# 创建一个RotatingFileHandler,最多备份5个日志文件,每个日志文件最大5M
file_handler = RotatingFileHandler(os.path.join(settings.log_dir, "uvicorn.log"), encoding='UTF-8', maxBytes=5*1024*1024, backupCount=5)
file_handler.setLevel(LOG_LEVELS[settings.log_level])
file_handler.setFormatter(logging.Formatter('%(asctime)s - %(levelname)s - %(name)s  %(message)s'))logging.basicConfig(handlers=[file_handler])# 获取Uvicorn的logger并添加文件处理器
logger = logging.getLogger("uvicorn")
logger.setLevel(LOG_LEVELS[settings.log_level])
# logger.addHandler(file_handler)  # 没打印日志到文件中,还没找到原因

uvicorn_config.json

{"version": 1,"disable_existing_loggers": false,"formatters": {"default": {"()": "uvicorn.logging.DefaultFormatter","fmt": "%(asctime)s - %(levelprefix)s %(message)s","use_colors": null},"access": {"()": "uvicorn.logging.AccessFormatter","fmt": "%(asctime)s - %(levelprefix)s %(client_addr)s - \"%(request_line)s\" %(status_code)s"}},"handlers": {"default": {"formatter": "default","class": "logging.StreamHandler","stream": "ext://sys.stderr"},"access": {"formatter": "access","class": "logging.StreamHandler","stream": "ext://sys.stdout"}},"loggers": {"fastapi": {"handlers": ["default"],"level": "DEBUG"},"uvicorn": {"handlers": ["default"],"level": "DEBUG"},"uvicorn.error": {"level": "INFO"},"uvicorn.access": {"handlers": ["access"],"level": "INFO","propagate": false}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/841594.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

生成式AI导论2024-李宏毅

生成式AI导论2024-李宏毅 第0讲: 课程说明第1讲:生成式AI是什么第2講:今日的生成式人工智慧厲害在哪裡?從「工具」變為「工具人」 第0讲: 课程说明 生成式AI的入门课程 第1讲:生成式AI是什么 生成式人…

python调用阿里云通义千问(q-wen-max)API-创建智能体Agent

文章目录 Assistant API简介创建和使用Assistant API1、调用Assistant API夸克搜索回答问题2、Agent智能体构建Assistant API简介 百炼Assistant API能够让用户定制化构建一个assistant,这个assistant支持多种不同的指令(instruction)和描述(prompt),并且可以使用各类工…

AI预测福彩3D采取888=3策略+和值012路一缩定乾坤测试5月26日预测第2弹

昨天的8883大底成功命中,但是由于昨天杀了对子,结果昨天开了对子,导致最终与中奖号码擦肩而过。今天继续基于8883的大底,使用尽可能少的条件进行缩号,同时,今天将准备两套方案,一套是我自己的条…

队列——顺序存储

核心思路: 1、使用顺序存储的方式定义队列时,使用数组存储队列元素,然后声明两个int类型的指针——rear和front,分别指向队尾元素的下一个位置和队头元素的位置。 2、初始化队列时,队列的首尾指针都指向0 。 3、当队列…

2021 年 3 月青少年软编等考 C 语言二级真题解析

目录 T1. 与指定数字相同的数的个数思路分析 T2. 合法 C 标识符思路分析 T3. 计算鞍点思路分析 T4. 谁考了第 k 名思路分析 T5. 石头剪刀布思路分析 T1. 与指定数字相同的数的个数 输出一个整数序列中与指定数字相同的数的个数。 时间限制:1 s 内存限制&#xff1…

【CSS】计算属性 calc 函数

CSS 中的 calc() 函数是用于动态计算数值的函数。它可以在 属性的值中执行基本的数学运算,包括加法、减法、乘法和除法,以及使用 CSS 单位进行计算。 calc() 函数的语法如下: calc(expression) 其中 expression 是包含数学运算和 CSS 单位表…

英语学习笔记28——Where are they?

Where are they? 他们在哪里? 课文部分

【NumPy】关于numpy.median()函数,看这一篇文章就够了

🧑 博主简介:阿里巴巴嵌入式技术专家,深耕嵌入式人工智能领域,具备多年的嵌入式硬件产品研发管理经验。 📒 博客介绍:分享嵌入式开发领域的相关知识、经验、思考和感悟,欢迎关注。提供嵌入式方向…

贝叶斯算法:机器学习中的“黄金法则”与性能提升之道

👀传送门👀 🔍机器学习概述🍀贝叶斯算法原理🚀贝叶斯算法的应用✨文本分类✨医疗系统 💖贝叶斯算法优化✨贝叶斯算法优化的主要步骤✨贝叶斯算法优化的优点✨贝叶斯算法优化的局限性 🚗贝叶斯算…

二维前缀异或和,1738. 找出第 K 大的异或坐标值

一、题目 1、题目描述 给你一个二维矩阵 matrix 和一个整数 k &#xff0c;矩阵大小为 m x n 由非负整数组成。 矩阵中坐标 (a, b) 的 值 可由对所有满足 0 < i < a < m 且 0 < j < b < n 的元素 matrix[i][j]&#xff08;下标从 0 开始计数&#xff09;执行…

「贪心算法」摆动序列

力扣原题链接&#xff0c;点击跳转。 如果连续数字之间的差严格地在正数和负数之间交替&#xff0c;则数字序列称为摆动序列。第一个差&#xff08;如果存在的话&#xff09;可能是正数或负数。仅有一个元素或者含两个不等元素的序列也视作摆动序列。简单来说&#xff0c;摆动…

【iOS开发】—— KVC

【iOS开发】—— KVC 一. KVC的定义key和keyPath的区别用法&#xff1a; 批量复制操作字典模型相互转化KVC的其他方法 KVC原理赋值原理取值原理 一. KVC的定义 KVC&#xff08;Key-value coding&#xff09;键值编码&#xff0c;就是指iOS的开发中&#xff0c;可以允许开发者通…

不用从头训练,通过知识融合创建强大的统一模型

在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;大型语言模型&#xff08;LLMs&#xff09;的开发和训练是一个复杂且成本高昂的过程。数据需求是一个主要问题&#xff0c;因为训练这些模型需要大量的标注数据来保证其准确性和泛化能力&#xff1b;计算资源也是一个…

Java学习路线思维导图

目录 Java学习流程1.学习大纲2.Java开发中常用的DOS命令 Java入门学习思维导图 Java学习流程 通过大纲了解学习的重点&#xff0c;通过目录依次深入【注&#xff1a;Java环境的搭建百度&#xff0c;提升自己百度的能力】 1.学习大纲 学习流程如下&#xff1a; Java基础语法 …

网络安全架构之零信任安全

网络安全架构之零信任安全 文章目录 网络安全架构之零信任安全零信任安全时代背景安全世界“新旧时代”各种攻击风险层出不穷网络安全边界逐渐瓦解内外部威胁愈演愈烈 零信任架构零信任的理念在不可信的网络环境下重建信任构建自适应内生安全机制以身份为基石业务安全访问持续信…

Linux服务的简介与分类

服务的简介与分类 服务的分类 查询已安装的服务和区分服务 #列出所有rpm包默认安装服务的自启动状态 [rootlocalhost ~]# chkconfig --list atd atd 0:关闭 1:关闭 2:关闭 3:启用 4:启用 5:启用 6:关闭 [rootlocalhost ~]# chkconfig --list sshd sshd …

SpringBoot项目中访问HTML页面

在这种情况下&#xff0c;如果你要访问静态页面&#xff0c;肯定是不能正确访问的&#xff1a;会出现如下错误&#xff1a; 那么&#xff0c;此时&#xff0c;你应该&#xff1a; 静态资源映射&#xff1a; import org.springframework.context.annotation.Configuration; im…

command not found: wire 解决方案【学习笔记,不作教程】

command not found: wire command not found: wire command not found: wire go get github.com/google/wire/cmd/wirego install github.com/google/wire/cmd/wirelatest再次在 /bubble/cmd/bubble目录下执行wire wire wire: bubble/cmd/bubble: wrote /Users/zhengshijie/go…

音视频开发5 补充 - Nginx搭建rtmp流媒体服务器,目的是让ffmpeg 可以直播推流

直播推流 ffmpeg -re -i out.mp4 -c copy flv rtmp://server/live/streamName -re, 表示按时间戳读取文件 参考&#xff1a; Nginx 搭建 rtmp 流媒体服务器 (Ubuntu 16.04) https://www.jianshu.com/p/16741e363a77 第一步 准备工作 安装nginx需要的依赖包 打开 ubutun 终端…

[emailprotected](9)属性默认值和类型验证

目录 1&#xff0c;属性默认值1.1&#xff0c;函数组件1.2&#xff0c;类组件 2&#xff0c;属性类型验证2.1&#xff0c;和默认值的关系2.2&#xff0c;使用2.3&#xff0c;举例说明 1&#xff0c;属性默认值 通过组件的 defaultProps 静态属性设置。 1.1&#xff0c;函数组…