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 日

相关推荐

  • CLion激活是否支持Toolbox?如何配合使用?

    重要提示:下文所涉及的破解补丁、激活码均来源于互联网公开分享,仅供个人学习研究,禁止商业用途。若经济允许,请支持正版! CLion 是 JetBrains 家族中针对 C/C++ 的重量级 IDE,跨 Windows、macOS、Linux 三大平台。下面手把手教你用破解补丁一键永久解锁全部高级特性,任何版本、任何系统通用。 激活后效果速览 补丁生效后,授权…

    2025 年 9 月 21 日
    2400
  • 2024 GoLand最新激活码,GoLand永久免费激活码2024-12-30 更新

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

    2024 年 12 月 30 日
    46700
  • 履约系统:应用层、领域层、集成关系设计

    大家好,我是汤师爷~ 在这篇文章中,我们一起探讨订单履约系统的应用架构设计。 应用架构设计 我们前面讨论了系统的核心概念模型和拆单逻辑。接下来,让我们从应用架构的角度,深入了解系统的各个层次。这包括应用层、领域层,以及与其他系统的集成关系。 应用层能力 应用层定义软件的应用功能,它负责接收用户请求,协调领域层能力来执行任务,并将结果返回给用户,核心模块包括:…

    2025 年 1 月 6 日
    37800
  • 2025年最新DataGrip激活码永久破解教程 – 支持JetBrains全家桶2099年授权

    前言 本教程适用于JetBrains系列开发工具,包括DataGrip、PyCharm、IDEA等主流IDE,是目前最完整的破解方案! 先展示DataGrip最新版破解效果,如图所示,软件授权已成功延长至2099年,彻底解决激活问题! 下面将通过详细的图文指导,一步步教你如何永久激活DataGrip至2099年。该方法兼容:- Windows/Mac/Lin…

    DataGrip激活码 2025 年 8 月 18 日
    41600
  • 2025年最新IDEA激活码及永久破解教程(支持Windows/Mac/Linux)

    IntelliJ IDEA作为Java开发者的首选IDE,以其强大的功能和丰富的插件生态著称。不过其高昂的授权费用也让不少开发者望而却步。本文将为大家详细介绍一套完整的IDEA永久激活方案,有效期至2099年! 特别声明:本教程仅供学习研究使用,请支持JetBrains官方正版软件。 一、准备工作 在开始破解前,请确保:1. 已卸载任何非官方渠道下载的IDE…

    IDEA破解教程 2025 年 8 月 15 日
    11000

发表回复

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

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

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

关注微信