- 新增图像生成接口,支持试用、积分和自定义API Key模式 - 实现生成图片结果异步上传至MinIO存储,带重试机制 - 优化积分预扣除和异常退还逻辑,保障用户积分准确 - 添加获取生成历史记录接口,支持时间范围和分页 - 提供本地字典配置接口,支持模型、比例、提示模板和尺寸 - 实现图片批量上传接口,支持S3兼容对象存储 feat(admin): 增加管理员角色管理与权限分配接口 - 实现角色列表查询、角色创建、更新及删除功能 - 增加权限列表查询接口 - 实现用户角色分配接口,便于统一管理用户权限 - 增加系统字典增删查改接口,支持分类过滤和排序 - 权限控制全面覆盖管理接口,保证安全访问 feat(auth): 完善用户登录注册及权限相关接口与页面 - 实现手机号验证码发送及校验功能,保障注册安全 - 支持手机号注册、登录及退出接口,集成日志记录 - 增加修改密码功能,验证原密码后更新 - 提供动态导航菜单接口,基于权限展示不同菜单 - 实现管理界面路由及日志、角色、字典管理页面访问权限控制 - 添加系统日志查询接口,支持关键词和等级筛选 feat(app): 初始化Flask应用并配置蓝图与数据库 - 创建应用程序工厂,加载配置,初始化数据库和Redis客户端 - 注册认证、API及管理员蓝图,整合路由 - 根路由渲染主页模板 - 应用上下文中自动创建数据库表,保证运行环境准备完毕 feat(database): 提供数据库创建与迁移支持脚本 - 新增数据库创建脚本,支持自动检测是否已存在 - 添加数据库表初始化脚本,支持创建和删除所有表 - 实现RBAC权限初始化,包含基础权限和角色创建 - 新增字段手动修复脚本,添加用户API Key和积分字段 - 强制迁移脚本支持清理连接和修复表结构,初始化默认数据及角色分配 feat(config): 新增系统配置参数 - 配置数据库、Redis、Session和MinIO相关参数 - 添加AI接口地址及试用Key配置 - 集成阿里云短信服务配置及开发模式相关参数 feat(extensions): 初始化数据库、Redis和MinIO客户端 - 创建全局SQLAlchemy数据库实例和Redis客户端 - 配置基于boto3的MinIO兼容S3客户端 chore(logs): 添加示例系统日志文件 - 记录用户请求、验证码发送成功与失败的日志信息
238 lines
7.5 KiB
Python
238 lines
7.5 KiB
Python
# orm/exc.py
|
|
# Copyright (C) 2005-2025 the SQLAlchemy authors and contributors
|
|
# <see AUTHORS file>
|
|
#
|
|
# This module is part of SQLAlchemy and is released under
|
|
# the MIT License: https://www.opensource.org/licenses/mit-license.php
|
|
|
|
"""SQLAlchemy ORM exceptions."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Any
|
|
from typing import Optional
|
|
from typing import Tuple
|
|
from typing import Type
|
|
from typing import TYPE_CHECKING
|
|
from typing import TypeVar
|
|
|
|
from .util import _mapper_property_as_plain_name
|
|
from .. import exc as sa_exc
|
|
from .. import util
|
|
from ..exc import MultipleResultsFound # noqa
|
|
from ..exc import NoResultFound # noqa
|
|
|
|
if TYPE_CHECKING:
|
|
from .interfaces import LoaderStrategy
|
|
from .interfaces import MapperProperty
|
|
from .state import InstanceState
|
|
|
|
_T = TypeVar("_T", bound=Any)
|
|
|
|
NO_STATE = (AttributeError, KeyError)
|
|
"""Exception types that may be raised by instrumentation implementations."""
|
|
|
|
|
|
class StaleDataError(sa_exc.SQLAlchemyError):
|
|
"""An operation encountered database state that is unaccounted for.
|
|
|
|
Conditions which cause this to happen include:
|
|
|
|
* A flush may have attempted to update or delete rows
|
|
and an unexpected number of rows were matched during
|
|
the UPDATE or DELETE statement. Note that when
|
|
version_id_col is used, rows in UPDATE or DELETE statements
|
|
are also matched against the current known version
|
|
identifier.
|
|
|
|
* A mapped object with version_id_col was refreshed,
|
|
and the version number coming back from the database does
|
|
not match that of the object itself.
|
|
|
|
* A object is detached from its parent object, however
|
|
the object was previously attached to a different parent
|
|
identity which was garbage collected, and a decision
|
|
cannot be made if the new parent was really the most
|
|
recent "parent".
|
|
|
|
"""
|
|
|
|
|
|
ConcurrentModificationError = StaleDataError
|
|
|
|
|
|
class FlushError(sa_exc.SQLAlchemyError):
|
|
"""A invalid condition was detected during flush()."""
|
|
|
|
|
|
class MappedAnnotationError(sa_exc.ArgumentError):
|
|
"""Raised when ORM annotated declarative cannot interpret the
|
|
expression present inside of the :class:`.Mapped` construct.
|
|
|
|
.. versionadded:: 2.0.40
|
|
|
|
"""
|
|
|
|
|
|
class UnmappedError(sa_exc.InvalidRequestError):
|
|
"""Base for exceptions that involve expected mappings not present."""
|
|
|
|
|
|
class ObjectDereferencedError(sa_exc.SQLAlchemyError):
|
|
"""An operation cannot complete due to an object being garbage
|
|
collected.
|
|
|
|
"""
|
|
|
|
|
|
class DetachedInstanceError(sa_exc.SQLAlchemyError):
|
|
"""An attempt to access unloaded attributes on a
|
|
mapped instance that is detached."""
|
|
|
|
code = "bhk3"
|
|
|
|
|
|
class UnmappedInstanceError(UnmappedError):
|
|
"""An mapping operation was requested for an unknown instance."""
|
|
|
|
@util.preload_module("sqlalchemy.orm.base")
|
|
def __init__(self, obj: object, msg: Optional[str] = None):
|
|
base = util.preloaded.orm_base
|
|
|
|
if not msg:
|
|
try:
|
|
base.class_mapper(type(obj))
|
|
name = _safe_cls_name(type(obj))
|
|
msg = (
|
|
"Class %r is mapped, but this instance lacks "
|
|
"instrumentation. This occurs when the instance "
|
|
"is created before sqlalchemy.orm.mapper(%s) "
|
|
"was called." % (name, name)
|
|
)
|
|
except UnmappedClassError:
|
|
msg = f"Class '{_safe_cls_name(type(obj))}' is not mapped"
|
|
if isinstance(obj, type):
|
|
msg += (
|
|
"; was a class (%s) supplied where an instance was "
|
|
"required?" % _safe_cls_name(obj)
|
|
)
|
|
UnmappedError.__init__(self, msg)
|
|
|
|
def __reduce__(self) -> Any:
|
|
return self.__class__, (None, self.args[0])
|
|
|
|
|
|
class UnmappedClassError(UnmappedError):
|
|
"""An mapping operation was requested for an unknown class."""
|
|
|
|
def __init__(self, cls: Type[_T], msg: Optional[str] = None):
|
|
if not msg:
|
|
msg = _default_unmapped(cls)
|
|
UnmappedError.__init__(self, msg)
|
|
|
|
def __reduce__(self) -> Any:
|
|
return self.__class__, (None, self.args[0])
|
|
|
|
|
|
class ObjectDeletedError(sa_exc.InvalidRequestError):
|
|
"""A refresh operation failed to retrieve the database
|
|
row corresponding to an object's known primary key identity.
|
|
|
|
A refresh operation proceeds when an expired attribute is
|
|
accessed on an object, or when :meth:`_query.Query.get` is
|
|
used to retrieve an object which is, upon retrieval, detected
|
|
as expired. A SELECT is emitted for the target row
|
|
based on primary key; if no row is returned, this
|
|
exception is raised.
|
|
|
|
The true meaning of this exception is simply that
|
|
no row exists for the primary key identifier associated
|
|
with a persistent object. The row may have been
|
|
deleted, or in some cases the primary key updated
|
|
to a new value, outside of the ORM's management of the target
|
|
object.
|
|
|
|
"""
|
|
|
|
@util.preload_module("sqlalchemy.orm.base")
|
|
def __init__(self, state: InstanceState[Any], msg: Optional[str] = None):
|
|
base = util.preloaded.orm_base
|
|
|
|
if not msg:
|
|
msg = (
|
|
"Instance '%s' has been deleted, or its "
|
|
"row is otherwise not present." % base.state_str(state)
|
|
)
|
|
|
|
sa_exc.InvalidRequestError.__init__(self, msg)
|
|
|
|
def __reduce__(self) -> Any:
|
|
return self.__class__, (None, self.args[0])
|
|
|
|
|
|
class UnmappedColumnError(sa_exc.InvalidRequestError):
|
|
"""Mapping operation was requested on an unknown column."""
|
|
|
|
|
|
class LoaderStrategyException(sa_exc.InvalidRequestError):
|
|
"""A loader strategy for an attribute does not exist."""
|
|
|
|
def __init__(
|
|
self,
|
|
applied_to_property_type: Type[Any],
|
|
requesting_property: MapperProperty[Any],
|
|
applies_to: Optional[Type[MapperProperty[Any]]],
|
|
actual_strategy_type: Optional[Type[LoaderStrategy]],
|
|
strategy_key: Tuple[Any, ...],
|
|
):
|
|
if actual_strategy_type is None:
|
|
sa_exc.InvalidRequestError.__init__(
|
|
self,
|
|
"Can't find strategy %s for %s"
|
|
% (strategy_key, requesting_property),
|
|
)
|
|
else:
|
|
assert applies_to is not None
|
|
sa_exc.InvalidRequestError.__init__(
|
|
self,
|
|
'Can\'t apply "%s" strategy to property "%s", '
|
|
'which is a "%s"; this loader strategy is intended '
|
|
'to be used with a "%s".'
|
|
% (
|
|
util.clsname_as_plain_name(actual_strategy_type),
|
|
requesting_property,
|
|
_mapper_property_as_plain_name(applied_to_property_type),
|
|
_mapper_property_as_plain_name(applies_to),
|
|
),
|
|
)
|
|
|
|
|
|
def _safe_cls_name(cls: Type[Any]) -> str:
|
|
cls_name: Optional[str]
|
|
try:
|
|
cls_name = ".".join((cls.__module__, cls.__name__))
|
|
except AttributeError:
|
|
cls_name = getattr(cls, "__name__", None)
|
|
if cls_name is None:
|
|
cls_name = repr(cls)
|
|
return cls_name
|
|
|
|
|
|
@util.preload_module("sqlalchemy.orm.base")
|
|
def _default_unmapped(cls: Type[Any]) -> Optional[str]:
|
|
base = util.preloaded.orm_base
|
|
|
|
try:
|
|
mappers = base.manager_of_class(cls).mappers # type: ignore
|
|
except (
|
|
UnmappedClassError,
|
|
TypeError,
|
|
) + NO_STATE:
|
|
mappers = {}
|
|
name = _safe_cls_name(cls)
|
|
|
|
if not mappers:
|
|
return f"Class '{name}' is not mapped"
|
|
else:
|
|
return None
|