- 新增图像生成接口,支持试用、积分和自定义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): 添加示例系统日志文件 - 记录用户请求、验证码发送成功与失败的日志信息
192 lines
6.8 KiB
Python
192 lines
6.8 KiB
Python
from __future__ import annotations
|
|
|
|
import typing as t
|
|
|
|
from . import typing as ft
|
|
from .globals import current_app
|
|
from .globals import request
|
|
|
|
F = t.TypeVar("F", bound=t.Callable[..., t.Any])
|
|
|
|
http_method_funcs = frozenset(
|
|
["get", "post", "head", "options", "delete", "put", "trace", "patch"]
|
|
)
|
|
|
|
|
|
class View:
|
|
"""Subclass this class and override :meth:`dispatch_request` to
|
|
create a generic class-based view. Call :meth:`as_view` to create a
|
|
view function that creates an instance of the class with the given
|
|
arguments and calls its ``dispatch_request`` method with any URL
|
|
variables.
|
|
|
|
See :doc:`views` for a detailed guide.
|
|
|
|
.. code-block:: python
|
|
|
|
class Hello(View):
|
|
init_every_request = False
|
|
|
|
def dispatch_request(self, name):
|
|
return f"Hello, {name}!"
|
|
|
|
app.add_url_rule(
|
|
"/hello/<name>", view_func=Hello.as_view("hello")
|
|
)
|
|
|
|
Set :attr:`methods` on the class to change what methods the view
|
|
accepts.
|
|
|
|
Set :attr:`decorators` on the class to apply a list of decorators to
|
|
the generated view function. Decorators applied to the class itself
|
|
will not be applied to the generated view function!
|
|
|
|
Set :attr:`init_every_request` to ``False`` for efficiency, unless
|
|
you need to store request-global data on ``self``.
|
|
"""
|
|
|
|
#: The methods this view is registered for. Uses the same default
|
|
#: (``["GET", "HEAD", "OPTIONS"]``) as ``route`` and
|
|
#: ``add_url_rule`` by default.
|
|
methods: t.ClassVar[t.Collection[str] | None] = None
|
|
|
|
#: Control whether the ``OPTIONS`` method is handled automatically.
|
|
#: Uses the same default (``True``) as ``route`` and
|
|
#: ``add_url_rule`` by default.
|
|
provide_automatic_options: t.ClassVar[bool | None] = None
|
|
|
|
#: A list of decorators to apply, in order, to the generated view
|
|
#: function. Remember that ``@decorator`` syntax is applied bottom
|
|
#: to top, so the first decorator in the list would be the bottom
|
|
#: decorator.
|
|
#:
|
|
#: .. versionadded:: 0.8
|
|
decorators: t.ClassVar[list[t.Callable[..., t.Any]]] = []
|
|
|
|
#: Create a new instance of this view class for every request by
|
|
#: default. If a view subclass sets this to ``False``, the same
|
|
#: instance is used for every request.
|
|
#:
|
|
#: A single instance is more efficient, especially if complex setup
|
|
#: is done during init. However, storing data on ``self`` is no
|
|
#: longer safe across requests, and :data:`~flask.g` should be used
|
|
#: instead.
|
|
#:
|
|
#: .. versionadded:: 2.2
|
|
init_every_request: t.ClassVar[bool] = True
|
|
|
|
def dispatch_request(self) -> ft.ResponseReturnValue:
|
|
"""The actual view function behavior. Subclasses must override
|
|
this and return a valid response. Any variables from the URL
|
|
rule are passed as keyword arguments.
|
|
"""
|
|
raise NotImplementedError()
|
|
|
|
@classmethod
|
|
def as_view(
|
|
cls, name: str, *class_args: t.Any, **class_kwargs: t.Any
|
|
) -> ft.RouteCallable:
|
|
"""Convert the class into a view function that can be registered
|
|
for a route.
|
|
|
|
By default, the generated view will create a new instance of the
|
|
view class for every request and call its
|
|
:meth:`dispatch_request` method. If the view class sets
|
|
:attr:`init_every_request` to ``False``, the same instance will
|
|
be used for every request.
|
|
|
|
Except for ``name``, all other arguments passed to this method
|
|
are forwarded to the view class ``__init__`` method.
|
|
|
|
.. versionchanged:: 2.2
|
|
Added the ``init_every_request`` class attribute.
|
|
"""
|
|
if cls.init_every_request:
|
|
|
|
def view(**kwargs: t.Any) -> ft.ResponseReturnValue:
|
|
self = view.view_class( # type: ignore[attr-defined]
|
|
*class_args, **class_kwargs
|
|
)
|
|
return current_app.ensure_sync(self.dispatch_request)(**kwargs) # type: ignore[no-any-return]
|
|
|
|
else:
|
|
self = cls(*class_args, **class_kwargs) # pyright: ignore
|
|
|
|
def view(**kwargs: t.Any) -> ft.ResponseReturnValue:
|
|
return current_app.ensure_sync(self.dispatch_request)(**kwargs) # type: ignore[no-any-return]
|
|
|
|
if cls.decorators:
|
|
view.__name__ = name
|
|
view.__module__ = cls.__module__
|
|
for decorator in cls.decorators:
|
|
view = decorator(view)
|
|
|
|
# We attach the view class to the view function for two reasons:
|
|
# first of all it allows us to easily figure out what class-based
|
|
# view this thing came from, secondly it's also used for instantiating
|
|
# the view class so you can actually replace it with something else
|
|
# for testing purposes and debugging.
|
|
view.view_class = cls # type: ignore
|
|
view.__name__ = name
|
|
view.__doc__ = cls.__doc__
|
|
view.__module__ = cls.__module__
|
|
view.methods = cls.methods # type: ignore
|
|
view.provide_automatic_options = cls.provide_automatic_options # type: ignore
|
|
return view
|
|
|
|
|
|
class MethodView(View):
|
|
"""Dispatches request methods to the corresponding instance methods.
|
|
For example, if you implement a ``get`` method, it will be used to
|
|
handle ``GET`` requests.
|
|
|
|
This can be useful for defining a REST API.
|
|
|
|
:attr:`methods` is automatically set based on the methods defined on
|
|
the class.
|
|
|
|
See :doc:`views` for a detailed guide.
|
|
|
|
.. code-block:: python
|
|
|
|
class CounterAPI(MethodView):
|
|
def get(self):
|
|
return str(session.get("counter", 0))
|
|
|
|
def post(self):
|
|
session["counter"] = session.get("counter", 0) + 1
|
|
return redirect(url_for("counter"))
|
|
|
|
app.add_url_rule(
|
|
"/counter", view_func=CounterAPI.as_view("counter")
|
|
)
|
|
"""
|
|
|
|
def __init_subclass__(cls, **kwargs: t.Any) -> None:
|
|
super().__init_subclass__(**kwargs)
|
|
|
|
if "methods" not in cls.__dict__:
|
|
methods = set()
|
|
|
|
for base in cls.__bases__:
|
|
if getattr(base, "methods", None):
|
|
methods.update(base.methods) # type: ignore[attr-defined]
|
|
|
|
for key in http_method_funcs:
|
|
if hasattr(cls, key):
|
|
methods.add(key.upper())
|
|
|
|
if methods:
|
|
cls.methods = methods
|
|
|
|
def dispatch_request(self, **kwargs: t.Any) -> ft.ResponseReturnValue:
|
|
meth = getattr(self, request.method.lower(), None)
|
|
|
|
# If the request method is HEAD and we don't have a handler for it
|
|
# retry with GET.
|
|
if meth is None and request.method == "HEAD":
|
|
meth = getattr(self, "get", None)
|
|
|
|
assert meth is not None, f"Unimplemented method {request.method!r}"
|
|
return current_app.ensure_sync(meth)(**kwargs) # type: ignore[no-any-return]
|