慕慕手记项目日志 项目从开发到部署多环境配置 2025-3-8

慕慕手记项目日志 项目从开发到部署多环境配置 2025-3-8

现在是已经到了课程的第十章了,开始进行配置项目环境了。现在要完成的任务是项目可以正常运行,而且可以自由切换配置,开发/测试。

下面是当前的目录结构图:

在这里插入图片描述

现在来解释一下这些目录的作用:

  1. app:这是项目的主目录,包含了整个应用的核心代码。

    • config

      :配置文件夹,包含项目的配置信息。

      • *init*.py:初始化文件,使这个目录成为一个包。
      • config.py:配置文件,可能包含数据库连接信息、环境变量等。
      • settings.py:设置文件,可能包含一些全局设置。
    • common

      :通用模块,可能包含一些公共函数或类。

      • *init*.py:初始化文件。
      • database.py:数据库相关的操作,如ORM模型定义。
    • controller

      :控制器模块,处理HTTP请求并返回响应。

      • *init*.py:初始化文件。
      • user.py:用户相关的控制器,处理与用户相关的请求。
    • model

      :模型模块,定义数据模型。

      • *init*.py:初始化文件。
      • user.py:用户模型,定义用户相关的数据结构。

其中的config软件包下面的config.py文件里面定义了,配置的区别根据不同的情况返回对应的配置

class Config:db_url = "mysql+pymysql://root:123456@172.27.13.88:3306/mumushouji"class TestConfig(Config):if_Echo = Trueclass ProductionConfig(Config):if_Echo = False# 根据环境变量选择不同的配置
config = {"test": TestConfig,"prod": ProductionConfig
}

然后settings.py文件内需要传入env变量的类型如以下就是使用的测试环境配置

env="test"

app.py文件内定义了两个函数,分别是注册蓝图和返回app对象

from flask import Flask
def create_app():app = Flask(__name__,template_folder="../templates",static_url_path="/",static_folder="../resource")init_Blueprint(app) # 注册蓝本return appdef init_Blueprint(app):from controller.user import userapp.register_blueprint(user)

common软件包下的database.py文件内包含了数据库对象初始化的一些代码,和使用的环境

from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker, scoped_session
from app.settings import env
from app.config.config import config
# \app\database.py
def db_connect():config_class = config[env]()engine = create_engine( config_class.db_url, echo=config_class.if_Echo)session = sessionmaker(engine)# 保证线程安全db_session = scoped_session(session)Base = declarative_base() # 创建一个基类return db_session, Base, engine

接下来是控制器和模型类,这里定义了蓝图对象

from flask import Blueprintfrom model.user import Useruser = Blueprint('user', __name__)
@user.route('/')
def get_one():user = User()result = user.get_one()print(result)return "ok"

模型类里面暂时定义了User表的数据库对象还有db_sessionengine以及基类Base,但是我们这里用到的这三个对象都需要从db_connect函数内返回

from sqlalchemy import Tablefrom common.database import db_connect # 导入db_connect函数db_session, Base, engine = db_connect()
# 定义表结构
class User(Base):__table__ = Table('user', Base.metadata,autoload_with=engine)def get_one(self):return db_session.query(User).first()

日志的基本配置与应用

在该项目中,我们使用这样的日志配置。日志等级会随着env参数的改变而改变

import logging
from logging.handlers import RotatingFileHandlerfrom app.config.config import config
from app.settings import env
def set_log():config_class = config[env]()logging.basicConfig(level=config_class.LOG_LEVEL,)file_log_handler = RotatingFileHandler("log/mumunote.log", maxBytes=1024 * 1024 * 300, backupCount=10)#添加全局日志记录器formater = logging.Formatter('%(asctime)s - %(filename)s[line:%(lineno)d] - %(levelname)s: %(message)s')file_log_handler.setFormatter(formater)logging.getLogger().addHandler(file_log_handler)set_log()
  • config_class = config[env](): 根据当前环境获取对应的配置类实例。
  • logging.basicConfig(level=config_class.LOG_LEVEL,): 设置全局日志级别。这将影响所有未明确设置级别的日志记录器。
  • file_log_handler = RotatingFileHandler(...): 创建一个RotatingFileHandler实例,指定日志文件名、单个日志文件的最大大小(这里是300MB),以及保留旧日志文件的最大数量(这里是10个)。
  • formater = logging.Formatter(...): 定义日志输出格式,包括时间戳、源文件名、行号、日志级别和消息内容。
  • file_log_handler.setFormatter(formater): 将上面定义的格式应用于RotatingFileHandler实例。
  • logging.getLogger().addHandler(file_log_handler): 将file_log_handler添加到根日志记录器中,这意味着所有的日志记录都会通过这个处理器输出到日志文件中,并遵循上述设定的规则

刚刚出现的一些问题

由于视频里面的那些源文件我没有,我只能从已有的项目代码里面拷过来一部分前端的代码进行使用。但是刚刚报错误了,提示少了东西

  File "D:\code\mumunote\templates\index.html", line 1, in top-level template code{% extends 'public/header.html' %}^^^^^^^^^^^^^^^^^^^^^^^^^File "D:\code\mumunote\templates\public\header.html", line 135, in top-level template code{% block content %}File "D:\code\mumunote\templates\index.html", line 7, in block 'content'{% for label_name,label_value in label_types.items() %}

就是这个错误。根据通义千问给我的反馈,我定义了些变量,然后前端就能正常加载了。

 label_types = {'type1': 'value1','type2': 'value2',# 根据实际情况添加更多键值对...}

当前状态:

在这里插入图片描述

附完整代码:

controllter/index.py

from flask import Blueprint, render_templateindex = Blueprint('index',__name__)
@index.route('/')
def home():label_types = {'type1': 'value1','type2': 'value2',# 根据实际情况添加更多键值对...}return render_template("index.html",label_types=label_types)

当前目录结构:

在这里插入图片描述

首页数据的填充 (1)

首页数据的填充,需要在在controller模块下面创建index.py,再实现home类中的find_article函数

from flask import Blueprint, render_template, requestfrom model.article import Articleindex = Blueprint('index',__name__)
@index.route('/')
def home():page = request.args.get("page")article_type = request.args.get("article_type")if page is None:page = 1if article_type is None:article_type = 'recommend'article = Article()db_result = article.find_article(page,article_type)label_types = {'type1': 'value1','type2': 'value2',# 根据实际情况添加更多键值对...}return render_template("index.html",label_types=label_types,result=db_result)

其中有一个很大的问题是没有label_types参数,页面就会503,暂时没有排查出来是哪里的问题,我们再来看模型类中的的Article类是怎么写的

from itertools import countfrom sqlalchemy import Tablefrom common.database import db_connect
from app.settings import env
from app.config.config import config
from model.user import User
# model/article.py
db_session, Base, engine = db_connect()
# 定义表结构
class Article(Base):__table__ = Table('article', Base.metadata,autoload_with=engine)def find_article(self, page, article_type="recommend"):#     一页显示多少内容呢,我们默认为一个10,page默认应该是从1开始if int(page) < 1:page = 1count = int(page) * config[env].page_count# 这就证明是来到了推荐的分类下边if article_type == "recommend":result = db_session.query(Article, User.nickname).join(User, User.user_id == Article.user_id).filter(Article.drafted == 1).order_by(Article.browse_num.desc()).limit(count).all()else:result = db_session.query(Article, User.nickname).join(User, User.user_id == Article.user_id).filter(Article.label_name == article_type,Article.drafted == 1).order_by(Article.browse_num.desc()).limit(count).all()return result
  1. 表结构定义:

    __table__ = Table('article', Base.metadata, autoload_with=engine)
    

    这行代码使用已有的数据库表 'article' 来动态加载表结构,而不是通过类属性来手动定义每个字段。autoload_with=engine 参数告诉 SQLAlchemy 使用提供的数据库连接(engine)来反射表结构。

  2. 分页与查询逻辑:

    • 方法 find_article(self, page, article_type="recommend") 被设计用来根据页面和文章类型检索文章。
    • page 参数表示当前请求的页数,默认从 1 开始。如果传入的 page 小于 1,则将其设置为 1。
    • count = int(page) * config[env].page_count 计算要获取的文章数量。这里假设在配置文件中已经定义了每页显示的文章数量 page_count
  3. 推荐文章查询:

    • 如果 article_type"recommend",则会执行特定的查询来获取被标记为推荐的文章。这些文章还需要满足 drafted == 1 的条件,这可能意味着它们是已发布的文章而不是草稿。
    • 查询语句通过 db_session.query(Article, User.nickname).join(...) 实现,其中包含了与 User 表的联结,以同时获取文章作者的昵称。
    • .filter(Article.drafted == 1) 添加过滤条件,确保只选择已发布的文章。
    • .order_by(Article.browse_num.desc()) 根据浏览次数对结果进行降序排序。
    • .limit(count).all() 限制返回的结果数量并获取所有匹配项。
  4. 按标签查询:

    • 如果 article_type 不是 "recommend",则根据提供的 article_type(即标签名)来筛选文章。查询逻辑与上述类似,但添加了额外的过滤条件 Article.label_name == article_type 来筛选特定标签的文章。

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

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

相关文章

《Python实战进阶》No15: 数据可视化:Matplotlib 与 Seaborn 的高级用法

No15: 数据可视化&#xff1a;Matplotlib 与 Seaborn 的高级用法 Matplotlib 是 Python 中最受欢迎的数据可视化软件包之一&#xff0c;支持跨平台运行&#xff0c;它是 Python 常用的 2D 绘图库&#xff0c;同时它也提供了一部分 3D 绘图接口。Matplotlib 通常与 NumPy、Pand…

C++学习之格斗小游戏综合案例

C格斗游戏效果视频 1.案例简介 #include "broadSword.h" //构造函数 BroadSword::BroadSword() { FileManager fm; map<string, map<string, string>> mWeapon; fm.loadCSVData("Weapons.csv", mWeapon); //武器id string id …

LeetCodeHot100

1.两数之和 解题思路&#xff1a; 1.暴力解法 两个for循环解决问题 时间复杂度为 O(n2) class Solution {public int[] twoSum(int[] nums, int target) {int n nums.length;for (int i 0; i < n; i) {for (int j i 1; j < n; j) {if (nums[i] nums[j] target) {…

大语言模型进化论:从达尔文到AI的启示与展望

文章大纲 引言大语言模型中的“进化论”思想体现遗传变异过度繁殖和生存斗争大模型“过度繁殖”与“生存竞争”机制解析**一、过度繁殖:技术迭代的指数级爆发****二、生存竞争:计算资源的达尔文战场****三、生存竞争胜出关键要素****四、行业竞争格局演化趋势**核心结论自然选…

Spring Boot自动装配原理

实例&#xff1a; 引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId> </dependency> 配置服务器 redis自动装配 原理&#xff1a; 一切都源于一个关键的注解…

在Windows系统上安装和配置Redis服务

&#x1f31f; 在Windows系统上安装和配置Redis服务 Redis是一个高性能的键值存储数据库&#xff0c;广泛用于缓存、消息队列和实时分析等场景。虽然Redis最初是为Linux设计的&#xff0c;但也有Windows版本可供使用。今天&#xff0c;我将详细介绍如何在Windows系统上安装Red…

《安富莱嵌入式周报》第351期:DIY半导体制造,工业设备抗干扰提升方法,NASA软件开发规范,小型LCD在线UI编辑器,开源USB PD电源,开源锂电池管理

周报汇总地址&#xff1a;嵌入式周报 - uCOS & uCGUI & emWin & embOS & TouchGFX & ThreadX - 硬汉嵌入式论坛 - Powered by Discuz! 视频版&#xff1a; https://www.bilibili.com/video/BV16C95YEEZs 《安富莱嵌入式周报》第351期&#xff1a;DIY半导体…

Vue3实战学习(IDEA中打开、启动与搭建Vue3工程极简脚手架教程(2025超详细教程)、Windows系统命令行启动Vue3工程)(2)

目录 一、命令行中重新启动已搭建好的Vue3工程。(快速上手) &#xff08;0&#xff09;Windows环境下使用命令行从零到一手动搭建Vue3工程教程。 &#xff08;1&#xff09;首先找到已建Vue3工程的目录。 &#xff08;2&#xff09;无需再下载依赖包&#xff0c;直接执行npm ru…

Visual Studio 2022新建c语言项目的详细步骤

步骤1&#xff1a;点击创建新项目 步骤2&#xff1a;到了项目模板 --> 选择“控制台应用” (在window终端运行代码。默认打印"Hello World") --> 点击 “下一步” 步骤3&#xff1a;到了配置新项目模块 --> 输入“项目名称” --> 更改“位置”路径&…

23年以后版本pycharm找不到conda可执行文件解决办法

这个问题很痛苦&#xff0c;折磨了我半天。 就是链接远程服务器的时候 就一直以为这三个都要配置 就这个conda环境这里怎么都找不到服务器的虚拟环境的python可执行文件&#xff0c;非常痛苦。 后面查找了资料&#xff0c;找了好久&#xff0c;才发现&#xff0c;原来只需要配…

基于SpringBoot的商城管理系统(源码+部署教程)

运行环境 数据库&#xff1a;MySql 编译器&#xff1a;Intellij IDEA 前端运行环境&#xff1a;node.js v12.13.0 JAVA版本&#xff1a;JDK 1.8 主要功能 基于Springboot的商城管理系统包含管理端和用户端两个部分&#xff0c;主要功能有&#xff1a; 管理端 首页商品列…

计算机网络软考

1.物理层 1.两个主机之间发送数据的过程 自上而下的封装数据&#xff0c;自下而上的解封装数据&#xff0c;实现数据的传输 2.数据、信号、码元 码元就是数字通信里用来表示信息的基本信号单元。比如在二进制中&#xff0c;用高电平代表 “1”、低电平代表 “0”&#xff0c…

静态时序分析:SDC约束命令set_ideal_latency详解

相关阅读 静态时序分析https://blog.csdn.net/weixin_45791458/category_12567571.html?spm1001.2014.3001.5482 当使用set_ideal_network命令将当前设计中的一组端口或引脚标记为理想网络源后&#xff0c;理想属性会沿着组合逻辑进行传播&#xff0c;理想网络中的线网和单元…

C语言(队列)

1、队列的原理和作用 1、1 队列的原理 队列的原理其实就像一个管道&#xff0c;如果我们不断的往管道里塞乒乓球&#xff0c;每个乒乓球在管道里就会排列一条队列&#xff0c;先进去的乒乓球会先出来&#xff0c;这个就是队列先进先出的规则 球从左边进去的动作叫入列&#xf…

十七、从0开始卷出一个新项目之瑞萨RZN2L定时器(GPT)+DMA生成PWM的运动控制

一、概述 嵌入式科普(34)通过对比看透DMA的本质 分享瑞萨RZN2L使用DMA生成PWM的运动控制的例程源码 rzn2l必要的外设资源&#xff1a; rzn2l拥有32-bit timer General PWM Timer (GPT) with 18 channels CPU、GPT最高频率400Mhz DMAC0 and DMAC1 8 channels 8 channels 还…

无人机推流/RTMP视频推拉流:EasyDSS无法卸载软件的原因及解决方法

视频推拉流/直播点播EasyDSS平台支持音视频采集、视频推拉流、播放H.265编码视频、存储、分发等视频能力服务&#xff0c;在应用场景中可实现视频直播、点播、转码、管理、录像、检索、时移回看等。此外&#xff0c;平台还支持用户自行上传视频文件&#xff0c;也可将上传的点播…

树莓派5首次开机保姆级教程(无显示器通过VNC连接树莓派桌面)

第一次开机详细步骤 步骤一&#xff1a;树莓派系统烧录1 搜索打开烧录软件“Raspberry Pi Imager”2 选择合适的设备、系统、SD卡3 烧录配置选项 步骤二&#xff1a;SSH远程树莓派1 树莓派插电2 网络连接&#xff08;有线或无线&#xff09;3 确定树莓派IP地址 步骤三&#xff…

关于 QPalette设置按钮背景未显示出来 的解决方法

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/146047054 长沙红胖子Qt&#xff08;长沙创微智科&#xff09;博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV…

文献分享: ConstBERT固定数目向量编码文档

&#x1f602;图放这了&#xff0c;大道至简的 idea \text{idea} idea不愧是 ECIR \text{ECIR} ECIR &#x1f449;原论文 1. ConstBERT \textbf{1. ConstBERT} 1. ConstBERT的原理 1️⃣模型的改进点&#xff1a;相较于 ColBERT \text{ColBERT} ColBERT为每个 Token \text{Tok…

DeepSeek 医疗大模型微调实战讨论版(第一部分)

DeepSeek医疗大模型微调实战指南第一部分 DeepSeek 作为一款具有独特优势的大模型,在医疗领域展现出了巨大的应用潜力。它采用了先进的混合专家架构(MoE),能够根据输入数据的特性选择性激活部分专家,避免了不必要的计算,极大地提高了计算效率和模型精度 。这种架构使得 …