update:1.更新用户管理-支持管理员新增用户
This commit is contained in:
@@ -0,0 +1,393 @@
|
||||
"""
|
||||
管理员API - 用户管理功能
|
||||
"""
|
||||
from fastapi import APIRouter, HTTPException, Request, Depends
|
||||
from pydantic import BaseModel, Field
|
||||
from typing import Optional, List
|
||||
from datetime import datetime
|
||||
import hashlib
|
||||
from sqlalchemy import select
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
from app.database import get_db, init_db
|
||||
from app.models.user import User
|
||||
from app.user_manager import user_manager
|
||||
from app.user_password import password_manager
|
||||
from app.logger import get_logger
|
||||
|
||||
logger = get_logger(__name__)
|
||||
|
||||
router = APIRouter(prefix="/admin", tags=["管理员"])
|
||||
|
||||
|
||||
# ==================== 请求/响应模型 ====================
|
||||
|
||||
class CreateUserRequest(BaseModel):
|
||||
"""创建用户请求"""
|
||||
username: str = Field(..., min_length=3, max_length=20, description="用户名")
|
||||
display_name: str = Field(..., min_length=2, max_length=50, description="显示名称")
|
||||
password: Optional[str] = Field(None, min_length=6, description="初始密码,留空则自动生成")
|
||||
avatar_url: Optional[str] = Field(None, description="头像URL")
|
||||
trust_level: int = Field(0, ge=0, le=9, description="信任等级")
|
||||
is_admin: bool = Field(False, description="是否为管理员")
|
||||
|
||||
|
||||
class UpdateUserRequest(BaseModel):
|
||||
"""更新用户请求"""
|
||||
display_name: Optional[str] = Field(None, min_length=2, max_length=50)
|
||||
avatar_url: Optional[str] = None
|
||||
trust_level: Optional[int] = Field(None, ge=-1, le=9)
|
||||
is_admin: Optional[bool] = Field(None, description="是否为管理员")
|
||||
|
||||
|
||||
class ToggleStatusRequest(BaseModel):
|
||||
"""切换用户状态请求"""
|
||||
is_active: bool = Field(..., description="true=启用, false=禁用")
|
||||
|
||||
|
||||
class ResetPasswordRequest(BaseModel):
|
||||
"""重置密码请求"""
|
||||
new_password: Optional[str] = Field(None, min_length=6, description="新密码,留空则重置为默认密码")
|
||||
|
||||
|
||||
class UserResponse(BaseModel):
|
||||
"""用户响应"""
|
||||
user_id: str
|
||||
username: str
|
||||
display_name: str
|
||||
avatar_url: Optional[str]
|
||||
trust_level: int
|
||||
is_admin: bool
|
||||
is_active: bool
|
||||
linuxdo_id: str
|
||||
created_at: str
|
||||
last_login: Optional[str]
|
||||
|
||||
|
||||
class CreateUserResponse(BaseModel):
|
||||
"""创建用户响应"""
|
||||
success: bool
|
||||
message: str
|
||||
user: dict
|
||||
default_password: Optional[str] = None
|
||||
|
||||
|
||||
# ==================== 权限检查依赖 ====================
|
||||
|
||||
async def check_admin(request: Request) -> User:
|
||||
"""检查管理员权限"""
|
||||
user = getattr(request.state, "user", None)
|
||||
if not user:
|
||||
raise HTTPException(status_code=401, detail="未登录")
|
||||
|
||||
if not user.is_admin:
|
||||
raise HTTPException(status_code=403, detail="需要管理员权限")
|
||||
|
||||
return user
|
||||
|
||||
|
||||
# ==================== API 端点 ====================
|
||||
|
||||
@router.get("/users")
|
||||
async def get_users(
|
||||
admin: User = Depends(check_admin),
|
||||
db: AsyncSession = Depends(get_db)
|
||||
):
|
||||
"""获取用户列表(仅管理员)"""
|
||||
try:
|
||||
all_users = await user_manager.get_all_users()
|
||||
|
||||
users_data = []
|
||||
for user in all_users:
|
||||
# user_manager 返回的是 Pydantic User 对象,直接转为 dict
|
||||
user_dict = user.model_dump()
|
||||
user_dict["is_active"] = user.trust_level != -1
|
||||
users_data.append(user_dict)
|
||||
|
||||
logger.info(f"管理员 {admin.user_id} 获取用户列表,共 {len(users_data)} 个用户")
|
||||
|
||||
return {
|
||||
"total": len(users_data),
|
||||
"users": users_data
|
||||
}
|
||||
except Exception as e:
|
||||
logger.error(f"获取用户列表失败: {str(e)}", exc_info=True)
|
||||
raise HTTPException(status_code=500, detail=f"获取用户列表失败: {str(e)}")
|
||||
|
||||
|
||||
@router.post("/users")
|
||||
async def create_user(
|
||||
data: CreateUserRequest,
|
||||
admin: User = Depends(check_admin),
|
||||
db: AsyncSession = Depends(get_db)
|
||||
):
|
||||
"""添加用户(仅管理员)"""
|
||||
try:
|
||||
# 检查用户名是否已存在
|
||||
all_users = await user_manager.get_all_users()
|
||||
for user in all_users:
|
||||
if user.username == data.username:
|
||||
raise HTTPException(status_code=409, detail="用户名已存在")
|
||||
|
||||
# 生成用户ID
|
||||
user_id = f"admin_created_{hashlib.md5(data.username.encode()).hexdigest()[:16]}"
|
||||
|
||||
# 创建用户
|
||||
new_user = await user_manager.create_or_update_from_linuxdo(
|
||||
linuxdo_id=user_id,
|
||||
username=data.username,
|
||||
display_name=data.display_name,
|
||||
avatar_url=data.avatar_url,
|
||||
trust_level=data.trust_level
|
||||
)
|
||||
|
||||
# 设置管理员标志
|
||||
if data.is_admin:
|
||||
# 直接更新数据库中的is_admin字段
|
||||
async with await user_manager._get_session() as session:
|
||||
result = await session.execute(
|
||||
select(User).where(User.user_id == user_id)
|
||||
)
|
||||
db_user = result.scalar_one_or_none()
|
||||
if db_user:
|
||||
db_user.is_admin = True
|
||||
await session.commit()
|
||||
new_user.is_admin = True
|
||||
|
||||
# 设置密码
|
||||
actual_password = await password_manager.set_password(
|
||||
user_id=new_user.user_id,
|
||||
username=data.username,
|
||||
password=data.password
|
||||
)
|
||||
|
||||
# 初始化用户数据库
|
||||
try:
|
||||
await init_db(new_user.user_id)
|
||||
logger.info(f"用户 {new_user.user_id} 数据库初始化成功")
|
||||
except Exception as e:
|
||||
logger.error(f"用户 {new_user.user_id} 数据库初始化失败: {e}")
|
||||
|
||||
logger.info(f"管理员 {admin.user_id} 创建了新用户 {new_user.user_id} ({data.username})")
|
||||
|
||||
return CreateUserResponse(
|
||||
success=True,
|
||||
message="用户创建成功",
|
||||
user=new_user.model_dump(),
|
||||
default_password=actual_password if not data.password else None
|
||||
)
|
||||
|
||||
except HTTPException:
|
||||
raise
|
||||
except Exception as e:
|
||||
logger.error(f"创建用户失败: {str(e)}", exc_info=True)
|
||||
raise HTTPException(status_code=500, detail=f"创建用户失败: {str(e)}")
|
||||
|
||||
|
||||
@router.put("/users/{user_id}")
|
||||
async def update_user(
|
||||
user_id: str,
|
||||
data: UpdateUserRequest,
|
||||
admin: User = Depends(check_admin),
|
||||
db: AsyncSession = Depends(get_db)
|
||||
):
|
||||
"""编辑用户信息(仅管理员)"""
|
||||
try:
|
||||
# 获取目标用户
|
||||
target_user = await user_manager.get_user(user_id)
|
||||
if not target_user:
|
||||
raise HTTPException(status_code=404, detail="用户不存在")
|
||||
|
||||
# 更新用户信息
|
||||
async with await user_manager._get_session() as session:
|
||||
result = await session.execute(
|
||||
select(User).where(User.user_id == user_id)
|
||||
)
|
||||
db_user = result.scalar_one_or_none()
|
||||
|
||||
if not db_user:
|
||||
raise HTTPException(status_code=404, detail="用户不存在")
|
||||
|
||||
# 更新字段
|
||||
if data.display_name is not None:
|
||||
db_user.display_name = data.display_name
|
||||
if data.avatar_url is not None:
|
||||
db_user.avatar_url = data.avatar_url
|
||||
if data.trust_level is not None:
|
||||
db_user.trust_level = data.trust_level
|
||||
if data.is_admin is not None:
|
||||
# 检查是否是最后一个管理员
|
||||
if db_user.is_admin and not data.is_admin:
|
||||
all_users = await user_manager.get_all_users()
|
||||
admin_count = sum(1 for u in all_users if u.is_admin)
|
||||
if admin_count <= 1:
|
||||
raise HTTPException(status_code=400, detail="不能取消最后一个管理员的权限")
|
||||
db_user.is_admin = data.is_admin
|
||||
|
||||
await session.commit()
|
||||
await session.refresh(db_user)
|
||||
|
||||
logger.info(f"管理员 {admin.user_id} 更新了用户 {user_id} 的信息")
|
||||
|
||||
updated_user = await user_manager.get_user(user_id)
|
||||
user_dict = updated_user.model_dump()
|
||||
user_dict["is_active"] = updated_user.trust_level != -1
|
||||
|
||||
return {
|
||||
"success": True,
|
||||
"message": "用户信息更新成功",
|
||||
"user": user_dict
|
||||
}
|
||||
|
||||
except HTTPException:
|
||||
raise
|
||||
except Exception as e:
|
||||
logger.error(f"更新用户失败: {str(e)}", exc_info=True)
|
||||
raise HTTPException(status_code=500, detail=f"更新用户失败: {str(e)}")
|
||||
|
||||
|
||||
@router.post("/users/{user_id}/toggle-status")
|
||||
async def toggle_user_status(
|
||||
user_id: str,
|
||||
data: ToggleStatusRequest,
|
||||
admin: User = Depends(check_admin),
|
||||
db: AsyncSession = Depends(get_db)
|
||||
):
|
||||
"""切换用户状态(启用/禁用)(仅管理员)"""
|
||||
try:
|
||||
# 不允许禁用自己
|
||||
if user_id == admin.user_id:
|
||||
raise HTTPException(status_code=400, detail="不能禁用自己的账号")
|
||||
|
||||
# 获取目标用户
|
||||
target_user = await user_manager.get_user(user_id)
|
||||
if not target_user:
|
||||
raise HTTPException(status_code=404, detail="用户不存在")
|
||||
|
||||
# 更新状态
|
||||
async with await user_manager._get_session() as session:
|
||||
result = await session.execute(
|
||||
select(User).where(User.user_id == user_id)
|
||||
)
|
||||
db_user = result.scalar_one_or_none()
|
||||
|
||||
if not db_user:
|
||||
raise HTTPException(status_code=404, detail="用户不存在")
|
||||
|
||||
if data.is_active:
|
||||
# 启用用户:恢复trust_level为0(或之前的值)
|
||||
db_user.trust_level = 0
|
||||
else:
|
||||
# 禁用用户:设置trust_level为-1
|
||||
db_user.trust_level = -1
|
||||
|
||||
await session.commit()
|
||||
|
||||
status_text = "启用" if data.is_active else "禁用"
|
||||
logger.info(f"管理员 {admin.user_id} {status_text}了用户 {user_id}")
|
||||
|
||||
return {
|
||||
"success": True,
|
||||
"message": f"用户已{status_text}",
|
||||
"is_active": data.is_active
|
||||
}
|
||||
|
||||
except HTTPException:
|
||||
raise
|
||||
except Exception as e:
|
||||
logger.error(f"切换用户状态失败: {str(e)}", exc_info=True)
|
||||
raise HTTPException(status_code=500, detail=f"切换用户状态失败: {str(e)}")
|
||||
|
||||
|
||||
@router.post("/users/{user_id}/reset-password")
|
||||
async def reset_password(
|
||||
user_id: str,
|
||||
data: ResetPasswordRequest,
|
||||
admin: User = Depends(check_admin),
|
||||
db: AsyncSession = Depends(get_db)
|
||||
):
|
||||
"""重置用户密码(仅管理员)"""
|
||||
try:
|
||||
# 获取目标用户
|
||||
target_user = await user_manager.get_user(user_id)
|
||||
if not target_user:
|
||||
raise HTTPException(status_code=404, detail="用户不存在")
|
||||
|
||||
# 重置密码
|
||||
actual_password = await password_manager.set_password(
|
||||
user_id=user_id,
|
||||
username=target_user.username,
|
||||
password=data.new_password
|
||||
)
|
||||
|
||||
logger.info(f"管理员 {admin.user_id} 重置了用户 {user_id} 的密码")
|
||||
|
||||
return {
|
||||
"success": True,
|
||||
"message": "密码重置成功",
|
||||
"new_password": actual_password
|
||||
}
|
||||
|
||||
except HTTPException:
|
||||
raise
|
||||
except Exception as e:
|
||||
logger.error(f"重置密码失败: {str(e)}", exc_info=True)
|
||||
raise HTTPException(status_code=500, detail=f"重置密码失败: {str(e)}")
|
||||
|
||||
|
||||
@router.delete("/users/{user_id}")
|
||||
async def delete_user(
|
||||
user_id: str,
|
||||
admin: User = Depends(check_admin),
|
||||
db: AsyncSession = Depends(get_db)
|
||||
):
|
||||
"""删除用户(仅管理员,慎用)"""
|
||||
try:
|
||||
# 不允许删除自己
|
||||
if user_id == admin.user_id:
|
||||
raise HTTPException(status_code=400, detail="不能删除自己的账号")
|
||||
|
||||
# 获取目标用户
|
||||
target_user = await user_manager.get_user(user_id)
|
||||
if not target_user:
|
||||
raise HTTPException(status_code=404, detail="用户不存在")
|
||||
|
||||
# 检查是否是最后一个管理员
|
||||
if target_user.is_admin:
|
||||
all_users = await user_manager.get_all_users()
|
||||
admin_count = sum(1 for u in all_users if u.is_admin)
|
||||
if admin_count <= 1:
|
||||
raise HTTPException(status_code=400, detail="不能删除最后一个管理员账号")
|
||||
|
||||
# 删除用户(包括密码记录)
|
||||
async with await user_manager._get_session() as session:
|
||||
# 删除用户记录
|
||||
result = await session.execute(
|
||||
select(User).where(User.user_id == user_id)
|
||||
)
|
||||
db_user = result.scalar_one_or_none()
|
||||
if db_user:
|
||||
await session.delete(db_user)
|
||||
|
||||
# 删除密码记录
|
||||
from app.models.user import UserPassword
|
||||
result = await session.execute(
|
||||
select(UserPassword).where(UserPassword.user_id == user_id)
|
||||
)
|
||||
pwd_record = result.scalar_one_or_none()
|
||||
if pwd_record:
|
||||
await session.delete(pwd_record)
|
||||
|
||||
await session.commit()
|
||||
|
||||
logger.warning(f"管理员 {admin.user_id} 删除了用户 {user_id}")
|
||||
|
||||
return {
|
||||
"success": True,
|
||||
"message": "用户已删除"
|
||||
}
|
||||
|
||||
except HTTPException:
|
||||
raise
|
||||
except Exception as e:
|
||||
logger.error(f"删除用户失败: {str(e)}", exc_info=True)
|
||||
raise HTTPException(status_code=500, detail=f"删除用户失败: {str(e)}")
|
||||
+2
-1
@@ -135,12 +135,13 @@ from app.api import (
|
||||
projects, outlines, characters, chapters,
|
||||
wizard_stream, relationships, organizations,
|
||||
auth, users, settings, writing_styles, memories,
|
||||
mcp_plugins
|
||||
mcp_plugins, admin
|
||||
)
|
||||
|
||||
app.include_router(auth.router, prefix="/api")
|
||||
app.include_router(users.router, prefix="/api")
|
||||
app.include_router(settings.router, prefix="/api")
|
||||
app.include_router(admin.router, prefix="/api")
|
||||
|
||||
app.include_router(projects.router, prefix="/api")
|
||||
app.include_router(wizard_stream.router, prefix="/api")
|
||||
|
||||
@@ -1,9 +1,12 @@
|
||||
"""
|
||||
认证中间件 - 从 Cookie 中提取用户信息并注入到 request.state
|
||||
"""
|
||||
from fastapi import Request
|
||||
from fastapi import Request, HTTPException
|
||||
from starlette.middleware.base import BaseHTTPMiddleware
|
||||
from app.user_manager import user_manager
|
||||
from app.logger import get_logger
|
||||
|
||||
logger = get_logger(__name__)
|
||||
|
||||
|
||||
class AuthMiddleware(BaseHTTPMiddleware):
|
||||
@@ -20,9 +23,18 @@ class AuthMiddleware(BaseHTTPMiddleware):
|
||||
if user_id:
|
||||
user = await user_manager.get_user(user_id)
|
||||
if user:
|
||||
request.state.user_id = user_id
|
||||
request.state.user = user
|
||||
request.state.is_admin = user.is_admin
|
||||
# 检查用户是否被禁用 (trust_level = -1)
|
||||
if user.trust_level == -1:
|
||||
logger.warning(f"禁用用户尝试访问: {user_id} ({user.username})")
|
||||
# 清除用户状态,视为未登录
|
||||
request.state.user_id = None
|
||||
request.state.user = None
|
||||
request.state.is_admin = False
|
||||
else:
|
||||
# 用户正常,注入状态
|
||||
request.state.user_id = user_id
|
||||
request.state.user = user
|
||||
request.state.is_admin = user.is_admin
|
||||
else:
|
||||
# 用户不存在,清除状态
|
||||
request.state.user_id = None
|
||||
|
||||
@@ -164,6 +164,9 @@ class MCPTestService:
|
||||
# 转换为OpenAI Function Calling格式
|
||||
openai_tools = self._convert_tools_to_openai_format(tools)
|
||||
|
||||
logger.info(f"📋 转换后的OpenAI工具数量: {len(openai_tools)}")
|
||||
logger.debug(f"📋 OpenAI工具列表: {[t['function']['name'] for t in openai_tools]}")
|
||||
|
||||
# 调用AI选择工具
|
||||
prompt = f"""你是MCP插件测试助手,需要测试插件 '{plugin.plugin_name}' 的功能。
|
||||
|
||||
|
||||
Reference in New Issue
Block a user