当前位置: 首页 > news >正文

毕昇入门学习

schemas.py

概述

这段代码主要定义了一系列基于 Pydantic 的数据模型(BaseModel),用于数据验证和序列化,通常用于构建 API(如使用 FastAPI)。这些模型涵盖了用户认证、聊天消息、知识库管理、模型配置等多个方面。此外,还定义了一些通用的响应模型和辅助函数,以标准化 API 的响应格式。

详细解析

导入部分

from datetime import datetime
from enum import Enum
from typing import Any, Dict, Generic, List, Optional, TypeVar, Union
from uuid import UUIDfrom langchain.docstore.document import Document
from orjson import orjson
from pydantic import BaseModel, Field, validator, root_validatorfrom bisheng.database.models.assistant import AssistantBase
from bisheng.database.models.finetune import TrainMethod
from bisheng.database.models.flow import FlowCreate, FlowRead
from bisheng.database.models.gpts_tools import GptsToolsRead, AuthMethod, AuthType
from bisheng.database.models.knowledge import KnowledgeRead
from bisheng.database.models.llm_server import LLMServerBase, LLMModelBase, LLMServerType, LLMModelType
from bisheng.database.models.message import ChatMessageRead
from bisheng.database.models.tag import Tag
  • 标准库导入
    • datetime:处理日期和时间。
    • Enum:定义枚举类型。
    • typing:用于类型注解,增强代码可读性和可维护性。
    • UUID:处理唯一标识符。
  • 第三方库导入
    • langchain.docstore.document.Document:可能用于文档存储和检索。
    • orjson:高性能的 JSON 库,用于快速序列化和反序列化 JSON 数据。
    • pydantic:用于数据验证和设置管理,广泛用于 FastAPI。
  • 项目内部模块导入
    • bisheng.database.models.* 导入多个 ORM 模型,代表数据库中的不同实体(如助手、微调方法、流程、工具、知识库、LLM 服务器和模型、聊天消息、标签等)。

数据模型

以下是代码中定义的主要数据模型和相关函数的详细解释。

基础输入模型
  1. 验证码输入

    class CaptchaInput(BaseModel):captcha_key: strcaptcha: str
    
    • 用途:用于用户提交验证码时的输入验证。
    • 字段:
      • captcha_key:验证码的唯一标识符。
      • captcha:用户输入的验证码内容。
  2. 分块输入

    class ChunkInput(BaseModel):knowledge_id: intdocuments: List[Document]
    
    • 用途:用于将文档分块处理的输入。
    • 字段:
      • knowledge_id:知识库的唯一标识符。
      • documents:文档列表,每个文档为 langchain.docstore.document.Document 类型。
枚举类型
  1. 构建状态

    class BuildStatus(Enum):"""Status of the build."""SUCCESS = 'success'FAILURE = 'failure'STARTED = 'started'IN_PROGRESS = 'in_progress'
    
    • 用途:表示构建过程的状态。
    • 枚举值:
      • SUCCESS:构建成功。
      • FAILURE:构建失败。
      • STARTED:构建已启动。
      • IN_PROGRESS:构建进行中。
图数据模型
  1. 图数据

    class GraphData(BaseModel):"""Data inside the exported flow."""nodes: List[Dict[str, Any]]edges: List[Dict[str, Any]]
    
    • 用途:表示导出流程中的图数据。
    • 字段:
      • nodes:节点列表,每个节点为字典类型,包含节点的详细信息。
      • edges:边列表,每条边为字典类型,描述节点之间的连接关系。
  2. 导出流程

    class ExportedFlow(BaseModel):"""Exported flow from bisheng."""description: strname: strid: strdata: GraphData
    
    • 用途:表示从 bisheng 导出的流程数据。
    • 字段:
      • description:流程描述。
      • name:流程名称。
      • id:流程的唯一标识符。
      • data:流程的图数据,类型为 GraphData
请求模型
  1. 输入请求

    class InputRequest(BaseModel):input: str = Field(description='question or command asked LLM to do')
    
    • 用途:表示用户向 LLM(大型语言模型)提出的问题或命令。
    • 字段:
      • input:用户的输入内容(问题或命令)。
  2. 调整请求

    class TweaksRequest(BaseModel):tweaks: Optional[Dict[str, Dict[str, str]]] = Field(default_factory=dict)
    
    • 用途:表示对某些设置或参数的调整请求。
    • 字段:
      • tweaks:一个可选的字典,嵌套字典用于描述具体的调整内容。
  3. 更新模板请求

    class UpdateTemplateRequest(BaseModel):template: dict
    
    • 用途:用于更新模板的请求。
    • 字段:
      • template:模板内容,类型为字典。
通用响应模型
  1. 统一响应模型

    DataT = TypeVar('DataT')class UnifiedResponseModel(Generic[DataT], BaseModel):"""统一响应模型"""status_code: intstatus_message: strdata: DataT = None
    
    • 用途:提供一个通用的 API 响应结构,适用于各种类型的数据。
    • 泛型:
      • DataT:数据的具体类型,可以是任意类型。
    • 字段:
      • status_code:响应状态码(如 200、500 等)。
      • status_message:响应状态信息(如 “SUCCESS”、“BAD REQUEST” 等)。
      • data:响应的数据,类型为泛型 DataT
  2. 成功响应函数

    def resp_200(data: Union[list, dict, str, Any] = None,message: str = 'SUCCESS') -> UnifiedResponseModel:"""成功的代码"""return UnifiedResponseModel(status_code=200, status_message=message, data=data)
    
    • 用途:生成一个状态码为 200 的成功响应。
    • 参数:
      • data:响应的数据,可以是列表、字典、字符串或任意类型。
      • message:响应的状态信息,默认值为 “SUCCESS”。
  3. 错误响应函数

    def resp_500(code: int = 500,data: Union[list, dict, str, Any] = None,message: str = 'BAD REQUEST') -> UnifiedResponseModel:"""错误的逻辑回复"""return UnifiedResponseModel(status_code=code, status_message=message, data=data)
    
    • 用途:生成一个错误响应,默认状态码为 500。
    • 参数:
      • code:错误状态码,默认值为 500。
      • data:响应的数据,可以是列表、字典、字符串或任意类型。
      • message:响应的状态信息,默认值为 “BAD REQUEST”。
处理响应模型
  1. 处理响应

    class ProcessResponse(BaseModel):"""Process response schema."""result: Any# task: Optional[TaskResponse] = Nonesession_id: Optional[str] = Nonebackend: Optional[str] = None
    
    • 用途:表示处理请求后的响应结构。
    • 字段:
      • result:处理结果,类型为任意类型。
      • session_id:会话标识符,可选。
      • backend:后端标识,可选。
聊天相关模型
  1. 聊天输入

    class ChatInput(BaseModel):message_id: intcomment: str = Noneliked: int = 0
    
    • 用途:表示用户对某条聊天消息的输入(如评论、点赞)。
    • 字段:
      • message_id:消息的唯一标识符。
      • comment:用户的评论,可选。
      • liked:点赞数,默认值为 0。
  2. 添加聊天消息

    class AddChatMessages(BaseModel):"""Add a pair of chat messages."""flow_id: UUID  # 技能或助手IDchat_id: str  # 会话IDhuman_message: str = None  # 用户问题answer_message: str = None  # 执行结果
    
    • 用途:用于向聊天会话中添加一对消息(用户消息和助手回复)。

    • 字段

      • flow_id:技能或助手的唯一标识符,类型为 UUID
      • chat_id:会话的唯一标识符。
      • human_message:用户的问题,可选。
      • answer_message:助手的回复,可选。
  3. 聊天列表

    class ChatList(BaseModel):"""Chat message list."""flow_name: str = Noneflow_description: str = Noneflow_id: UUID = Nonechat_id: str = Nonecreate_time: datetime = Noneupdate_time: datetime = Noneflow_type: str = None  # flow: 技能 assistant:gpts助手latest_message: ChatMessageRead = Nonelogo: Optional[str] = None
    
    • 用途:表示一个聊天会话的简要信息,用于展示聊天列表。
    • 字段:
      • flow_name:技能或助手的名称。
      • flow_description:技能或助手的描述。
      • flow_id:技能或助手的唯一标识符,类型为 UUID
      • chat_id:会话的唯一标识符。
      • create_time:会话的创建时间。
      • update_time:会话的更新时间。
      • flow_type:技能类型,flow 表示技能,assistant 表示 GPTs 助手。
      • latest_message:最新的聊天消息,类型为 ChatMessageRead
      • logo:技能或助手的 logo 地址,可选。
  4. 在线流程列表

    class FlowGptsOnlineList(BaseModel):id: str = Field('唯一ID')name: str = Nonedesc: str = Nonelogo: str = Nonecreate_time: datetime = Noneupdate_time: datetime = Noneflow_type: str = None  # flow: 技能 assistant:gpts助手count: int = 0
    
    • 用途:表示在线流程(技能或助手)的列表信息。
    • 字段:
      • id:流程的唯一标识符。
      • name:流程名称。
      • desc:流程描述。
      • logo:流程的 logo 地址。
      • create_time:流程的创建时间。
      • update_time:流程的更新时间。
      • flow_type:流程类型,flow 表示技能,assistant 表示 GPTs 助手。
      • count:关联的某些数量(具体含义需结合业务逻辑理解)。
  5. 聊天消息

    class ChatMessage(BaseModel):"""Chat message schema."""is_bot: bool = Falsemessage: Union[str, None, dict] = ''type: str = 'human'category: str = 'processing'  # system processing answer toolintermediate_steps: str = Nonefiles: list = []user_id: int = Nonemessage_id: int = Nonesource: int = 0sender: str = Nonereceiver: dict = Noneliked: int = 0extra: str = '{}'flow_id: str = Nonechat_id: str = None
    
    • 用途:表示一条聊天消息的详细信息。
    • 字段:
      • is_bot:是否为机器人消息,默认值为 False
      • message:消息内容,可以是字符串、字典或 None
      • type:消息类型,默认值为 'human'(表示用户消息)。
      • category:消息分类,如 'system''processing''answer''tool'
      • intermediate_steps:中间步骤信息,可选。
      • files:关联的文件列表,默认空列表。
      • user_id:用户的唯一标识符,可选。
      • message_id:消息的唯一标识符,可选。
      • source:消息来源,默认值为 0
      • sender:发送者信息,可选。
      • receiver:接收者信息,字典类型,可选。
      • liked:点赞数,默认值为 0
      • extra:额外信息,默认值为 '{}'
      • flow_id:关联的流程(技能或助手)的唯一标识符,可选。
      • chat_id:关联的聊天会话的唯一标识符,可选。
  6. 聊天响应

    class ChatResponse(ChatMessage):"""Chat response schema."""intermediate_steps: str = ''type: stris_bot: bool = Truefiles: list = []category: str = 'processing'@validator('type')def validate_message_type(cls, v):"""end_cover: 结束并覆盖上一条message"""if v not in ['start', 'stream', 'end', 'error', 'info', 'file', 'begin', 'close', 'end_cover']:raise ValueError('type must be start, stream, end, error, info, or file')return v
    
    • 用途:表示机器人(助手)的聊天响应消息。
    • 继承自ChatMessage,并对部分字段进行了重写或扩展。
    • 字段
      • intermediate_steps:中间步骤信息,默认值为空字符串。
      • type:消息类型,必须在指定的枚举值中。
      • is_bot:是否为机器人消息,固定为 True
      • files:关联的文件列表,默认空列表。
      • category:消息分类,固定为 'processing'
    • 验证器
      • validate_message_type:确保 type 字段的值在指定的枚举值范围内,否则抛出错误。
  7. 文件响应

    class FileResponse(ChatMessage):"""File response schema."""data: Anydata_type: strtype: str = 'file'is_bot: bool = True@validator('data_type')def validate_data_type(cls, v):if v not in ['image', 'csv']:raise ValueError('data_type must be image or csv')return v
    
    • 用途:表示机器人发送的文件类型消息。
    • 继承自ChatMessage,并对部分字段进行了重写或扩展。
    • 字段
      • data:文件数据,类型为任意类型。
      • data_type:文件类型,必须是 'image''csv'
      • type:消息类型,固定为 'file'
      • is_bot:是否为机器人消息,固定为 True
    • 验证器
      • validate_data_type:确保 data_type 字段的值为 'image''csv',否则抛出错误。
流程相关模型
  1. 流程列表创建

    class FlowListCreate(BaseModel):flows: List[FlowCreate]
    
    • 用途:用于批量创建流程(技能或助手)。
    • 字段:
      • flows:流程创建请求的列表,类型为 List[FlowCreate]
  2. 流程列表读取

    class FlowListRead(BaseModel):flows: List[FlowRead]
    
    • 用途:用于批量读取流程的信息。
    • 字段:
      • flows:流程读取响应的列表,类型为 List[FlowRead]
  3. 初始化响应

    class InitResponse(BaseModel):flowId: str
    
    • 用途:表示初始化操作后的响应,通常返回一个流程的唯一标识符。
    • 字段:
      • flowId:流程的唯一标识符。
  4. 构建响应

    class BuiltResponse(BaseModel):built: bool
    
    • 用途:表示构建操作的结果。
    • 字段:
      • built:布尔值,表示是否构建成功。
  5. 上传文件响应

    class UploadFileResponse(BaseModel):"""Upload file response schema."""flowId: Optional[str]file_path: strrelative_path: Optional[str]  # minio的相对路径,即object_name
    
    • 用途:表示文件上传后的响应。
    • 字段:
      • flowId:关联的流程(技能或助手)的唯一标识符,可选。
      • file_path:文件的完整路径。
      • relative_path:文件在存储系统(如 MinIO)中的相对路径,即对象名称,可选。
  6. 流数据

    class StreamData(BaseModel):event: strdata: dict | strdef __str__(self) -> str:if isinstance(self.data, dict):return f'event: {self.event}\ndata: {orjson.dumps(self.data).decode()}\n\n'return f'event: {self.event}\ndata: {self.data}\n\n'
    
    • 用途:表示流式数据传输的结构,常用于服务器发送事件(SSE)。
    • 字段
      • event:事件类型。
      • data:事件数据,可以是字典或字符串。
    • 方法
      • __str__:将 StreamData 对象转换为字符串格式,适用于流式传输。
微调相关模型
  1. 微调创建请求

    class FinetuneCreateReq(BaseModel):server: int = Field(description='关联的RT服务ID')base_model: int = Field(description='基础模型ID')model_name: str = Field(max_length=50, description='模型名称')method: TrainMethod = Field(description='训练方法')extra_params: Dict = Field(default_factory=dict, description='训练任务所需额外参数')train_data: Optional[List[Dict]] = Field(default=None, description='个人训练数据')preset_data: Optional[List[Dict]] = Field(default=None, description='预设训练数据')
    
    • 用途:表示创建微调任务的请求。
    • 字段:
      • server:关联的 RT 服务的唯一标识符。
      • base_model:基础模型的唯一标识符。
      • model_name:模型名称,最大长度为 50。
      • method:训练方法,类型为 TrainMethod(枚举类型)。
      • extra_params:训练任务所需的额外参数,默认为空字典。
      • train_data:个人训练数据,可选。
      • preset_data:预设训练数据,可选。
组件相关模型
  1. 创建组件请求

    class CreateComponentReq(BaseModel):name: str = Field(max_length=50, description='组件名称')data: Any = Field(default='', description='组件数据')description: Optional[str] = Field(default='', description='组件描述')
    
    • 用途:表示创建组件的请求。
    • 字段:
      • name:组件名称,最大长度为 50。
      • data:组件的数据,类型为任意类型,默认值为空字符串。
      • description:组件描述,可选,默认值为空字符串。
  2. 自定义组件代码

    class CustomComponentCode(BaseModel):code: strfield: Optional[str] = Nonefrontend_node: Optional[dict] = None
    
    • 用途:表示自定义组件的代码。
    • 字段:
      • code:组件的代码。
      • field:可选字段,可能用于描述组件的特定字段。
      • frontend_node:前端节点的信息,类型为字典,可选。
助手相关模型
  1. 创建助手请求

    class AssistantCreateReq(BaseModel):name: str = Field(max_length=50, description='助手名称')prompt: str = Field(min_length=20, max_length=1000, description='助手提示词')logo: str = Field(description='logo文件的相对地址')
    
    • 用途:表示创建助手的请求。
    • 字段:
      • name:助手名称,最大长度为 50。
      • prompt:助手的提示词,最小长度为 20,最大长度为 1000。
      • logo:logo 文件的相对地址。
  2. 更新助手请求

    class AssistantUpdateReq(BaseModel):id: UUID = Field(description='助手ID')name: Optional[str] = Field('', description='助手名称, 为空则不更新')desc: Optional[str] = Field('', description='助手描述, 为空则不更新')logo: Optional[str] = Field('', description='logo文件的相对地址,为空则不更新')prompt: Optional[str] = Field('', description='用户可见prompt, 为空则不更新')guide_word: Optional[str] = Field('', description='开场白, 为空则不更新')guide_question: Optional[List] = Field([], description='引导问题列表, 为空则不更新')model_name: Optional[str] = Field('', description='选择的模型名, 为空则不更新')temperature: Optional[float] = Field(None, description='模型温度, 不传则不更新')tool_list: List[int] | None = Field(default=None,description='助手的工具ID列表,空列表则清空绑定的工具,为None则不更新')flow_list: List[str] | None = Field(default=None, description='助手的技能ID列表,为None则不更新')knowledge_list: List[int] | None = Field(default=None, description='知识库ID列表,为None则不更新')
    
    • 用途:表示更新助手的请求。
    • 字段:
      • id:助手的唯一标识符,类型为 UUID
      • name:助手名称,可选,默认为空字符串,空则不更新。
      • desc:助手描述,可选,默认为空字符串,空则不更新。
      • logo:logo 文件的相对地址,可选,默认为空字符串,空则不更新。
      • prompt:用户可见的提示词,可选,默认为空字符串,空则不更新。
      • guide_word:开场白,可选,默认为空字符串,空则不更新。
      • guide_question:引导问题列表,可选,默认为空列表,空则不更新。
      • model_name:选择的模型名称,可选,默认为空字符串,空则不更新。
      • temperature:模型温度,可选,默认为 None,不传则不更新。
      • tool_list:助手的工具 ID 列表,可选,None 表示不更新,空列表表示清空绑定的工具。
      • flow_list:助手的技能 ID 列表,可选,None 表示不更新。
      • knowledge_list:知识库 ID 列表,可选,None 表示不更新。
  3. 助手简单信息

    class AssistantSimpleInfo(BaseModel):id: UUIDname: strdesc: strlogo: struser_id: intuser_name: strstatus: intwrite: Optional[bool] = Field(default=False)group_ids: Optional[List[int]]tags: Optional[List[Tag]]create_time: datetimeupdate_time: datetime
    
    • 用途:表示助手的基本信息,用于展示列表或概要信息。
    • 字段:
      • id:助手的唯一标识符,类型为 UUID
      • name:助手名称。
      • desc:助手描述。
      • logo:助手的 logo 地址。
      • user_id:创建者的用户 ID。
      • user_name:创建者的用户名。
      • status:助手的状态,类型为整数(具体含义需结合业务逻辑理解)。
      • write:是否可写,默认值为 False
      • group_ids:关联的用户组 ID 列表,可选。
      • tags:关联的标签列表,类型为 List[Tag],可选。
      • create_time:创建时间。
      • update_time:更新时间。
  4. 助手信息

    class AssistantInfo(AssistantBase):tool_list: List[GptsToolsRead] = Field(default=[], description='助手的工具ID列表')flow_list: List[FlowRead] = Field(default=[], description='助手的技能ID列表')knowledge_list: List[KnowledgeRead] = Field(default=[], description='知识库ID列表')
    
    • 用途:表示助手的详细信息,继承自 AssistantBase(数据库模型)。
    • 字段:
      • tool_list:助手的工具列表,类型为 List[GptsToolsRead]
      • flow_list:助手的技能列表,类型为 List[FlowRead]
      • knowledge_list:知识库列表,类型为 List[KnowledgeRead]
流程版本与对比模型
  1. 流程版本创建

    class FlowVersionCreate(BaseModel):name: Optional[str] = Field(default=..., description="版本的名字")description: Optional[str] = Field(default=None, description="版本的描述")data: Optional[Dict] = Field(default=None, description='技能版本的节点数据数据')original_version_id: Optional[int] = Field(default=None, description="版本的来源版本ID")
    
    • 用途:表示创建流程版本的请求。
    • 字段:
      • name:版本名称。
      • description:版本描述。
      • data:技能版本的节点数据,类型为字典,可选。
      • original_version_id:来源版本的 ID,可选。
  2. 流程对比请求

    class FlowCompareReq(BaseModel):inputs: Any = Field(default=None, description='技能运行所需要的输入')question_list: List[str] = Field(default=[], description='测试case列表')version_list: List[int] = Field(default=[], description='对比版本ID列表')node_id: str = Field(default=None, description='需要对比的节点唯一ID')thread_num: Optional[int] = Field(default=1, description='对比线程数')
    
    • 用途:表示对比不同版本流程的请求。
    • 字段:
      • inputs:技能运行所需的输入,类型为任意类型。
      • question_list:测试用例列表,类型为 List[str]
      • version_list:对比的版本 ID 列表,类型为 List[int]
      • node_id:需要对比的节点的唯一标识符,类型为字符串,可选。
      • thread_num:对比的线程数,默认值为 1
工具类型与测试模型
  1. 删除工具类型请求

    class DeleteToolTypeReq(BaseModel):tool_type_id: int = Field(description="要删除的工具类别ID")
    
    • 用途:表示删除工具类别的请求。
    • 字段:
      • tool_type_id:要删除的工具类别的唯一标识符。
  2. 测试工具请求

    class TestToolReq(BaseModel):server_host: str = Field(default='', description="服务的根地址")extra: str = Field(default='', description="Api 对象解析后的extra字段")auth_method: int = Field(default=AuthMethod.NO.value, description="认证类型")auth_type: Optional[str] = Field(default=AuthType.BASIC.value, description="Auth Type")api_key: Optional[str] = Field(default='', description="api key")request_params: Dict = Field(default=None, description="用户填写的请求参数")
    
    • 用途:表示测试工具接口的请求。
    • 字段:
      • server_host:服务的根地址,默认值为空字符串。
      • extra:API 对象解析后的额外字段,默认值为空字符串。
      • auth_method:认证方法,默认值为 AuthMethod.NO.value
      • auth_type:认证类型,默认值为 AuthType.BASIC.value,可选。
      • api_key:API 密钥,默认值为空字符串,可选。
      • request_params:用户填写的请求参数,类型为字典,可选。
用户与角色模型
  1. 用户组与角色

    class GroupAndRoles(BaseModel):group_id: introle_ids: List[int]
    
    • 用途:表示用户所属的组和角色。
    • 字段:
      • group_id:用户组的唯一标识符。
      • role_ids:角色的唯一标识符列表。
  2. 创建用户请求

    class CreateUserReq(BaseModel):user_name: str = Field(max_length=30, description='用户名')password: str = Field(description='密码')group_roles: List[GroupAndRoles] = Field(description='要加入的用户组和角色列表')
    
    • 用途:表示创建新用户的请求。
    • 字段:
      • user_name:用户名,最大长度为 30。
      • password:用户密码。
      • group_roles:用户所属的用户组和角色列表,类型为 List[GroupAndRoles]
OpenAI 聊天模型相关
  1. OpenAI 聊天完成请求

    class OpenAIChatCompletionReq(BaseModel):messages: List[dict] = Field(..., description="聊天消息列表,只支持user、assistant。system用数据库内的数据")model: str = Field(..., description="助手的唯一ID")n: int = Field(default=1, description="返回的答案个数, 助手侧默认为1,暂不支持多个回答")stream: bool = Field(default=False, description="是否开启流式回复")temperature: float = Field(default=0.0, description="模型温度, 传入0或者不传表示不覆盖")tools: List[dict] = Field(default=[], description="工具列表, 助手暂不支持,使用助手的配置")
    
    • 用途:表示向 OpenAI 的聊天完成 API 发送请求的结构。
    • 字段:
      • messages:聊天消息列表,类型为 List[dict],只支持 userassistantsystem 消息使用数据库内的数据。
      • model:助手的唯一标识符。
      • n:返回的答案个数,默认值为 1,当前不支持多个回答。
      • stream:是否开启流式回复,默认值为 False
      • temperature:模型温度,默认值为 0.0,传入 0 或不传表示不覆盖。
      • tools:工具列表,默认值为空列表,目前助手暂不支持,使用助手的配置。
  2. OpenAI 选择

    class OpenAIChoice(BaseModel):index: int = Field(..., description="选项的索引")message: dict = Field(default=None, description="对应的消息内容,和输入的格式一致")finish_reason: str = Field(default='stop', description="结束原因, 助手只有stop")delta: dict = Field(default=None, description="对应的openai流式返回消息内容")
    
    • 用途:表示 OpenAI 聊天完成 API 返回的单个选择项。
    • 字段:
      • index:选项的索引。
      • message:对应的消息内容,格式与输入一致。
      • finish_reason:结束原因,助手只有 'stop'
      • delta:对应的 OpenAI 流式返回消息内容。
  3. OpenAI 聊天完成响应

    class OpenAIChatCompletionResp(BaseModel):id: str = Field(..., description="请求的唯一ID")object: str = Field(default='chat.completion', description="返回的类型")created: int = Field(default=..., description="返回的创建时间戳")model: str = Field(..., description="返回的模型,对应助手的id")choices: List[OpenAIChoice] = Field(..., description="返回的答案列表")usage: dict = Field(default=None, description="返回的token用量, 助手此值为空")system_fingerprint: Optional[str] = Field(default=None, description="系统指纹")
    
    • 用途:表示 OpenAI 聊天完成 API 的响应结构。
    • 字段:
      • id:请求的唯一标识符。
      • object:返回的类型,默认值为 'chat.completion'
      • created:返回的创建时间戳。
      • model:返回的模型名称,对应助手的 ID。
      • choices:返回的答案列表,类型为 List[OpenAIChoice]
      • usage:返回的 token 用量,助手此值为空。
      • system_fingerprint:系统指纹,可选。
LLM 模型与服务器配置
  1. LLM 模型创建请求

    class LLMModelCreateReq(BaseModel):id: Optional[int] = Field(default=None, description="模型唯一ID, 更新时需要传")name: str = Field(..., description="模型展示名称")description: Optional[str] = Field(default='', description="模型描述")model_name: str = Field(..., description="模型名称")model_type: str = Field(..., description="模型类型")online: bool = Field(default=True, description='是否在线')config: Optional[dict] = Field(default=None, description="模型配置")
    
    • 用途:表示创建或更新 LLM 模型的请求。
    • 字段:
      • id:模型的唯一标识符,更新时需要传。
      • name:模型的展示名称。
      • description:模型描述,可选,默认值为空字符串。
      • model_name:模型名称。
      • model_type:模型类型。
      • online:是否在线,默认值为 True
      • config:模型的配置,类型为字典,可选。
  2. LLM 服务器创建请求

    class LLMServerCreateReq(BaseModel):id: Optional[int] = Field(default=None, description="服务提供方ID, 更新时需要传")name: str = Field(..., description="服务提供方名称")description: Optional[str] = Field(default='', description="服务提供方描述")type: str = Field(..., description="服务提供方类型")limit_flag: Optional[bool] = Field(default=False, description="是否开启每日调用次数限制")limit: Optional[int] = Field(default=0, description="每日调用次数限制")config: Optional[dict] = Field(default=None, description="服务提供方配置")models: Optional[List[LLMModelCreateReq]] = Field(default=[], description="服务提供方下的模型列表")
    
    • 用途:表示创建或更新 LLM 服务器的请求。
    • 字段:
      • id:服务提供方的唯一标识符,更新时需要传。
      • name:服务提供方名称。
      • description:服务提供方描述,可选,默认值为空字符串。
      • type:服务提供方类型。
      • limit_flag:是否开启每日调用次数限制,默认值为 False
      • limit:每日调用次数限制,默认值为 0
      • config:服务提供方的配置,类型为字典,可选。
      • models:服务提供方下的模型列表,类型为 List[LLMModelCreateReq],默认值为空列表。
  3. LLM 模型信息

    class LLMModelInfo(LLMModelBase):id: Optional[int]
    
    • 用途:表示 LLM 模型的详细信息,继承自 LLMModelBase(数据库模型)。
    • 字段:
      • id:模型的唯一标识符,可选。
  4. LLM 服务器信息

    class LLMServerInfo(LLMServerBase):id: Optional[int]models: List[LLMModelInfo] = Field(default=[], description="模型列表")
    
    • 用途:表示 LLM 服务器的详细信息,继承自 LLMServerBase(数据库模型)。
    • 字段:
      • id:服务器的唯一标识符,可选。
      • models:服务器下的模型列表,类型为 List[LLMModelInfo],默认值为空列表。
知识库配置模型
  1. 知识库 LLM 配置

    class KnowledgeLLMConfig(BaseModel):embedding_model_id: Optional[int] = Field(description="知识库默认embedding模型的ID")source_model_id: Optional[int] = Field(description="知识库溯源模型的ID")extract_title_model_id: Optional[int] = Field(description="文档知识库提取标题模型的ID")qa_similar_model_id: Optional[int] = Field(description="QA知识库相似问模型的ID")
    
    • 用途:表示知识库的 LLM 配置。
    • 字段:
      • embedding_model_id:默认的 embedding 模型 ID,可选。
      • source_model_id:溯源模型 ID,可选。
      • extract_title_model_id:文档知识库提取标题模型的 ID,可选。
      • qa_similar_model_id:QA 知识库相似问模型的 ID,可选。
  2. 助手 LLM 项目

    class AssistantLLMItem(BaseModel):model_id: Optional[int] = Field(description="模型的ID")agent_executor_type: Optional[str] = Field(default="ReAct", description="执行模式。function call 或者 ReAct")knowledge_max_content: Optional[int] = Field(default=15000, description="知识库检索最大字符串数")knowledge_sort_index: Optional[bool] = Field(default=False, description="知识库检索后是否重排")streaming: Optional[bool] = Field(default=True, description="是否开启流式")default: Optional[bool] = Field(default=False, description="是否为默认模型")
    
    • 用途:表示助手的单个 LLM 配置项。
    • 字段:
      • model_id:模型的唯一标识符,可选。
      • agent_executor_type:执行模式,默认值为 "ReAct",可选,其他值如 "function call"
      • knowledge_max_content:知识库检索的最大字符串数,默认值为 15000
      • knowledge_sort_index:知识库检索后是否重排,默认值为 False
      • streaming:是否开启流式回复,默认值为 True
      • default:是否为默认模型,默认值为 False
  3. 助手 LLM 配置

    class AssistantLLMConfig(BaseModel):llm_list: Optional[List[AssistantLLMItem]] = Field(default=[], description="助手可选的LLM列表")auto_llm: Optional[AssistantLLMItem] = Field(description="助手画像自动优化模型的配置")
    
    • 用途:表示助手的 LLM 配置。
    • 字段:
      • llm_list:助手可选的 LLM 列表,类型为 List[AssistantLLMItem],默认值为空列表。
      • auto_llm:助手画像自动优化模型的配置,类型为 AssistantLLMItem,可选。
  4. 评测 LLM 配置

    class EvaluationLLMConfig(BaseModel):model_id: Optional[int] = Field(description="评测功能默认模型的ID")
    
    • 用途:表示评测功能的 LLM 配置。
    • 字段:
      • model_id:评测功能默认模型的 ID,可选。
文件处理模型
  1. 文件处理基础请求

    class FileProcessBase(BaseModel):knowledge_id: int = Field(..., description="知识库ID")separator: Optional[List[str]] = Field(default=['\n\n'], description="切分文本规则, 不传则为默认")separator_rule: Optional[List[str]] = Field(default=['after'],description="切分规则前还是后进行切分;before/after")chunk_size: Optional[int] = Field(default=1000, description="切分文本长度,不传则为默认")chunk_overlap: Optional[int] = Field(default=100, description="切分文本重叠长度,不传则为默认")@root_validatordef check_separator_rule(cls, values):if values['separator'] is None:values['separator'] = ['\n\n']if values['separator_rule'] is None:values['separator_rule'] = ['after' for _ in values["separator"]]if values['chunk_size'] is None:values['chunk_size'] = 1000if values['chunk_overlap'] is None:values['chunk_overlap'] = 100return values
    
    • 用途:表示文件处理的基础请求参数,用于文件切分。
    • 字段
      • knowledge_id:知识库的唯一标识符。
      • separator:切分文本的规则,默认值为 ['\n\n'],可选。
      • separator_rule:切分规则前还是后进行切分,默认值为 ['after'],可选。
      • chunk_size:切分文本的长度,默认值为 1000,可选。
      • chunk_overlap:切分文本的重叠长度,默认值为 100,可选。
    • 验证器
      • check_separator_rule:确保所有可选字段在未提供时设置为默认值。
  2. 文件分块元数据

    class FileChunkMetadata(BaseModel):source: str = Field(default='', description="源文件名")title: str = Field(default='', description="源文件内容总结的标题")chunk_index: int = Field(default=0, description="文本块索引")bbox: str = Field(default='', description="文本块bbox信息")page: int = Field(default=0, description="文本块所在页码")extra: str = Field(default='', description="文本块额外信息")file_id: int = Field(default=0, description="文本块所属文件ID")
    
    • 用途:表示文件分块的元数据信息。
    • 字段:
      • source:源文件名,默认值为空字符串。
      • title:源文件内容总结的标题,默认值为空字符串。
      • chunk_index:文本块的索引,默认值为 0
      • bbox:文本块的 bounding box 信息,默认值为空字符串。
      • page:文本块所在的页码,默认值为 0
      • extra:文本块的额外信息,默认值为空字符串。
      • file_id:文本块所属文件的唯一标识符,默认值为 0
  3. 文件分块

    class FileChunk(BaseModel):text: str = Field(..., description="文本块内容")parse_type: Optional[str] = Field(default=None, description="文本所属的文件解析类型")metadata: FileChunkMetadata = Field(..., description="文本块元数据")
    
    • 用途:表示文件的一个分块。
    • 字段:
      • text:文本块的内容。
      • parse_type:文本所属的文件解析类型,可选。
      • metadata:文本块的元数据,类型为 FileChunkMetadata
  4. 预览文件分块请求

    class PreviewFileChunk(FileProcessBase):file_path: str = Field(..., description="文件路径")cache: bool = Field(default=True, description="是否从缓存获取")
    
    • 用途:表示预览文件分块内容的请求。
    • 字段:
      • file_path:文件的路径。
      • cache:是否从缓存获取,默认值为 True
  5. 更新预览文件分块

    class UpdatePreviewFileChunk(BaseModel):knowledge_id: int = Field(..., description="知识库ID")file_path: str = Field(..., description="文件路径")text: str = Field(..., description="文本块内容")chunk_index: int = Field(..., description="文本块索引, 在metadata里")bbox: Optional[str] = Field(default='', description="文本块bbox信息")
    
    • 用途:表示更新预览文件分块内容的请求。
    • 字段:
      • knowledge_id:知识库的唯一标识符。
      • file_path:文件的路径。
      • text:文本块的内容。
      • chunk_index:文本块的索引。
      • bbox:文本块的 bounding box 信息,可选,默认值为空字符串。
  6. 知识库文件单项

    class KnowledgeFileOne(BaseModel):file_path: str = Field(..., description="文件路径")
    
    • 用途:表示知识库中单个文件的路径。
    • 字段:
      • file_path:文件的路径。
  7. 知识库文件处理

    class KnowledgeFileProcess(FileProcessBase):file_list: List[KnowledgeFileOne] = Field(..., description="文件列表")callback_url: Optional[str] = Field(default=None, description="异步任务回调地址")extra: Optional[str] = Field(default=None, description="附加信息")
    
    • 用途:表示知识库文件处理的请求。
    • 字段:
      • file_list:文件列表,类型为 List[KnowledgeFileOne]
      • callback_url:异步任务的回调地址,可选。
      • extra:附加信息,可选。

总结

这段代码定义了一系列用于数据验证和序列化的模型,涵盖了用户认证、聊天管理、知识库处理、模型和服务器配置等多个方面。以下是关键点总结:

  1. 数据验证和序列化:通过 PydanticBaseModel,确保 API 接收和返回的数据结构符合预期,提升代码的可靠性和可维护性。
  2. 通用响应结构:使用泛型的 UnifiedResponseModel 和辅助函数 resp_200resp_500,统一 API 的响应格式,便于前端处理和错误管理。
  3. 业务逻辑覆盖
    • 用户认证:如 CaptchaInputCreateUserReq
    • 聊天管理:如 ChatMessageChatResponseAddChatMessages
    • 知识库处理:如 KnowledgeFileProcessFileChunk
    • 模型和服务器配置:如 LLMModelCreateReqLLMServerCreateReq
    • 流程管理:如 FlowListCreateFlowCompareReq
  4. 验证器:通过 @validator@root_validator,进一步确保数据的合法性和完整性。
  5. 类型注解:广泛使用类型注解(如 OptionalListDict 等),提高代码的可读性和静态分析工具的效果。

base.py

import uuid
from contextlib import contextmanagerfrom bisheng.database.service import DatabaseService
from bisheng.settings import settings
from bisheng.utils.logger import logger
from sqlmodel import Sessiondb_service: 'DatabaseService' = DatabaseService(settings.database_url)@contextmanager
def session_getter() -> Session:"""轻量级session context"""try:session = Session(db_service.engine)yield sessionexcept Exception as e:logger.info('Session rollback because of exception:{}', e)session.rollback()raisefinally:session.close()def generate_uuid() -> str:"""生成uuid的字符串"""return uuid.uuid4().hex

1. 导入部分

标准库导入

import uuid
from contextlib import contextmanager
  • uuid
    • 用途:用于生成唯一标识符(UUID)。
    • 常用功能:
      • uuid.uuid4():生成一个随机的 UUID。
      • uuid.uuid4().hex:获取 UUID 的 32 位十六进制字符串表示。
  • contextmanager
    • 用途:用于创建上下文管理器,通常与 with 语句一起使用。
    • 功能:简化上下文管理器的创建,避免手动编写类。

项目内部模块导入

from bisheng.database.service import DatabaseService
from bisheng.settings import settings
from bisheng.utils.logger import logger
  • bisheng.database.service.DatabaseService
    • 用途:数据库服务类,通常封装了与数据库的连接和操作逻辑。
    • 假设功能:
      • 初始化数据库引擎。
      • 提供数据库会话管理。
      • 可能包含其他数据库相关的服务方法。
  • bisheng.settings.settings
    • 用途:项目的配置对象,包含了各种配置参数,如数据库 URL、API 密钥等。
    • 功能:通过 settings.database_url 访问数据库连接字符串。
  • bisheng.utils.logger.logger
    • 用途:日志记录器,用于记录日志信息。
    • 功能:
      • 记录信息、警告、错误等不同级别的日志。
      • 便于调试和监控应用程序运行状态。

第三方库导入

from sqlmodel import Session
  • sqlmodel.Session
    • 用途:用于与数据库交互的会话对象。
    • 功能:
      • 提供 CRUD(创建、读取、更新、删除)操作。
      • 管理数据库事务。
      • 处理数据库连接的打开和关闭。

2. 数据库服务实例化

db_service: 'DatabaseService' = DatabaseService(settings.database_url)
  • db_service
    • 类型注解'DatabaseService'(使用字符串注解以避免循环导入问题)。
    • 用途:创建一个 DatabaseService 的实例,用于管理与数据库的连接。
    • 初始化:
      • 参数settings.database_url,即数据库的连接字符串。
      • 功能:内部会根据 database_url 配置数据库引擎(如 SQLAlchemy 引擎)。

示例

假设 settings.database_url"postgresql://user:password@localhost/dbname",那么 DatabaseService 会使用这个 URL 初始化与 PostgreSQL 数据库的连接。

3. 上下文管理器:session_getter

@contextmanager
def session_getter() -> Session:"""轻量级session context"""try:session = Session(db_service.engine)yield sessionexcept Exception as e:logger.info('Session rollback because of exception:{}', e)session.rollback()raisefinally:session.close()

功能和用途

  • 目的:提供一个安全且便捷的方式来管理数据库会话,确保会话在使用后正确关闭,并在发生异常时回滚事务。
  • 使用场景:
    • 在执行数据库操作时使用 with session_getter() as session:,确保会话的正确管理。
    • 处理事务,确保数据一致性和完整性。

工作原理

  1. 创建会话

    session = Session(db_service.engine)
    
    • db_service.engine:获取数据库引擎,用于创建会话。
  2. 使用 yield 传递会话对象

    yield session
    
    • 允许 with 语句块内的代码使用该会话对象。
  3. 异常处理

    except Exception as e:logger.info('Session rollback because of exception:{}', e)session.rollback()raise
    
    • 记录日志:使用 logger 记录异常信息。
    • 事务回滚:确保在发生异常时,事务不会部分提交,保持数据一致性。
    • 重新抛出异常:让上层调用者知道发生了错误。
  4. 关闭会话

    finally:session.close()
    
    • 确保资源释放:无论是否发生异常,都会关闭会话,释放数据库连接资源。

示例用法

from bisheng.database.utils import session_getterdef get_user(user_id: int):with session_getter() as session:user = session.query(User).filter(User.id == user_id).first()return user

在这个示例中:

  • with session_getter() as session:获取一个数据库会话。
  • 执行查询:通过会话对象执行数据库查询。
  • 自动管理:退出 with 块时,会自动关闭会话,确保资源释放。

4. UUID 生成函数:generate_uuid

def generate_uuid() -> str:"""生成uuid的字符串"""return uuid.uuid4().hex

功能和用途

  • 目的:生成一个唯一的 UUID 字符串,用于标识对象或记录。
  • 用途:
    • 为数据库记录生成唯一标识符。
    • 生成会话 ID、交易 ID 等。
    • 确保标识符的唯一性,避免冲突。

工作原理

  1. 生成 UUID

    uuid.uuid4()
    
    • uuid4:生成一个随机的 UUID(基于随机数)。
  2. 获取十六进制表示

    .hex
    
    • hex 属性:将 UUID 对象转换为 32 位十六进制字符串,没有连字符(-)。

示例

unique_id = generate_uuid()
print(unique_id)  # 输出示例:e4eaaaf2dcd24d5f8e4e8c1b0b3f1f6a
  • 优势

    • 唯一性:UUID 生成的概率几乎为零,确保每个 ID 的唯一性。
    • 不可预测性:基于随机数生成,难以预测下一个 UUID 的值。

注意事项

  • 长度:UUID 的 hex 表示长度为 32 个字符。
  • 存储:在数据库中存储时,考虑使用合适的数据类型(如 VARCHAR(32)CHAR(32))。
  • 性能:UUID 的随机性可能导致数据库索引性能下降,特别是在高频率写入的情况下。可以考虑使用有序 UUID(如 UUID1)以优化索引性能。

5. 总结

这段代码主要涵盖了以下几个关键方面:

  1. 数据库服务初始化
    • 使用 DatabaseService 类初始化数据库连接,基于项目配置中的 database_url
  2. 会话管理
    • 定义了一个上下文管理器 session_getter,用于安全地管理数据库会话,确保会话在使用后正确关闭,并在异常发生时回滚事务。
  3. UUID 生成
    • 提供了一个简单的函数 generate_uuid,用于生成唯一的 UUID 字符串,确保标识符的唯一性和不可预测性。

如何在项目中使用

  • 数据库操作

    • 使用 session_getter 上下文管理器来执行数据库操作,确保会话的正确管理和资源的释放。
    from bisheng.database.utils import session_getter, generate_uuiddef create_user(name: str, email: str):user_id = generate_uuid()with session_getter() as session:new_user = User(id=user_id, name=name, email=email)session.add(new_user)session.commit()return new_user
    
  • 生成唯一标识符

    • 使用 generate_uuid 函数为新记录生成唯一的 ID。
    unique_id = generate_uuid()
    print(f"Generated UUID: {unique_id}")
    

进一步阅读

  • DatabaseService
    • 查看 bisheng.database.service 模块中的 DatabaseService 类实现,了解其具体功能和方法。
  • 项目配置
    • 查看 bisheng.settings 模块,了解项目的配置管理,尤其是 database_url 的设置方式。
  • 日志记录器
    • 查看 bisheng.utils.logger 模块,了解日志记录器的配置和使用方法,以便更好地调试和监控应用程序。
  • sqlmodel
    • sqlmodel 是一个用于与 SQL 数据库交互的库,结合了 SQLAlchemy 和 Pydantic 的特性。建议阅读 SQLModel 官方文档 以深入理解其用法和最佳实践。

常见问题

  1. 为什么使用 contextmanager 装饰器创建 session_getter
    • 使用 contextmanager 可以简化上下文管理器的创建,使代码更加简洁和易读。它允许使用 with 语句块自动管理资源,如数据库会话的打开和关闭。
  2. 为什么在异常处理中回滚事务?
    • 当数据库操作过程中发生异常时,事务可能处于不一致状态。回滚事务可以确保数据库不会部分提交数据,保持数据的一致性和完整性。
  3. UUID 为什么使用 .hex 属性而不是直接使用 uuid4() 生成的对象?
    • 使用 .hex 属性可以获得一个标准的 32 位十六进制字符串表示,便于存储和传输。相比直接使用 UUID 对象,字符串表示更容易与其他系统集成和调试。

UserService.py

这段代码主要涉及用户身份验证、权限管理、用户创建以及与数据库交互的服务逻辑。代码使用了 FastAPI 框架,并结合了 PydanticJWTRedis 等技术。以下是对每一部分的详细解析:

import functools
import json
from base64 import b64decode
from typing import Listimport rsa
from bisheng.api.errcode.base import UnAuthorizedError
from bisheng.api.errcode.user import (UserLoginOfflineError, UserNameAlreadyExistError,UserNeedGroupAndRoleError)
from bisheng.api.JWT import ACCESS_TOKEN_EXPIRE_TIME
from bisheng.api.utils import md5_hash
from bisheng.api.v1.schemas import CreateUserReq
from bisheng.cache.redis import redis_client
from bisheng.database.models.assistant import Assistant, AssistantDao
from bisheng.database.models.flow import Flow, FlowDao, FlowRead
from bisheng.database.models.knowledge import Knowledge, KnowledgeDao, KnowledgeRead
from bisheng.database.models.role import AdminRole
from bisheng.database.models.role_access import AccessType, RoleAccessDao
from bisheng.database.models.user import User, UserDao
from bisheng.database.models.user_group import UserGroupDao
from bisheng.database.models.user_role import UserRoleDao
from bisheng.settings import settings
from bisheng.utils.constants import RSA_KEY, USER_CURRENT_SESSION
from fastapi import Depends, HTTPException, Request
from fastapi_jwt_auth import AuthJWT

1. 导入部分

标准库导入

import functools
import json
from base64 import b64decode
from typing import List
  • functools

    • 用途:用于高级函数操作,如装饰器。

    • 常用功能

      • functools.wraps:用于装饰器,保持原函数的元数据。
  • json

    • 用途:用于 JSON 数据的序列化和反序列化。

    • 常用功能

      • json.loads:将 JSON 字符串转换为 Python 对象。
      • json.dumps:将 Python 对象转换为 JSON 字符串。
  • base64.b64decode

    • 用途:用于解码 Base64 编码的数据。
    • 应用场景:常用于处理加密数据或传输二进制数据。
  • typing.List

    • 用途:用于类型注解,表示列表类型。
    • 示例List[int] 表示整数列表。

第三方库导入

import rsa
from fastapi import Depends, HTTPException, Request
from fastapi_jwt_auth import AuthJWT
  • rsa

    • 用途:用于 RSA 加密和解密。

    • 常用功能

      • rsa.decrypt:使用私钥解密数据。
      • rsa.encrypt:使用公钥加密数据。
  • fastapi

    • 用途:用于构建 Web API。

    • 常用功能

      • Depends:用于依赖注入。
      • HTTPException:用于抛出 HTTP 异常。
      • Request:表示请求对象。
  • fastapi_jwt_auth.AuthJWT

    • 用途:用于处理 JWT(JSON Web Token)认证。

    • 常用功能

      • create_access_token:创建访问令牌。
      • create_refresh_token:创建刷新令牌。
      • jwt_required:装饰器,用于保护路由,需要 JWT 认证。

项目内部模块导入

from bisheng.api.errcode.base import UnAuthorizedError
from bisheng.api.errcode.user import (UserLoginOfflineError, UserNameAlreadyExistError,UserNeedGroupAndRoleError)
from bisheng.api.JWT import ACCESS_TOKEN_EXPIRE_TIME
from bisheng.api.utils import md5_hash
from bisheng.api.v1.schemas import CreateUserReq
from bisheng.cache.redis import redis_client
from bisheng.database.models.assistant import Assistant, AssistantDao
from bisheng.database.models.flow import Flow, FlowDao, FlowRead
from bisheng.database.models.knowledge import Knowledge, KnowledgeDao, KnowledgeRead
from bisheng.database.models.role import AdminRole
from bisheng.database.models.role_access import AccessType, RoleAccessDao
from bisheng.database.models.user import User, UserDao
from bisheng.database.models.user_group import UserGroupDao
from bisheng.database.models.user_role import UserRoleDao
from bisheng.settings import settings
from bisheng.utils.constants import RSA_KEY, USER_CURRENT_SESSION
  • bisheng.api.errcode.\*

    • 用途:定义自定义的 API 错误码和异常类,用于统一错误处理。

    • 常用异常

      • UnAuthorizedError:未授权错误。
      • UserLoginOfflineError:用户登录离线错误。
      • UserNameAlreadyExistError:用户名已存在错误。
      • UserNeedGroupAndRoleError:用户需要组和角色错误。
  • bisheng.api.JWT

    • 用途:存储 JWT 相关的配置常量,如访问令牌过期时间。

    • 字段

      • ACCESS_TOKEN_EXPIRE_TIME:访问令牌的过期时间。
  • bisheng.api.utils

    • 用途:存储各种实用函数。

    • 常用函数

      • md5_hash:生成字符串的 MD5 哈希值。
  • bisheng.api.v1.schemas.CreateUserReq

    • 用途:定义创建用户请求的数据模型,通常使用 Pydantic BaseModel
  • bisheng.cache.redis.redis_client

    • 用途:Redis 客户端实例,用于与 Redis 缓存交互。
  • bisheng.database.models.\*

    • 用途:导入数据库模型和数据访问对象(DAO)。

    • 常用模型和 DAO

      • Assistant, AssistantDao:助手模型和 DAO。
      • Flow, FlowDao, FlowRead:流程模型、DAO 和读取模型。
      • Knowledge, KnowledgeDao, KnowledgeRead:知识库模型、DAO 和读取模型。
      • AdminRole:管理员角色模型。
      • AccessType, RoleAccessDao:访问类型和角色访问 DAO。
      • User, UserDao:用户模型和 DAO。
      • UserGroupDao:用户组 DAO。
      • UserRoleDao:用户角色 DAO。
  • bisheng.settings.settings

    • 用途:项目的配置对象,包含了各种配置参数,如数据库 URL、RSA 密钥等。
  • bisheng.utils.constants

    • 用途:存储项目中使用的常量。

    • 字段

      • RSA_KEY:RSA 密钥的常量键。
      • USER_CURRENT_SESSION:用户当前会话的 Redis 键模板。

2. UserPayload

class UserPayload:def __init__(self, **kwargs):self.user_id = kwargs.get('user_id')self.user_role = kwargs.get('role')if self.user_role != 'admin':  # 非管理员用户,需要获取他的角色列表roles = UserRoleDao.get_user_roles(self.user_id)self.user_role = [one.role_id for one in roles]self.user_name = kwargs.get('user_name')def is_admin(self):if self.user_role == 'admin':return Trueif isinstance(self.user_role, list):for one in self.user_role:if one == AdminRole:return Truereturn False@staticmethoddef wrapper_access_check(func):"""权限检查的装饰器如果是admin用户则不执行后续具体的检查逻辑"""@functools.wraps(func)def wrapper(*args, **kwargs):if args[0].is_admin():return Truereturn func(*args, **kwargs)return wrapper@wrapper_access_checkdef access_check(self, owner_user_id: int, target_id: str, access_type: AccessType) -> bool:"""检查用户是否有某个资源的权限"""# 判断是否属于本人资源if self.user_id == owner_user_id:return True# 判断授权if RoleAccessDao.judge_role_access(self.user_role, target_id, access_type):return Truereturn False@wrapper_access_checkdef check_group_admin(self, group_id: int) -> bool:"""检查用户是否是某个组的管理员"""# 判断是否是用户组的管理员user_group = UserGroupDao.get_user_admin_group(self.user_id)if not user_group:return Falsefor one in user_group:if one.group_id == group_id:return Truereturn False@wrapper_access_checkdef check_groups_admin(self, group_ids: List[int]) -> bool:"""检查用户是否是用户组列表中的管理员,有一个就是true"""user_groups = UserGroupDao.get_user_admin_group(self.user_id)for one in user_groups:if one.is_group_admin and one.group_id in group_ids:return Truereturn False

功能和用途

UserPayload 类用于封装当前登录用户的相关信息,并提供权限检查的方法。这在构建基于角色的访问控制(RBAC)系统中非常常见。

详细解析

构造函数 __init__
def __init__(self, **kwargs):self.user_id = kwargs.get('user_id')self.user_role = kwargs.get('role')if self.user_role != 'admin':  # 非管理员用户,需要获取他的角色列表roles = UserRoleDao.get_user_roles(self.user_id)self.user_role = [one.role_id for one in roles]self.user_name = kwargs.get('user_name')
  • 功能:初始化用户的基本信息,包括 user_iduser_roleuser_name

  • 逻辑

    • kwargs 获取 user_idroleuser_name
    • 如果用户不是管理员 (self.user_role != 'admin'),则从数据库获取该用户的所有角色,并将其存储为角色 ID 的列表。
方法 is_admin
def is_admin(self):if self.user_role == 'admin':return Trueif isinstance(self.user_role, list):for one in self.user_role:if one == AdminRole:return Truereturn False
  • 功能:判断用户是否具有管理员权限。

  • 逻辑

    • 如果 user_role'admin',返回 True
    • 如果 user_role 是一个列表,遍历列表,检查是否包含 AdminRole,如果包含,返回 True
    • 否则,返回 False
装饰器 wrapper_access_check
@staticmethod
def wrapper_access_check(func):"""权限检查的装饰器如果是admin用户则不执行后续具体的检查逻辑"""@functools.wraps(func)def wrapper(*args, **kwargs):if args[0].is_admin():return Truereturn func(*args, **kwargs)return wrapper
  • 功能:定义一个装饰器,用于权限检查。

  • 逻辑

    • 如果用户是管理员,直接返回 True,跳过后续的具体权限检查逻辑。
    • 否则,执行被装饰的函数。
方法 access_check
@wrapper_access_check
def access_check(self, owner_user_id: int, target_id: str, access_type: AccessType) -> bool:"""检查用户是否有某个资源的权限"""# 判断是否属于本人资源if self.user_id == owner_user_id:return True# 判断授权if RoleAccessDao.judge_role_access(self.user_role, target_id, access_type):return Truereturn False
  • 功能:检查用户是否有访问某个资源的权限。

  • 逻辑

    • 如果资源的拥有者是当前用户,返回 True
    • 否则,调用 RoleAccessDao.judge_role_access 方法检查用户角色是否有访问该资源的权限。
    • 返回相应的布尔值。
方法 check_group_admin
@wrapper_access_check
def check_group_admin(self, group_id: int) -> bool:"""检查用户是否是某个组的管理员"""# 判断是否是用户组的管理员user_group = UserGroupDao.get_user_admin_group(self.user_id)if not user_group:return Falsefor one in user_group:if one.group_id == group_id:return Truereturn False
  • 功能:检查用户是否是某个用户组的管理员。

  • 逻辑

    • 调用 UserGroupDao.get_user_admin_group 获取用户所属的管理员组。
    • 如果用户不属于任何管理员组,返回 False
    • 遍历管理员组,检查是否包含指定的 group_id,如果包含,返回 True
    • 否则,返回 False
方法 check_groups_admin
@wrapper_access_check
def check_groups_admin(self, group_ids: List[int]) -> bool:"""检查用户是否是用户组列表中的管理员,有一个就是true"""user_groups = UserGroupDao.get_user_admin_group(self.user_id)for one in user_groups:if one.is_group_admin and one.group_id in group_ids:return Truereturn False
  • 功能:检查用户是否是给定用户组列表中的任意一个用户组的管理员。

  • 逻辑

    • 调用 UserGroupDao.get_user_admin_group 获取用户所属的管理员组。
    • 遍历管理员组,检查是否有任何一个用户组的 group_id 在给定的 group_ids 列表中,并且用户是该组的管理员 (one.is_group_admin)。
    • 如果找到符合条件的用户组,返回 True
    • 否则,返回 False

3. UserService

class UserService:@classmethoddef decrypt_md5_password(cls, password: str):if value := redis_client.get(RSA_KEY):private_key = value[1]password = md5_hash(rsa.decrypt(b64decode(password), private_key).decode('utf-8'))else:password = md5_hash(password)return password@classmethoddef create_user(cls, request: Request, login_user: UserPayload, req_data: CreateUserReq):"""创建用户"""exists_user = UserDao.get_user_by_username(req_data.user_name)if exists_user:# 抛出异常raise UserNameAlreadyExistError.http_exception()user = User(user_name=req_data.user_name,password=cls.decrypt_md5_password(req_data.password),)group_ids = []role_ids = []for one in req_data.group_roles:group_ids.append(one.group_id)role_ids.extend(one.role_ids)if not group_ids or not role_ids:raise UserNeedGroupAndRoleError.http_exception()user = UserDao.add_user_with_groups_and_roles(user, group_ids, role_ids)return user

功能和用途

UserService 类提供了与用户相关的业务逻辑,如密码解密和用户创建。这种类通常包含静态方法或类方法,用于处理特定的业务操作。

详细解析

方法 decrypt_md5_password
@classmethod
def decrypt_md5_password(cls, password: str):if value := redis_client.get(RSA_KEY):private_key = value[1]password = md5_hash(rsa.decrypt(b64decode(password), private_key).decode('utf-8'))else:password = md5_hash(password)return password
  • 功能:解密密码并生成 MD5 哈希值。
  • 逻辑
    • 尝试从 Redis 获取 RSA 密钥 (RSA_KEY)。
    • 如果存在 RSA 密钥:
      • 使用 RSA 私钥解密 Base64 编码的密码。
      • 将解密后的密码转换为 UTF-8 字符串。
      • 对解密后的密码进行 MD5 哈希。
    • 如果不存在 RSA 密钥:
      • 直接对原始密码进行 MD5 哈希。
    • 返回哈希后的密码。
  • 应用场景
    • 用户注册或密码更新时,将用户输入的密码加密存储到数据库。
    • 增强密码的安全性,通过加密和哈希减少明文密码的存储风险。
方法 create_user
@classmethod
def create_user(cls, request: Request, login_user: UserPayload, req_data: CreateUserReq):"""创建用户"""exists_user = UserDao.get_user_by_username(req_data.user_name)if exists_user:# 抛出异常raise UserNameAlreadyExistError.http_exception()user = User(user_name=req_data.user_name,password=cls.decrypt_md5_password(req_data.password),)group_ids = []role_ids = []for one in req_data.group_roles:group_ids.append(one.group_id)role_ids.extend(one.role_ids)if not group_ids or not role_ids:raise UserNeedGroupAndRoleError.http_exception()user = UserDao.add_user_with_groups_and_roles(user, group_ids, role_ids)return user
  • 功能:创建新用户,并将其分配到指定的用户组和角色中。
  • 参数
    • request:请求对象,包含请求相关的信息。
    • login_user:当前登录用户的信息,类型为 UserPayload
    • req_data:创建用户的请求数据,类型为 CreateUserReq
  • 逻辑
    1. 检查用户名是否存在
      • 调用 UserDao.get_user_by_username 检查请求中的用户名是否已经存在。
      • 如果用户已存在,抛出 UserNameAlreadyExistError 异常。
    2. 创建用户实例
      • 使用 CreateUserReq 中的 user_name 和解密后的密码创建一个新的 User 实例。
    3. 收集用户组和角色 ID
      • 遍历 req_data.group_roles,收集所有的 group_idrole_id
    4. 检查用户组和角色
      • 如果用户没有指定任何用户组或角色,抛出 UserNeedGroupAndRoleError 异常。
    5. 添加用户并关联组和角色
      • 调用 UserDao.add_user_with_groups_and_roles 方法,将用户添加到数据库,并关联指定的用户组和角色。
    6. 返回创建的用户
      • 返回新创建的 User 实例。
  • 应用场景
    • 用户注册功能,允许管理员或系统自动创建新用户,并分配相应的权限和组。

4. 辅助函数

函数 sso_login

def sso_login():pass
  • 功能:此函数目前为空,实现细节未提供。
  • 用途:通常用于单点登录(SSO)功能,允许用户通过单一认证源登录多个系统。

函数 gen_user_role

def gen_user_role(db_user: User):# 查询用户的角色列表db_user_role = UserRoleDao.get_user_roles(db_user.user_id)role = ''role_ids = []for user_role in db_user_role:if user_role.role_id == 1:# 是管理员,忽略其他的角色role = 'admin'else:role_ids.append(user_role.role_id)if role != 'admin':# 判断是否是用户组管理员db_user_groups = UserGroupDao.get_user_admin_group(db_user.user_id)if len(db_user_groups) > 0:role = 'group_admin'else:role = role_ids# 获取用户的菜单栏权限列表web_menu = RoleAccessDao.get_role_access(role_ids, AccessType.WEB_MENU)web_menu = list(set([one.third_id for one in web_menu]))return role, web_menu
  • 功能:生成用户的角色和相应的菜单权限。
  • 参数
    • db_user:数据库中的 User 实例。
  • 逻辑
    1. 查询用户角色
      • 调用 UserRoleDao.get_user_roles 获取用户的角色列表。
    2. 确定角色
      • 遍历用户角色列表:
        • 如果用户具有角色 ID 为 1,将角色设置为 'admin',并忽略其他角色。
        • 否则,将角色 ID 添加到 role_ids 列表中。
    3. 检查用户组管理员身份
      • 如果用户不是管理员,调用 UserGroupDao.get_user_admin_group 检查用户是否是某个用户组的管理员。
      • 如果是用户组管理员,设置角色为 'group_admin'
      • 否则,角色保持为角色 ID 的列表。
    4. 获取菜单权限
      • 调用 RoleAccessDao.get_role_access 获取用户角色对应的菜单权限。
      • 提取并去重 third_id,形成菜单权限列表 web_menu
    5. 返回角色和菜单权限
      • 返回用户的角色(字符串或列表)和菜单权限列表。
  • 应用场景
    • 在用户登录后,生成用户的权限信息,用于前端展示和后续的权限校验。

函数 gen_user_jwt

def gen_user_jwt(db_user: User):if 1 == db_user.delete:raise HTTPException(status_code=500, detail='该账号已被禁用,请联系管理员')# 查询角色role, web_menu = gen_user_role(db_user)# 生成JWT令牌payload = {'user_name': db_user.user_name, 'user_id': db_user.user_id, 'role': role}# Create the tokens and passing to set_access_cookies or set_refresh_cookiesaccess_token = AuthJWT().create_access_token(subject=json.dumps(payload),expires_time=ACCESS_TOKEN_EXPIRE_TIME)refresh_token = AuthJWT().create_refresh_token(subject=db_user.user_name)# Set the JWT cookies in the responsereturn access_token, refresh_token, role, web_menu
  • 功能:为用户生成 JWT 访问令牌和刷新令牌。
  • 参数
    • db_user:数据库中的 User 实例。
  • 逻辑
    1. 检查用户状态
      • 如果 db_user.delete 等于 1,表示账号被禁用,抛出 HTTPException,状态码为 500,提示用户联系管理员。
    2. 获取用户角色和菜单权限
      • 调用 gen_user_role 函数,获取用户的角色和菜单权限。
    3. 生成 JWT 载荷
      • 创建一个包含 user_nameuser_idrole 的字典 payload
    4. 生成访问令牌和刷新令牌
      • 调用 AuthJWT().create_access_token 生成访问令牌,主题为 payload 的 JSON 字符串,过期时间为 ACCESS_TOKEN_EXPIRE_TIME
      • 调用 AuthJWT().create_refresh_token 生成刷新令牌,主题为用户的用户名。
    5. 返回令牌和权限信息
      • 返回访问令牌、刷新令牌、角色和菜单权限。
  • 应用场景
    • 在用户成功登录后,生成并返回 JWT 令牌,用于后续的认证和授权。

函数 get_knowledge_list_by_access

def get_knowledge_list_by_access(role_id: int, name: str, page_num: int, page_size: int):count_filter = []if name:count_filter.append(Knowledge.name.like('%{}%'.format(name)))db_role_access = KnowledgeDao.get_knowledge_by_access(role_id, page_num, page_size)total_count = KnowledgeDao.get_count_by_filter(count_filter)# 补充用户名user_ids = [access[0].user_id for access in db_role_access]db_users = UserDao.get_user_by_ids(user_ids)user_dict = {user.user_id: user.user_name for user in db_users}return {'data': [KnowledgeRead.validate({'name': access[0].name,'user_name': user_dict.get(access[0].user_id),'user_id': access[0].user_id,'update_time': access[0].update_time,'id': access[0].id}) for access in db_role_access],'total':total_count}
  • 功能:根据用户角色获取有访问权限的知识库列表,并支持分页和名称过滤。
  • 参数
    • role_id:用户的角色 ID。
    • name:知识库名称的搜索关键字。
    • page_num:分页的页码。
    • page_size:每页显示的记录数。
  • 逻辑
    1. 构建过滤条件
      • 如果提供了 name,则添加一个 SQL LIKE 过滤条件,用于模糊匹配知识库名称。
    2. 查询有访问权限的知识库
      • 调用 KnowledgeDao.get_knowledge_by_access,传入 role_idpage_numpage_size,获取用户有权限访问的知识库列表。
    3. 查询总数
      • 调用 KnowledgeDao.get_count_by_filter,传入过滤条件,获取满足条件的知识库总数。
    4. 补充用户名
      • 从知识库访问列表中提取所有的 user_id
      • 调用 UserDao.get_user_by_ids 获取对应的用户信息。
      • 构建一个字典 user_dict,将 user_id 映射到 user_name
    5. 构建响应数据
      • 遍历 db_role_access,为每个知识库构建一个包含 nameuser_nameuser_idupdate_timeid 的字典,并通过 KnowledgeRead.validate 进行数据验证和序列化。
      • 返回包含 datatotal 的字典。
  • 应用场景
    • 用于前端页面展示用户有权限访问的知识库列表,并支持搜索和分页功能。

函数 get_flow_list_by_access

def get_flow_list_by_access(role_id: int, name: str, page_num: int, page_size: int):count_filter = []if name:count_filter.append(Flow.name.like('%{}%'.format(name)))db_role_access = FlowDao.get_flow_by_access(role_id, name, page_num, page_size)total_count = FlowDao.get_count_by_filters(count_filter)# 补充用户名user_ids = [access[0].user_id for access in db_role_access]db_users = UserDao.get_user_by_ids(user_ids)user_dict = {user.user_id: user.user_name for user in db_users}return {'data': [FlowRead.validate({'name': access[0].name,'user_name': user_dict.get(access[0].user_id),'user_id': access[0].user_id,'update_time': access[0].update_time,'id': access[0].id}) for access in db_role_access],'total':total_count}
  • 功能:根据用户角色获取有访问权限的流程列表,并支持分页和名称过滤。
  • 参数
    • role_id:用户的角色 ID。
    • name:流程名称的搜索关键字。
    • page_num:分页的页码。
    • page_size:每页显示的记录数。
  • 逻辑
    1. 构建过滤条件
      • 如果提供了 name,则添加一个 SQL LIKE 过滤条件,用于模糊匹配流程名称。
    2. 查询有访问权限的流程
      • 调用 FlowDao.get_flow_by_access,传入 role_idnamepage_numpage_size,获取用户有权限访问的流程列表。
    3. 查询总数
      • 调用 FlowDao.get_count_by_filters,传入过滤条件,获取满足条件的流程总数。
    4. 补充用户名
      • 从流程访问列表中提取所有的 user_id
      • 调用 UserDao.get_user_by_ids 获取对应的用户信息。
      • 构建一个字典 user_dict,将 user_id 映射到 user_name
    5. 构建响应数据
      • 遍历 db_role_access,为每个流程构建一个包含 nameuser_nameuser_idupdate_timeid 的字典,并通过 FlowRead.validate 进行数据验证和序列化。
      • 返回包含 datatotal 的字典。
  • 应用场景
    • 用于前端页面展示用户有权限访问的流程列表,并支持搜索和分页功能。

函数 get_assistant_list_by_access

def get_assistant_list_by_access(role_id: int, name: str, page_num: int, page_size: int):count_filter = []if name:count_filter.append(Assistant.name.like('%{}%'.format(name)))db_role_access = AssistantDao.get_assistants_by_access(role_id, name, page_size, page_num)total_count = AssistantDao.get_count_by_filters(count_filter)# 补充用户名user_ids = [access[0].user_id for access in db_role_access]db_users = UserDao.get_user_by_ids(user_ids)user_dict = {user.user_id: user.user_name for user in db_users}return {'data': [{'name': access[0].name,'user_name': user_dict.get(access[0].user_id),'user_id': access[0].user_id,'update_time': access[0].update_time,'id': access[0].id} for access in db_role_access],'total':total_count}
  • 功能:根据用户角色获取有访问权限的助手列表,并支持分页和名称过滤。
  • 参数
    • role_id:用户的角色 ID。
    • name:助手名称的搜索关键字。
    • page_num:分页的页码。
    • page_size:每页显示的记录数。
  • 逻辑
    1. 构建过滤条件
      • 如果提供了 name,则添加一个 SQL LIKE 过滤条件,用于模糊匹配助手名称。
    2. 查询有访问权限的助手
      • 调用 AssistantDao.get_assistants_by_access,传入 role_idnamepage_sizepage_num,获取用户有权限访问的助手列表。
    3. 查询总数
      • 调用 AssistantDao.get_count_by_filters,传入过滤条件,获取满足条件的助手总数。
    4. 补充用户名
      • 从助手访问列表中提取所有的 user_id
      • 调用 UserDao.get_user_by_ids 获取对应的用户信息。
      • 构建一个字典 user_dict,将 user_id 映射到 user_name
    5. 构建响应数据
      • 遍历 db_role_access,为每个助手构建一个包含 nameuser_nameuser_idupdate_timeid 的字典。
      • 返回包含 datatotal 的字典。
  • 应用场景
    • 用于前端页面展示用户有权限访问的助手列表,并支持搜索和分页功能。

5. 获取当前登录用户的依赖项

函数 get_login_user

async def get_login_user(authorize: AuthJWT = Depends()) -> UserPayload:"""获取当前登录的用户"""# 校验是否过期,过期则直接返回http 状态码的 401authorize.jwt_required()current_user = json.loads(authorize.get_jwt_subject())user = UserPayload(**current_user)# 判断是否允许多点登录if not settings.get_system_login_method().allow_multi_login:# 获取access_tokencurrent_token = redis_client.get(USER_CURRENT_SESSION.format(user.user_id))# 登录被挤下线了,http状态码是200, status_code是特殊codeif current_token != authorize._token:raise UserLoginOfflineError.http_exception()return user
  • 功能:获取当前登录的用户信息,并进行相关的登录状态检查。
  • 参数
    • authorizeAuthJWT 实例,通过 FastAPI 的 Depends 注入。
  • 逻辑
    1. JWT 验证
      • 调用 authorize.jwt_required(),确保请求包含有效的 JWT 令牌。
      • 如果 JWT 过期或无效,自动抛出 401 Unauthorized 异常。
    2. 获取 JWT 载荷
      • 调用 authorize.get_jwt_subject() 获取 JWT 的主题部分(通常是用户信息的 JSON 字符串)。
      • 使用 json.loads 将 JSON 字符串转换为 Python 字典。
      • 使用 UserPayload 类初始化用户信息对象。
    3. 检查多点登录
      • 调用 settings.get_system_login_method().allow_multi_login 判断系统是否允许多点登录。
      • 如果不允许多点登录:
        • 从 Redis 获取当前用户的 access_token,键格式为 USER_CURRENT_SESSION.format(user.user_id)
        • 比较 Redis 中存储的 current_token 与当前请求中的 _token
        • 如果两者不匹配,说明用户在其他地方登录,抛出 UserLoginOfflineError 异常。
    4. 返回用户信息
      • 返回 UserPayload 实例,包含用户的 ID、角色和用户名。
  • 应用场景
    • 用于保护需要登录才能访问的 API 路由,确保只有认证用户可以访问。
    • 结合 FastAPI 的依赖注入机制,简化路由函数中的用户信息获取。

函数 get_admin_user

async def get_admin_user(authorize: AuthJWT = Depends()) -> UserPayload:"""获取超级管理账号,非超级管理员用户,抛出异常"""login_user = await get_login_user(authorize)if not login_user.is_admin():raise UnAuthorizedError.http_exception()return login_user
  • 功能:获取当前登录的超级管理员用户,如果用户不是管理员,则抛出未授权异常。
  • 参数
    • authorizeAuthJWT 实例,通过 FastAPI 的 Depends 注入。
  • 逻辑
    1. 获取当前登录用户
      • 调用 get_login_user 函数,获取当前登录用户的 UserPayload 实例。
    2. 检查管理员权限
      • 调用 login_user.is_admin() 判断用户是否是管理员。
      • 如果用户不是管理员,抛出 UnAuthorizedError 异常。
    3. 返回管理员用户信息
      • 如果用户是管理员,返回 UserPayload 实例。
  • 应用场景
    • 用于保护需要管理员权限才能访问的 API 路由,确保只有管理员用户可以访问。
    • 结合 FastAPI 的依赖注入机制,简化路由函数中的管理员用户信息获取和权限校验。

6. 辅助函数 gen_user_rolegen_user_jwtget_*_list_by_access

函数 gen_user_role

def gen_user_role(db_user: User):# 查询用户的角色列表db_user_role = UserRoleDao.get_user_roles(db_user.user_id)role = ''role_ids = []for user_role in db_user_role:if user_role.role_id == 1:# 是管理员,忽略其他的角色role = 'admin'else:role_ids.append(user_role.role_id)if role != 'admin':# 判断是否是用户组管理员db_user_groups = UserGroupDao.get_user_admin_group(db_user.user_id)if len(db_user_groups) > 0:role = 'group_admin'else:role = role_ids# 获取用户的菜单栏权限列表web_menu = RoleAccessDao.get_role_access(role_ids, AccessType.WEB_MENU)web_menu = list(set([one.third_id for one in web_menu]))return role, web_menu
  • 功能:生成用户的角色和菜单权限。
  • 参数
    • db_user:数据库中的 User 实例。
  • 逻辑
    1. 查询用户角色
      • 调用 UserRoleDao.get_user_roles 获取用户的角色列表。
    2. 确定用户角色
      • 遍历用户的角色列表:
        • 如果用户拥有角色 ID 为 1,则将角色设置为 'admin',并忽略其他角色。
        • 否则,将角色 ID 添加到 role_ids 列表中。
    3. 检查用户组管理员身份
      • 如果用户不是管理员,调用 UserGroupDao.get_user_admin_group 检查用户是否是某个用户组的管理员。
      • 如果是用户组管理员,设置角色为 'group_admin'
      • 否则,角色保持为角色 ID 的列表。
    4. 获取菜单权限
      • 调用 RoleAccessDao.get_role_access,传入 role_idsAccessType.WEB_MENU,获取用户的菜单权限。
      • 提取并去重 third_id,形成 web_menu 列表。
    5. 返回角色和菜单权限
      • 返回用户的角色(字符串或列表)和菜单权限列表。
  • 应用场景
    • 在用户登录后,生成用户的角色和权限信息,用于前端展示和权限校验。

函数 gen_user_jwt

def gen_user_jwt(db_user: User):if 1 == db_user.delete:raise HTTPException(status_code=500, detail='该账号已被禁用,请联系管理员')# 查询角色role, web_menu = gen_user_role(db_user)# 生成JWT令牌payload = {'user_name': db_user.user_name, 'user_id': db_user.user_id, 'role': role}# Create the tokens and passing to set_access_cookies or set_refresh_cookiesaccess_token = AuthJWT().create_access_token(subject=json.dumps(payload),expires_time=ACCESS_TOKEN_EXPIRE_TIME)refresh_token = AuthJWT().create_refresh_token(subject=db_user.user_name)# Set the JWT cookies in the responsereturn access_token, refresh_token, role, web_menu
  • 功能:为用户生成 JWT 访问令牌和刷新令牌,并返回角色和菜单权限信息。
  • 参数
    • db_user:数据库中的 User 实例。
  • 逻辑
    1. 检查用户状态
      • 如果 db_user.delete 等于 1,表示账号被禁用,抛出 HTTPException,状态码为 500,提示用户联系管理员。
    2. 获取用户角色和菜单权限
      • 调用 gen_user_role 函数,获取用户的角色和菜单权限。
    3. 生成 JWT 载荷
      • 创建一个包含 user_nameuser_idrole 的字典 payload
    4. 生成访问令牌和刷新令牌
      • 调用 AuthJWT().create_access_token 生成访问令牌,主题为 payload 的 JSON 字符串,过期时间为 ACCESS_TOKEN_EXPIRE_TIME
      • 调用 AuthJWT().create_refresh_token 生成刷新令牌,主题为用户的用户名。
    5. 返回令牌和权限信息
      • 返回访问令牌、刷新令牌、角色和菜单权限。
  • 应用场景
    • 在用户成功登录后,生成并返回 JWT 令牌,用于后续的认证和授权。

函数 get_knowledge_list_by_access

def get_knowledge_list_by_access(role_id: int, name: str, page_num: int, page_size: int):count_filter = []if name:count_filter.append(Knowledge.name.like('%{}%'.format(name)))db_role_access = KnowledgeDao.get_knowledge_by_access(role_id, page_num, page_size)total_count = KnowledgeDao.get_count_by_filter(count_filter)# 补充用户名user_ids = [access[0].user_id for access in db_role_access]db_users = UserDao.get_user_by_ids(user_ids)user_dict = {user.user_id: user.user_name for user in db_users}return {'data': [KnowledgeRead.validate({'name': access[0].name,'user_name': user_dict.get(access[0].user_id),'user_id': access[0].user_id,'update_time': access[0].update_time,'id': access[0].id}) for access in db_role_access],'total':total_count}
  • 功能:根据用户角色获取有访问权限的知识库列表,并支持分页和名称过滤。
  • 参数
    • role_id:用户的角色 ID。
    • name:知识库名称的搜索关键字。
    • page_num:分页的页码。
    • page_size:每页显示的记录数。
  • 逻辑
    1. 构建过滤条件
      • 如果提供了 name,则添加一个 SQL LIKE 过滤条件,用于模糊匹配知识库名称。
    2. 查询有访问权限的知识库
      • 调用 KnowledgeDao.get_knowledge_by_access,传入 role_idpage_numpage_size,获取用户有权限访问的知识库列表。
    3. 查询总数
      • 调用 KnowledgeDao.get_count_by_filter,传入过滤条件,获取满足条件的知识库总数。
    4. 补充用户名
      • 从知识库访问列表中提取所有的 user_id
      • 调用 UserDao.get_user_by_ids 获取对应的用户信息。
      • 构建一个字典 user_dict,将 user_id 映射到 user_name
    5. 构建响应数据
      • 遍历 db_role_access,为每个知识库构建一个包含 nameuser_nameuser_idupdate_timeid 的字典,并通过 KnowledgeRead.validate 进行数据验证和序列化。
      • 返回包含 datatotal 的字典。
  • 应用场景
    • 用于前端页面展示用户有权限访问的知识库列表,并支持搜索和分页功能。

函数 get_flow_list_by_access

def get_flow_list_by_access(role_id: int, name: str, page_num: int, page_size: int):count_filter = []if name:count_filter.append(Flow.name.like('%{}%'.format(name)))db_role_access = FlowDao.get_flow_by_access(role_id, name, page_num, page_size)total_count = FlowDao.get_count_by_filters(count_filter)# 补充用户名user_ids = [access[0].user_id for access in db_role_access]db_users = UserDao.get_user_by_ids(user_ids)user_dict = {user.user_id: user.user_name for user in db_users}return {'data': [FlowRead.validate({'name': access[0].name,'user_name': user_dict.get(access[0].user_id),'user_id': access[0].user_id,'update_time': access[0].update_time,'id': access[0].id}) for access in db_role_access],'total':total_count}
  • 功能:根据用户角色获取有访问权限的流程列表,并支持分页和名称过滤。
  • 参数
    • role_id:用户的角色 ID。
    • name:流程名称的搜索关键字。
    • page_num:分页的页码。
    • page_size:每页显示的记录数。
  • 逻辑
    1. 构建过滤条件
      • 如果提供了 name,则添加一个 SQL LIKE 过滤条件,用于模糊匹配流程名称。
    2. 查询有访问权限的流程
      • 调用 FlowDao.get_flow_by_access,传入 role_idnamepage_numpage_size,获取用户有权限访问的流程列表。
    3. 查询总数
      • 调用 FlowDao.get_count_by_filters,传入过滤条件,获取满足条件的流程总数。
    4. 补充用户名
      • 从流程访问列表中提取所有的 user_id
      • 调用 UserDao.get_user_by_ids 获取对应的用户信息。
      • 构建一个字典 user_dict,将 user_id 映射到 user_name
    5. 构建响应数据
      • 遍历 db_role_access,为每个流程构建一个包含 nameuser_nameuser_idupdate_timeid 的字典,并通过 FlowRead.validate 进行数据验证和序列化。
      • 返回包含 datatotal 的字典。
  • 应用场景
    • 用于前端页面展示用户有权限访问的流程列表,并支持搜索和分页功能。

函数 get_assistant_list_by_access

def get_assistant_list_by_access(role_id: int, name: str, page_num: int, page_size: int):count_filter = []if name:count_filter.append(Assistant.name.like('%{}%'.format(name)))db_role_access = AssistantDao.get_assistants_by_access(role_id, name, page_size, page_num)total_count = AssistantDao.get_count_by_filters(count_filter)# 补充用户名user_ids = [access[0].user_id for access in db_role_access]db_users = UserDao.get_user_by_ids(user_ids)user_dict = {user.user_id: user.user_name for user in db_users}return {'data': [{'name': access[0].name,'user_name': user_dict.get(access[0].user_id),'user_id': access[0].user_id,'update_time': access[0].update_time,'id': access[0].id} for access in db_role_access],'total':total_count}
  • 功能:根据用户角色获取有访问权限的助手列表,并支持分页和名称过滤。
  • 参数
    • role_id:用户的角色 ID。
    • name:助手名称的搜索关键字。
    • page_num:分页的页码。
    • page_size:每页显示的记录数。
  • 逻辑
    1. 构建过滤条件
      • 如果提供了 name,则添加一个 SQL LIKE 过滤条件,用于模糊匹配助手名称。
    2. 查询有访问权限的助手
      • 调用 AssistantDao.get_assistants_by_access,传入 role_idnamepage_sizepage_num,获取用户有权限访问的助手列表。
    3. 查询总数
      • 调用 AssistantDao.get_count_by_filters,传入过滤条件,获取满足条件的助手总数。
    4. 补充用户名
      • 从助手访问列表中提取所有的 user_id
      • 调用 UserDao.get_user_by_ids 获取对应的用户信息。
      • 构建一个字典 user_dict,将 user_id 映射到 user_name
    5. 构建响应数据
      • 遍历 db_role_access,为每个助手构建一个包含 nameuser_nameuser_idupdate_timeid 的字典。
      • 返回包含 datatotal 的字典。
  • 应用场景
    • 用于前端页面展示用户有权限访问的助手列表,并支持搜索和分页功能。

7. 认证和权限获取

函数 get_login_user

async def get_login_user(authorize: AuthJWT = Depends()) -> UserPayload:"""获取当前登录的用户"""# 校验是否过期,过期则直接返回http 状态码的 401authorize.jwt_required()current_user = json.loads(authorize.get_jwt_subject())user = UserPayload(**current_user)# 判断是否允许多点登录if not settings.get_system_login_method().allow_multi_login:# 获取access_tokencurrent_token = redis_client.get(USER_CURRENT_SESSION.format(user.user_id))# 登录被挤下线了,http状态码是200, status_code是特殊codeif current_token != authorize._token:raise UserLoginOfflineError.http_exception()return user
  • 功能:获取当前登录的用户信息,并进行多点登录状态检查。
  • 参数
    • authorizeAuthJWT 实例,通过 FastAPI 的 Depends 注入。
  • 逻辑
    1. JWT 验证
      • 调用 authorize.jwt_required(),确保请求包含有效的 JWT 令牌。
      • 如果 JWT 过期或无效,自动抛出 401 Unauthorized 异常。
    2. 获取 JWT 载荷
      • 调用 authorize.get_jwt_subject() 获取 JWT 的主题部分(通常是用户信息的 JSON 字符串)。
      • 使用 json.loads 将 JSON 字符串转换为 Python 字典。
      • 使用 UserPayload 类初始化用户信息对象。
    3. 检查多点登录
      • 调用 settings.get_system_login_method().allow_multi_login 判断系统是否允许多点登录。
      • 如果不允许多点登录:
        • 从 Redis 获取当前用户的 access_token,键格式为 USER_CURRENT_SESSION.format(user.user_id)
        • 比较 Redis 中存储的 current_token 与当前请求中的 _token
        • 如果两者不匹配,说明用户在其他地方登录,抛出 UserLoginOfflineError 异常。
    4. 返回用户信息
      • 返回 UserPayload 实例,包含用户的 ID、角色和用户名。
  • 应用场景
    • 用于保护需要登录才能访问的 API 路由,确保只有认证用户可以访问。
    • 结合 FastAPI 的依赖注入机制,简化路由函数中的用户信息获取。

函数 get_admin_user

async def get_admin_user(authorize: AuthJWT = Depends()) -> UserPayload:"""获取超级管理账号,非超级管理员用户,抛出异常"""login_user = await get_login_user(authorize)if not login_user.is_admin():raise UnAuthorizedError.http_exception()return login_user
  • 功能:获取当前登录的超级管理员用户,如果用户不是管理员,则抛出未授权异常。
  • 参数
    • authorizeAuthJWT 实例,通过 FastAPI 的 Depends 注入。
  • 逻辑
    1. 获取当前登录用户
      • 调用 get_login_user 函数,获取当前登录用户的 UserPayload 实例。
    2. 检查管理员权限
      • 调用 login_user.is_admin() 判断用户是否是管理员。
      • 如果用户不是管理员,抛出 UnAuthorizedError 异常。
    3. 返回管理员用户信息
      • 如果用户是管理员,返回 UserPayload 实例。
  • 应用场景
    • 用于保护需要管理员权限才能访问的 API 路由,确保只有管理员用户可以访问。
    • 结合 FastAPI 的依赖注入机制,简化路由函数中的管理员用户信息获取和权限校验。

8. 其他辅助函数

函数 gen_user_role

此函数已在前面详细讲解,此处不再重复。

函数 gen_user_jwt

此函数已在前面详细讲解,此处不再重复。

函数 get_knowledge_list_by_access

此函数已在前面详细讲解,此处不再重复。

函数 get_flow_list_by_access

此函数已在前面详细讲解,此处不再重复。

函数 get_assistant_list_by_access

此函数已在前面详细讲解,此处不再重复。

9. 总结

这段代码主要涵盖了以下几个关键方面:

  1. 用户身份验证
    • 使用 JWT 进行用户认证,确保每个请求都携带有效的令牌。
    • 提供了 get_login_userget_admin_user 函数,用于获取当前登录用户的信息,并进行权限校验。
  2. 权限管理
    • UserPayload 类封装了用户的角色和权限信息,提供了权限检查的方法。
    • 通过装饰器 wrapper_access_check 简化权限检查逻辑,提高代码复用性。
  3. 用户管理
    • UserService 类提供了用户创建和密码解密的功能。
    • 处理用户注册时的逻辑,如检查用户名是否存在、分配用户组和角色等。
  4. 与数据库交互
    • 使用 DAO(数据访问对象)模式,通过 UserDaoFlowDaoKnowledgeDao 等类与数据库进行交互。
    • 通过 UserRoleDaoRoleAccessDao 等类管理用户角色和权限。
  5. 缓存和配置管理
    • 使用 Redis 作为缓存,存储和获取用户会话信息和 RSA 密钥。
    • 通过 settings 对象管理项目的配置参数,如数据库连接字符串、是否允许多点登录等。
  6. 异常处理
    • 定义了自定义的异常类,用于统一错误响应和处理。
    • 在关键业务逻辑中抛出自定义异常,确保错误信息的一致性和可读性。
  7. 实用功能
    • 提供了 generate_uuid 函数,用于生成唯一的标识符。
    • 提供了 md5_hash 和 RSA 解密功能,增强密码的安全性。

如何在项目中使用

  • 保护路由

    • 使用 Depends(get_login_user)Depends(get_admin_user) 作为路由的依赖项,确保只有经过认证的用户或管理员可以访问特定的路由。
    from fastapi import APIRouter, Dependsrouter = APIRouter()@router.get("/protected-route")
    async def protected_route(user: UserPayload = Depends(get_login_user)):return {"message": f"Hello, {user.user_name}!"}@router.get("/admin-route")
    async def admin_route(user: UserPayload = Depends(get_admin_user)):return {"message": f"Welcome, admin {user.user_name}!"}
    
  • 创建用户

    • 使用 UserService.create_user 方法处理用户创建逻辑。
    @router.post("/create-user")
    async def create_user(request: Request, req_data: CreateUserReq, user: UserPayload = Depends(get_admin_user)):new_user = UserService.create_user(request, user, req_data)return {"user_id": new_user.user_id, "user_name": new_user.user_name}
    
  • 生成和返回 JWT

    • 在用户登录成功后,调用 gen_user_jwt 函数生成 JWT 令牌,并将其返回给前端。
    @router.post("/login")
    async def login(credentials: LoginSchema):db_user = UserDao.authenticate(credentials.username, credentials.password)if not db_user:raise HTTPException(status_code=401, detail="Invalid credentials")access_token, refresh_token, role, web_menu = gen_user_jwt(db_user)return {"access_token": access_token, "refresh_token": refresh_token, "role": role, "web_menu": web_menu}
    

进一步阅读

  • FastAPI 文档
    • FastAPI 官方文档 提供了详细的框架使用指南和最佳实践。
  • Pydantic 文档
    • Pydantic 官方文档 详细介绍了数据模型和验证功能。
  • JWT 认证
    • 学习 JWT(JSON Web Token) 的基本概念和使用方法,了解如何在 API 中实现安全的认证机制。
  • Redis 缓存
    • Redis 官方文档 了解如何使用 Redis 进行缓存和会话管理。
  • RSA 加密
    • 学习 RSA 加密算法 的基本原理和应用场景,了解如何在 Python 中使用 rsa 库进行加密和解密操作。

常见问题

  1. 为什么使用装饰器进行权限检查?
    • 使用装饰器可以将权限检查逻辑与业务逻辑分离,提高代码的可读性和可维护性。通过装饰器,可以复用权限检查逻辑,避免在每个方法中重复编写相同的代码。
  2. 如何确保 JWT 的安全性?
    • 确保 JWT 使用强密码进行签名,避免暴露私钥。
    • 设置合理的过期时间,防止令牌被滥用。
    • 使用 HTTPS 保护令牌在传输过程中的安全性。
  3. 为什么需要检查多点登录?
    • 检查多点登录可以防止用户账号被多人同时使用,提升账号的安全性。
    • 如果系统不允许多点登录,当用户在其他设备登录时,可以自动将之前的会话踢下线。
  4. 如何处理 RSA 密钥管理?
    • 确保 RSA 密钥安全存储,不要将私钥暴露在代码库中。
    • 使用环境变量或安全的密钥管理服务(如 AWS KMS)来存储和管理 RSA 密钥。
  5. 如何优化数据库查询性能?
    • 确保数据库表中有适当的索引,尤其是在常用的查询字段上。
    • 使用分页查询减少一次查询返回的数据量。
    • 避免 N+1 查询问题,通过使用 JOIN 或预加载相关数据提高查询效率。

相关文章:

毕昇入门学习

schemas.py 概述 这段代码主要定义了一系列基于 Pydantic 的数据模型(BaseModel),用于数据验证和序列化,通常用于构建 API(如使用 FastAPI)。这些模型涵盖了用户认证、聊天消息、知识库管理、模型配置等多…...

2411C++,学习C++提示4

结构绑定 auto [first, ...ts] std::tuple{1, 2 ,3};assert(1 first);浮点作为非类型模板参数 template<double Value> constexpr auto value Value;int main() {std::cout << value<4.2>; // prints 4.2 }template<double... Vl1s, double... Vl2s&g…...

STM32-- 看门狗--介绍、使用场景、失效场景

STM32 中的看门狗&#xff08;Watchdog Timer&#xff0c;简称 WDG&#xff09;有两种主要类型&#xff1a;独立看门狗&#xff08;IWDG&#xff09; 和 窗口看门狗&#xff08;WWDG&#xff09;。它们的喂狗机制各有特点&#xff0c;主要区别如下&#xff1a; 1. 独立看门狗&a…...

【赵渝强老师】PostgreSQL的数据库

PostgreSQL的逻辑存储结构主要是指数据库中的各种数据库对象&#xff0c;包括&#xff1a;数据库集群、数据库、表、索引、视图等等。所有数据库对象都有各自的对象标识符oid&#xff08;object identifiers&#xff09;,它是一个无符号的四字节整数&#xff0c;相关对象的oid都…...

linux安全管理-会话安全

文章目录 1 设置命令行界面超时退出2 配置终端登录失败策略3 配置 SSH 登录失败策略 1 设置命令行界面超时退出 1、检查内容 检查操作系统是否设置命令行界面超时退出。 2、配置要求 操作系统设置命令行界面超时退出。 3、配置方法 配置命令行界面超时时间&#xff0c;编辑/et…...

Ubuntu监视显卡占用情况

在终端中运行 watch -n 0.5 nvidia-smi【以下内容由大模型生成】 watch -n 0.5 nvidia-smi 是一个组合命令&#xff0c;用于在 Linux 终端中定期执行 nvidia-smi 命令并显示其输出。让我们分解一下这个命令的各个部分&#xff1a; watch: watch 是一个用于定期执行其他命令并显…...

学成在线day06

上传视屏 断点续传 通常视频文件都比较大&#xff0c;所以对于媒资系统上传文件的需求要满足大文件的上传要求。http协议本身对上传文件大小没有限制&#xff0c;但是客户的网络环境质量、电脑硬件环境等参差不齐&#xff0c;如果一个大文件快上传完了网断了没有上传完成&…...

Mac安装及合规无限使用Beyond Compare

文章目录 Beyond CompareBeyond Compare简介Beyond Compare安装Beyond Compare到期后继续免费使用 Beyond Compare Beyond Compare简介 Beyond Compare 是一款由 Scooter Software 开发的文件和文件夹比较工具。它主要用于对比两个文件或文件夹之间的差异&#xff0c;并支持文…...

【青牛科技】2K02 电动工具专用调速电路芯片描述

概述&#xff1a; 2K02 是电动工具专用调速电路。内置稳压电路&#xff0c;温度系数好&#xff0c;可以调节输出频率以及占空比的振荡输出&#xff0c;广泛的应用于小型电钻&#xff0c;割草机等工具。 主要特点&#xff1a; ● 电源电压范围宽 ● 占空比可调 ● 温度系数好 …...

基于SpringBoot实现的民宿管理系统(代码+论文)

&#x1f389;博主介绍&#xff1a;Java领域优质创作者&#xff0c;阿里云博客专家&#xff0c;计算机毕设实战导师。专注Java项目实战、毕设定制/协助 &#x1f4e2;主要服务内容&#xff1a;选题定题、开题报告、任务书、程序开发、项目定制、论文辅导 &#x1f496;精彩专栏…...

安装QT6.8(MSVC MinGW)+QT webengine+QT5.15.2

本篇主要针对只使用过QT5的qmake&#xff0c;没有用过MSVC&#xff0c;VS的老同学。 建议一部分一部分安装&#xff0c;全部勾选安装遇到问题会中断&#xff0c;前功尽弃。 我自己需要的是QT5&#xff0c;编出的软件用在公司设备上。 QT6&#xff1a;建议也安装学习&#xf…...

MinIO常见操作及Python实现对象的增删改查

MinIO常见操作 MinIO是一个高性能的开源对象存储服务&#xff0c;它兼容Amazon S3云存储服务API。在MinIO中&#xff0c;常见的操作包括&#xff1a; 存储桶操作&#xff1a; 创建、列出、获取信息、删除存储桶。 对象操作&#xff1a; 上传、下载、列出、删除对象。 权限管理&…...

网络编程中的字节序函数htonl()、htons()、ntohl()和ntohs()

目录 1&#xff0c;网络字节序和主机字节序 2. 函数的具体作用 2.1,htonl&#xff08;Host to Network Long&#xff09; 2.2,htons&#xff08;Host to Network Short) 2.3,ntohl&#xff08;Network to Host Long&#xff09; 2.4,ntohs&#xff08;Network to Host Sho…...

【dvwa靶场:File Upload系列】File Upload低-中-高级别,通关啦

目录 一、low级别,直接上传木马文件 1.1、准备一个木马文件 1.2、直接上传木马文件 1.3、访问木马链接 1.4、连接蚁剑 二、medium级别&#xff1a;抓包文件缀名 2.1、准备一个木马文件&#xff0c;修改后缀名为图片的后缀名 2.2、上传文件&#xff0c;打开burpSuite&…...

RHCE NFS

RHCE NFS 1.11. 2 NFS 主机名格式1.3 NFS 服务器配置1.3.1 /etc/exports 配置文件1.3.1.1 导出条目1.3.1.2 默认选项1.3.1.3 默认和覆盖选项 1.4 启动 NFS 服务器1.5 练习1.5.1 配置 NFS 服务器和客户端挂载1.5.2 配置autofs自动挂载&#xff08;需要时才挂载&#xff09; 1.6 …...

【数据结构】ArrayList与顺序表

ArrayList与顺序表 1.线性表2.顺序表2.1 接口的实现 3. ArrayList简介4. ArrayList使用4.2 ArrayList常见操作4.3 ArrayList的遍历4.4 ArrayList的扩容机制 5. ArrayList的具体使用5.1 杨辉三角5.2 简单的洗牌算法 6. ArrayList的问题及思考 【本节目标】 线性表顺序表ArrayLis…...

互联网基础

TCP/IP协议&#xff08;协议组&#xff09; 分层名称TCP/IP协议应用层HTTP,FTP,mDNS,WebSocket,OSC...传输层TCP&#xff0c;UDP网络层IP链路层&#xff08;网络接口层&#xff09;Ethernet&#xff0c;Wi-Fi... 链路层&#xff08;网络接口层&#xff09; 链路层的主要作用…...

ffmpeg.js视频播放(转换)

chrome 临时设置SharedArrayBuffer "C:\Program Files\Google\Chrome\Application\chrome.exe" --enable-featuresSharedArrayBuffer 引用的js及相关文件 ffmpeg.min.js ffmpeg.min.js.map ffmpeg-core.js ffmpeg-core.wasm ffmpeg-core.worker.js 以上几个现…...

后端 Java发送邮件 JavaMail 模版 20241128测试可用

配置授权码 依赖 <dependency><groupId>javax.mail</groupId><artifactId>javax.mail-api</artifactId><version>1.5.5</version> </dependency> <dependency><groupId>com.sun.mail</groupId><artifa…...

电脑中的vcruntime140_1.dll文件有问题要怎么解决?一键修复vcruntime140_1.dll

遇到“vcruntime140_1.dll无法继续执行代码”的错误通常表明电脑中的vcruntime140_1.dll文件有问题。这个文件属于Visual C Redistributable&#xff0c;对很多程序的运行至关重要。本文将提供几个步骤&#xff0c;帮助你迅速修复这一错误&#xff0c;使电脑恢复正常工作状态。…...

探索 Vue 3.0中Treeshaking特性?

Vue 3.0 中的 Tree Shaking 特性 Tree Shaking 是一种优化技术,旨在通过静态分析代码,去除未使用的模块或函数,从而减小最终的打包文件大小。在 Vue 3.0 中,由于其模块化设计和代码按需引入的特性,Vue 的 Tree Shaking 特性得到了增强,能够有效地去除那些在生产环境中未…...

Paddle Inference部署推理(十)

十&#xff1a;Paddle Inference推理 &#xff08;python&#xff09;API详解 9. 启用内存优化 API定义如下&#xff1a; # 开启内存 / 显存复用&#xff0c;具体降低内存效果取决于模型结构 # 参数&#xff1a;None # 返回&#xff1a;None paddle.inference.Config.enable…...

万能门店小程序管理系统 doPageGetFormList SQL注入漏洞复现

0x01 产品简介 万能门店小程序管理系统是一款功能强大的工具,旨在为各行业商家提供线上线下融合的全方位解决方案。是一个集成了会员管理和会员营销两大核心功能的综合性平台。它支持多行业使用,通过后台一键切换版本,满足不同行业商家的个性化需求。该系统采用轻量后台,搭…...

全面+彻底解决VMware安装后没有VMnet1和VMnet8的问题

目录 1、摘要 &#xff08;1&#xff09;问题 &#xff08;2&#xff09;所用工具 ① Everything软件 ② CCleaner软件 2、问题的检查与确认 3、解决过程 &#xff08;1&#xff09;卸载已经安装的VMware &#xff08;2&#xff09;设置services.mcs&#xff1a;服务自…...

什么是堆?

堆&#xff08;Heap&#xff09;&#xff1a;堆可以看做是一颗用数组实现的二叉树&#xff0c;所以它没有使用父指针或者子指针。堆根据“堆属性”来排序&#xff0c;“堆属性”决定了树中节点的位置。 堆的特性 1.堆是完全二叉树&#xff0c;除了树的最后一层节点不需要是满的…...

微距动物和植物摄影后期森系风格Lr调色教程,手机滤镜PS+Lightroom预设下载!

调色教程 微距动物和植物摄影后期采用森系风格的 Lightroom 调色&#xff0c;将微距下的动植物世界打造成充满自然气息和梦幻感的画面。这种调色风格旨在突出动植物的细腻之美&#xff0c;同时营造出宁静、清新的森林氛围。 预设信息 调色风格&#xff1a;森系风格预设适合类…...

Qt6.8安卓Android开发环境配置

时隔多年&#xff0c;重拾QtCreator下Android开发。发现Qt6下安卓开发环境配置变简单不少&#xff01;只需三步即可在QtCreator下进行Android开发&#xff1a; 一、使用Qt Mantenance Tool进行Android模块的安装&#xff1a; 如果感觉安装网速较慢&#xff0c;可以查看本人另外…...

RK3568部署yolo8记录

本教程记录自己一下在RK3568上部署yolo8的步骤 板端驱动 在板端&#xff0c;首先查看rknpu驱动是否安装、存在。若键入下面的命令有返回则&#xff0c;证明驱动已安装。 dmesg | grep -i rknpu 瑞芯微官方说&#xff0c;驱动版本最好大于0.9.2。但是我看有的博主说&#xff…...

数据可视化复习2-绘制折线图+条形图(叠加条形图,并列条形图,水平条形图)+ 饼状图 + 直方图

目录 目录 一、绘制折线图 1.使用pyplot 2.使用numpy ​编辑 3.使用DataFrame ​编辑 二、绘制条形图&#xff08;柱状图&#xff09; 1.简单条形图 2.绘制叠加条形图 3.绘制并列条形图 4.水平条形图 ​编辑 三、绘制饼状图 四、绘制散点图和直方图 1.散点图 2…...

JavaScript原生深拷贝方法 structuredClone使用

structuredClone 简介 structuredClone 是现代浏览器提供的原生 JavaScript 方法&#xff0c;用于深拷贝对象。它可以处理各种复杂数据结构&#xff0c;包括嵌套对象、数组、Date、Map、Set 等&#xff0c;且支持循环引用。 语法 const clone structuredClone(value);value:…...

建设部网站1667号下载/网络营销推广外包平台

京东2021校招&#xff0c;投了两个岗位&#xff0c;一个是物流方向管培生&#xff0c;一个是物流规划&#xff0c;管培生当时未参加一面(形式群面)&#xff0c;后来面了物流规划这个。1、初试(通过)一面为一对一专业面试&#xff0c;形式为线上视频面试&#xff0c;问了一些实习…...

查询站长工具会给网站带来外链这样好吗/线上营销推广方案

今天来分享一下如何将PCB的3D图到导出来&#xff0c;我们除了采用截图工具之外&#xff0c;AD软件自带一个3D实时快照功能&#xff0c;可以自动生成图片。 只要我们在3D模式下按下“crtl c”后&#xff0c;AD就会弹出如图所示提示&#xff1a; 该窗口是提示我们选择多大的分…...

在线设计平台行业的发展趋势/海外seo培训

对于刚进入数据分析行业新手来说&#xff0c;EXCEL可以被当做一款入门的软件。在学习R或Python前&#xff0c;事先掌握一定的EXCEL知识是百利而无一害。EXCEL凭借其功能强大的函数、可视化图表、以及整齐排列的电子表格功能&#xff0c;使你能够快速而深入的洞察到数据不轻易为…...

网站被做301跳转了怎么办/百度 站长工具

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼在C语言函数库中包含了一个产生随机数的函数&#xff1a;int rand( void );在函数库中对这个函数的说明是&#xff1a;The rand function returns a pseudorandom integer in the range0 to RAND_MAX. Use the srand function to s…...

怎么做模板网站的报价表/外贸网站哪个比较好

引用参考 【分区表】 --对oracle分区表的理解整理 [url]http://blog.itpub.net/30130773/viewspace-2121413/[/url]【导入dmp文件到本地数据库】 --本地安装了oracle 11g&#xff0c;需要导入E:\oracle\bakDB\demo.dmp到本地用户tester的表空间中 [b]方法一&#xff1a;plsql导…...

什么网站做前端练手好/四川网站推广公司

这篇文章的开始先给大家看一个图片 image用过或者看过springboot的人都知道&#xff0c;这就是springboot启动的banner&#xff0c;这一篇介绍如何自定义springboot的启动bannner。 先介绍一个可以制作自定义banner的网站&#xff0c;传送门&#xff1a;http://patorjk.com/sof…...