FastAPI中构建安全且弹性的权限管控方案

在FastAPI中构建可靠且可变通的权限管控体系

环境配置要求(使用前需安装)

需安装以下依赖:
- fastapi==0.95.0
- uvicorn==0.21.1
- python-multipart==0.0.6
- sqlalchemy==1.4.46
- pydantic==1.10.7
- passlib==1.7.4

1. 权限系统核心原理

权限系统的核心是对请求的过滤机制,FastAPI利用依赖注入体系来实现层级化的验证流程。当请求到来时,会依次经历:
- 身份认证 → 角色验证 → 权限校验 这三级验证
- 每一层级均为独立的依赖项
- 权限数据存储于关系型数据库,支持动态化管理

2. 数据库模型设计

定义了用于存储用户、角色及权限关联信息的数据库模型:

from sqlalchemy import Boolean, Column, ForeignKey, Integer, String
from sqlalchemy.orm import relationship
from databases import Base


class User(Base):
    __tablename__ = "users"
    id = Column(Integer, primary_key=True)
    username = Column(String(50), unique=True)
    hashed_password = Column(String(300))
    is_active = Column(Boolean, default=True)
    role_id = Column(Integer, ForeignKey("roles.id"))

    role = relationship("Role", back_populates="users")


class Role(Base):
    __tablename__ = "roles"
    id = Column(Integer, primary_key=True)
    name = Column(String(20), unique=True)
    permissions = Column(String(500))  # 以逗号分隔存储权限标识
    users = relationship("User", back_populates="role")


class PermissionRegistry(Base):
    __tablename__ = "permission_registry"
    id = Column(Integer, primary_key=True)
    endpoint = Column(String(100))  # 路由路径
    method = Column(String(10))  # HTTP方法
    perm_code = Column(String(50))  # 权限标识

3. 权限验证依赖项

创建权限验证器类,在调用时检查当前用户的角色权限是否包含所需权限:

from fastapi import Depends, HTTPException, status
from pydantic import BaseModel


class PermissionValidator:
    def __init__(self, required_perm: str):
        self.required_perm = required_perm

    async def __call__(self,
                       current_user: User = Depends(get_current_user),
                       db: Session = Depends(get_db)):
        # 获取角色关联的权限
        role_perms = current_user.role.permissions.split(",")

        # 验证权限是否存在
        if self.required_perm not in role_perms:
            raise HTTPException(
                status_code=status.HTTP_403_FORBIDDEN,
                detail="无访问权限"
            )

        # 记录审计日志(示例)
        audit_log = AuditLog(
            user_id=current_user.id,
            action=f"访问需要 {self.required_perm} 权限的端点"
        )
        db.add(audit_log)
        db.commit()


# 使用示例
@app.get("/admin/dashboard")
async def admin_dashboard(
        perm_check: bool = Depends(PermissionValidator("admin_dashboard"))):
    return {"message": "欢迎来到管理面板"}

4. 动态路由权限注册

通过定义模型接收权限信息并动态注册到数据库:

class PermissionRegistration(BaseModel):
    endpoint: str
    methods: List[str]
    perm_code: str


@app.post("/manage/permissions")
async def register_permission(
        perm_data: PermissionRegistration,
        db: Session = Depends(get_db)
):
    for method in perm_data.methods:
        existing = db.query(PermissionRegistry).filter_by(
            endpoint=perm_data.endpoint,
            method=method
        ).first()

        if not existing:
            new_perm = PermissionRegistry(
                endpoint=perm_data.endpoint,
                method=method,
                perm_code=perm_data.perm_code
            )
            db.add(new_perm)

    db.commit()
    return {"status": "权限注册成功"}

5. 实时权限检查中间件

设置中间件实时检查请求权限,跳过非业务端点:

@app.middleware("http")
async def dynamic_permission_check(request: Request, call_next):
    # 跳过文档相关端点
    if request.url.path.startswith(("/docs", "/redoc")):
        return await call_next(request)

    # 查询权限注册表
    db = SessionLocal()
    perm_record = db.query(PermissionRegistry).filter_by(
        endpoint=request.url.path,
        method=request.method
    ).first()

    if perm_record:
        # 验证用户权限
        current_user = await get_current_user(request)
        if perm_record.perm_code not in current_user.role.permissions.split(","):
            return JSONResponse(
                status_code=403,
                content={"detail": "权限不足"}
            )

    response = await call_next(request)
    return response

课后小测

  1. 当用户访问需要特定权限的接口但自身权限不足时,系统会返回什么状态码?
    A) 401 B) 403 C) 404 D) 500
    答案:B) 403。权限验证不通过时会返回403 Forbidden状态码。

  2. 如何防止权限注册接口被未授权访问?
    A) 添加JWT认证依赖
    B) 限制仅管理员角色可访问
    C) 同时实现A和B
    D) 无需保护该接口
    答案:C) 需同时进行JWT认证并限制仅管理员角色访问。

常见报错处理

  1. 422 Unprocessable Entity
    原因:请求体不符合Pydantic模型验证
    解决:检查字段类型是否正确,补充缺失的必填字段

  2. AttributeError: 'NoneType' has no attribute 'permissions'
    原因:用户角色未正确关联
    解决:检查数据库中用户与角色的关联关系,确保每个用户都有对应的角色

  3. 数据库连接超时
    预防:配置SQLAlchemy连接池,示例配置:
    python
    SQLALCHEMY_DATABASE_URL = "postgresql://user:pass@localhost/dbname?connect_timeout=10"
    engine = create_engine(
    SQLALCHEMY_DATABASE_URL,
    pool_size=20,
    max_overflow=10,
    pool_timeout=30
    )

更多详细内容可跳转至个人博客查看:如何在FastAPI中打造一个既安全又灵活的权限管理系统? | cmdragon's Blog

往期文章回顾

文章整理自互联网,只做测试使用。发布者:Lomu,转转请注明出处:https://www.it1024doc.com/12618.html

(0)
LomuLomu
上一篇 2025 年 6 月 23 日
下一篇 2025 年 6 月 24 日

相关推荐

  • IDEA激活秘籍公开,永久免费不是梦!

    声明:以下补丁与激活码均搜集自互联网,仅供个人学习参考,禁止商业用途。若条件允许,请支持正版:https://panghu.hicxy.com/shop/?id=18 废话少说,先放一张成功激活到 2099 年的截图镇楼,爽到飞起! 下面用图文方式手把手演示最新版 IntelliJ IDEA 2025.2.1 的激活流程。 前期准备 如果之前尝试过其他补丁失…

    IDEA破解教程 2025 年 9 月 21 日
    9000
  • 2024 WebStorm最新激活码,WebStorm永久免费激活码2025-01-11 更新

    WebStorm 2024最新激活码 以下是最新的WebStorm激活码,更新时间:2025-01-11 🔑 激活码使用说明 1️⃣ 复制下方激活码 2️⃣ 打开 WebStorm 软件 3️⃣ 在菜单栏中选择 Help -> Register 4️⃣ 选择 Activation Code 5️⃣ 粘贴激活码,点击 Activate ⚠️ 必看!必看! 🔥 …

    2025 年 1 月 11 日
    59500
  • SpringBoot与Langchain4j AIService深度融合运用指南

    文章标题: SpringBoot与Langchain4j的AIService深度融合应用指引 文章内容: 目录 一、开篇阐述 二、AIService相关讲解 2.1 AiService的定义 2.2 AiService的主要功能 2.3 AiService的使用流程 三、AIService操作实战 3.1 前期准备 3.1.1 获取apikey 3.1.2 …

    2025 年 9 月 14 日
    10300
  • IDEA破解从入门到精通|实用技巧全公开!

    免责声明:下文所提及的 IntelliJ IDEA 破解补丁与激活码均源自互联网公开渠道,仅供个人学习与研究,禁止任何商业用途。若条件允许,请支持正版:https://www.jetbrains.com/idea/buy/ IntelliJ IDEA 是 JetBrains 旗下一款跨平台 IDE,支持 Windows、macOS 与 Linux。下面手把手…

    IDEA破解教程 2025 年 9 月 19 日
    12700
  • 2024 PyCharm最新激活码,PyCharm永久免费激活码2025-01-15 更新

    PyCharm 2024最新激活码 以下是最新的PyCharm激活码,更新时间:2025-01-15 🔑 激活码使用说明 1️⃣ 复制下方激活码 2️⃣ 打开 PyCharm 软件 3️⃣ 在菜单栏中选择 Help -> Register 4️⃣ 选择 Activation Code 5️⃣ 粘贴激活码,点击 Activate ⚠️ 必看!必看! 🔥 获取最…

    2025 年 1 月 15 日
    67900

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信