- 新增图像生成接口,支持试用、积分和自定义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): 添加示例系统日志文件 - 记录用户请求、验证码发送成功与失败的日志信息
168 lines
6.0 KiB
Python
168 lines
6.0 KiB
Python
# Copyright 2014 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"). You
|
|
# may not use this file except in compliance with the License. A copy of
|
|
# the License is located at
|
|
#
|
|
# https://aws.amazon.com/apache2.0/
|
|
#
|
|
# or in the "license" file accompanying this file. This file is
|
|
# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF
|
|
# ANY KIND, either express or implied. See the License for the specific
|
|
# language governing permissions and limitations under the License.
|
|
|
|
import re
|
|
|
|
import jmespath
|
|
from botocore import xform_name
|
|
|
|
from ..exceptions import ResourceLoadException
|
|
|
|
INDEX_RE = re.compile(r'\[(.*)\]$')
|
|
|
|
|
|
def get_data_member(parent, path):
|
|
"""
|
|
Get a data member from a parent using a JMESPath search query,
|
|
loading the parent if required. If the parent cannot be loaded
|
|
and no data is present then an exception is raised.
|
|
|
|
:type parent: ServiceResource
|
|
:param parent: The resource instance to which contains data we
|
|
are interested in.
|
|
:type path: string
|
|
:param path: The JMESPath expression to query
|
|
:raises ResourceLoadException: When no data is present and the
|
|
resource cannot be loaded.
|
|
:returns: The queried data or ``None``.
|
|
"""
|
|
# Ensure the parent has its data loaded, if possible.
|
|
if parent.meta.data is None:
|
|
if hasattr(parent, 'load'):
|
|
parent.load()
|
|
else:
|
|
raise ResourceLoadException(
|
|
f'{parent.__class__.__name__} has no load method!'
|
|
)
|
|
|
|
return jmespath.search(path, parent.meta.data)
|
|
|
|
|
|
def create_request_parameters(parent, request_model, params=None, index=None):
|
|
"""
|
|
Handle request parameters that can be filled in from identifiers,
|
|
resource data members or constants.
|
|
|
|
By passing ``params``, you can invoke this method multiple times and
|
|
build up a parameter dict over time, which is particularly useful
|
|
for reverse JMESPath expressions that append to lists.
|
|
|
|
:type parent: ServiceResource
|
|
:param parent: The resource instance to which this action is attached.
|
|
:type request_model: :py:class:`~boto3.resources.model.Request`
|
|
:param request_model: The action request model.
|
|
:type params: dict
|
|
:param params: If set, then add to this existing dict. It is both
|
|
edited in-place and returned.
|
|
:type index: int
|
|
:param index: The position of an item within a list
|
|
:rtype: dict
|
|
:return: Pre-filled parameters to be sent to the request operation.
|
|
"""
|
|
if params is None:
|
|
params = {}
|
|
|
|
for param in request_model.params:
|
|
source = param.source
|
|
target = param.target
|
|
|
|
if source == 'identifier':
|
|
# Resource identifier, e.g. queue.url
|
|
value = getattr(parent, xform_name(param.name))
|
|
elif source == 'data':
|
|
# If this is a data member then it may incur a load
|
|
# action before returning the value.
|
|
value = get_data_member(parent, param.path)
|
|
elif source in ['string', 'integer', 'boolean']:
|
|
# These are hard-coded values in the definition
|
|
value = param.value
|
|
elif source == 'input':
|
|
# This is provided by the user, so ignore it here
|
|
continue
|
|
else:
|
|
raise NotImplementedError(f'Unsupported source type: {source}')
|
|
|
|
build_param_structure(params, target, value, index)
|
|
|
|
return params
|
|
|
|
|
|
def build_param_structure(params, target, value, index=None):
|
|
"""
|
|
This method provides a basic reverse JMESPath implementation that
|
|
lets you go from a JMESPath-like string to a possibly deeply nested
|
|
object. The ``params`` are mutated in-place, so subsequent calls
|
|
can modify the same element by its index.
|
|
|
|
>>> build_param_structure(params, 'test[0]', 1)
|
|
>>> print(params)
|
|
{'test': [1]}
|
|
|
|
>>> build_param_structure(params, 'foo.bar[0].baz', 'hello world')
|
|
>>> print(params)
|
|
{'test': [1], 'foo': {'bar': [{'baz': 'hello, world'}]}}
|
|
|
|
"""
|
|
pos = params
|
|
parts = target.split('.')
|
|
|
|
# First, split into parts like 'foo', 'bar[0]', 'baz' and process
|
|
# each piece. It can either be a list or a dict, depending on if
|
|
# an index like `[0]` is present. We detect this via a regular
|
|
# expression, and keep track of where we are in params via the
|
|
# pos variable, walking down to the last item. Once there, we
|
|
# set the value.
|
|
for i, part in enumerate(parts):
|
|
# Is it indexing an array?
|
|
result = INDEX_RE.search(part)
|
|
if result:
|
|
if result.group(1):
|
|
if result.group(1) == '*':
|
|
part = part[:-3]
|
|
else:
|
|
# We have an explicit index
|
|
index = int(result.group(1))
|
|
part = part[: -len(f"{index}[]")]
|
|
else:
|
|
# Index will be set after we know the proper part
|
|
# name and that it's a list instance.
|
|
index = None
|
|
part = part[:-2]
|
|
|
|
if part not in pos or not isinstance(pos[part], list):
|
|
pos[part] = []
|
|
|
|
# This means we should append, e.g. 'foo[]'
|
|
if index is None:
|
|
index = len(pos[part])
|
|
|
|
while len(pos[part]) <= index:
|
|
# Assume it's a dict until we set the final value below
|
|
pos[part].append({})
|
|
|
|
# Last item? Set the value, otherwise set the new position
|
|
if i == len(parts) - 1:
|
|
pos[part][index] = value
|
|
else:
|
|
# The new pos is the *item* in the array, not the array!
|
|
pos = pos[part][index]
|
|
else:
|
|
if part not in pos:
|
|
pos[part] = {}
|
|
|
|
# Last item? Set the value, otherwise set the new position
|
|
if i == len(parts) - 1:
|
|
pos[part] = value
|
|
else:
|
|
pos = pos[part]
|