全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

FastAPI实现多重可选认证:同时支持Basic Auth与JWT Auth

本文详细阐述如何在FastAPI中实现灵活的多重认证机制,允许客户端通过Basic Auth或JWT Bearer Token中的任意一种方式访问受保护的API端点。核心策略是将各个认证依赖项配置为在认证失败时不立即抛出异常,而是返回None,从而将最终的授权决策和错误处理集中到一个高级组合依赖中。

在构建现代Web API时,常常需要支持多种认证方式以适应不同的客户端或使用场景。例如,某些内部服务可能使用Basic Auth,而外部客户端则可能依赖JWT。FastAPI的依赖注入系统非常强大,但默认情况下,当一个端点声明了多个认证依赖时,FastAPI会尝试强制所有依赖都成功。这导致了一个常见问题:如何让端点支持“或”逻辑,即只要任意一种认证方式成功即可访问?

本文将通过一个具体的示例,演示如何改造FastAPI的认证依赖,使其能够灵活地支持Basic Auth或JWT Auth的任选模式。

1. 核心理念:非阻塞式认证依赖

解决此问题的关键在于,让单个的认证依赖项在认证失败时不再立即抛出HTTPException,而是返回一个指示失败的值(通常是None)。这样,上层组合依赖就可以根据这些返回值来判断最终的认证状态。

对于FastAPI内置的认证方案,如HTTPBasic和OAuth2PasswordBearer,可以通过设置auto_error=False来实现这一目标。

2. 实现非阻塞式 Basic Authentication

首先,我们需要修改HTTPBasic实例,使其在缺少或无效的Basic Auth头时不会自动抛出401错误。

from fastapi.security import HTTPBasic, HTTPBasicCredentials
from fastapi import Depends, HTTPException, status
from typing import Optional, Annotated
import secrets # 用于安全地比较字符串

# 假设 settings.SESSION_LOGIN_USER 和 settings.SESSION_LOGIN_PASS 存储了正确的用户名和密码
# 假设 router 和 db_session 已定义
# 假设 User 模型已定义

# 1. 初始化 HTTPBasic,设置 auto_error=False
security = HTTPBasic(auto_error=False)

# 2. 创建 Basic Auth 依赖函数
def basic_logged_user(credentials: Annotated[Optional[HTTPBasicCredentials], Depends(security)]):
    """
    Basic Auth 认证依赖。
    如果认证失败,不抛出异常,而是返回 None。
    """
    if credentials is None:
        # 没有提供 Basic Auth 凭据
        return None

    # 安全地比较用户名和密码
    current_username_bytes = credentials.username.encode("utf8")
    correct_username_bytes = settings.SESSION_LOGIN_USER.encode("utf8")
    is_correct_username = secrets.compare_digest(current_username_bytes, correct_username_bytes)

    current_password_bytes = credentials.password.encode("utf8")
    correct_password_bytes = settings.SESSION_LOGIN_PASS.encode("utf8")
    is_correct_password = secrets.compare_digest(current_password_bytes, correct_password_bytes)

    if not (is_correct_username and is_correct_password):
        # 凭据无效,返回 None
        return None

    # 认证成功,返回用户名
    return credentials.username

在这个basic_logged_user函数中:

  • Depends(security)会尝试解析Basic Auth凭据。由于auto_error=False,如果凭据缺失,credentials将为None。
  • 我们显式地将credentials声明为Optional[HTTPBasicCredentials],以处理None的情况。
  • 无论凭据缺失还是无效,函数都返回None,而不是抛出HTTPException。

3. 实现非阻塞式 JWT Authentication

类似地,我们需要对JWT认证方案进行调整。通常,JWT认证会使用OAuth2PasswordBearer。我们也需要将其初始化为auto_error=False。

from fastapi.security import OAuth2PasswordBearer
# ... 其他导入 ...
# 假设 utils.verify_token 是一个验证JWT并返回用户数据的函数
# 假设 db_session 和 User 模型已定义

# 1. 初始化 OAuth2PasswordBearer,设置 auto_error=False
oauth2_scheme = OAuth2PasswordBearer(tokenUrl="token", auto_error=False) # tokenUrl 根据你的实际情况设置

# 2. 创建 JWT Auth 依赖函数
def jwt_logged_user(token: Annotated[Optional[str], Depends(oauth2_scheme)], 
                    db: Session = Depends(db_session)):
    """
    JWT Auth 认证依赖。
    如果认证失败,不抛出异常,而是返回 None。
    """
    if token is None:
        # 没有提供 JWT token
        return None

    try:
        # 假设 utils.verify_token 会验证 token 并返回一个包含用户信息的对象
        # 如果 token 无效,utils.verify_token 内部应该捕获异常或返回 None
        # 这里我们假设它在验证失败时会抛出异常,我们捕获它
        payload = utils.verify_token(token) # 假设 verify_token 成功返回 payload,失败抛异常
        username = payload.get("sub") # 假设 sub 字段存储用户名
        if not username:
            return None

        user = db.query(User).filter(User.username == username).first()
        if not user:
            return None

        return user # 认证成功,返回用户对象
    except Exception:
        # token 验证失败(过期、无效签名等)
        return None

在这个jwt_logged_user函数中:

  • Depends(oauth2_scheme)会尝试从请求头中提取JWT token。由于auto_error=False,如果token缺失,token参数将为None。
  • 我们同样将token声明为Optional[str]。
  • utils.verify_token应该被设计为在验证失败时抛出异常或返回一个可识别的失败状态。这里我们用try-except块来捕获验证失败的情况,并返回None。

4. 创建组合认证依赖 (auth_user)

现在我们有了两个非阻塞式的认证依赖,可以创建一个新的依赖函数来组合它们,实现“或”逻辑。

from fastapi import HTTPException, status
# ... 其他导入 ...

def auth_user(jwt_auth_result: Annotated[Optional[User], Depends(jwt_logged_user)], 
              basic_auth_result: Annotated[Optional[str], Depends(basic_logged_user)]):
    """
    组合认证依赖,支持 Basic Auth 或 JWT Auth。
    如果任一认证方式成功,则返回相应的认证用户。
    如果两种方式都失败,则抛出 401 异常。
    """
    if jwt_auth_result:
        # JWT 认证成功
        return jwt_auth_result

    if basic_auth_result:
        # Basic Auth 认证成功
        # 注意:这里返回的类型需要与 jwt_auth_result 的类型保持一致,
        # 或者根据你的业务逻辑决定返回哪个。
        # 为了简化,这里假设可以返回一个字符串作为认证成功的标识。
        # 如果需要返回统一的用户对象,则需要从 basic_auth_result 中获取用户对象。
        return basic_auth_result

    # 两种认证方式都失败,抛出 401 未授权异常
    raise HTTPException(
        status_code=status.HTTP_401_UNAUTHORIZED, 
        detail='Invalid Credentials',
        headers={"WWW-Authenticate": "Bearer, Basic"} # 提示客户端支持的认证方式
    )

在这个auth_user函数中:

  • 它接收jwt_logged_user和basic_logged_user的返回值。
  • 如果jwt_auth_result不为None(即JWT认证成功),则直接返回该结果。
  • 否则,如果basic_auth_result不为None(即Basic Auth认证成功),则返回该结果。
  • 只有当两种认证方式都返回None时,才抛出HTTP_401_UNAUTHORIZED异常。

5. 在API端点中使用组合认证

最后,将auth_user依赖添加到你的API端点中。

from fastapi import APIRouter, Depends
from sqlalchemy.orm import Session
# ... 其他导入 ...

router = APIRouter() # 假设你有一个 APIRouter 实例

@router.get("/users")
async def get_users(db: Session = Depends(db_session), 
                    logged_user: Annotated[Union[User, str], Depends(auth_user)]):
    """
    获取用户列表,需要 Basic Auth 或 JWT Auth 认证。
    """
    query_users = db.query(User).all()
    return query_users

现在,当客户端请求/users端点时:

  • 如果提供了有效的JWT Bearer Token,即使没有提供Basic Auth,请求也会成功。
  • 如果提供了有效的Basic Auth凭据,即使没有提供JWT Token,请求也会成功。
  • 只有当两种认证方式的凭据都缺失或无效时,才会收到401 Unauthorized响应。

总结与注意事项

通过将各个认证依赖项配置为非阻塞模式(auto_error=False并返回None),并创建一个中央组合依赖来处理最终的授权逻辑,我们成功地实现了FastAPI中多重可选认证的需求。

注意事项:

  1. 返回类型统一: 在auth_user中,jwt_auth_result可能返回User对象,而basic_auth_result可能返回str(用户名)。在实际应用中,你可能希望将它们统一为某种形式的用户对象,例如通过用户名从数据库中加载用户对象,确保auth_user始终返回User类型或其接口类型。
  2. 错误信息: 在auth_user抛出HTTPException时,WWW-Authenticate头部应包含所有支持的认证方案,以便客户端知道如何进行认证。
  3. 安全性: 在实现认证逻辑时,务必使用secrets.compare_digest等安全函数来比较密码,以防止定时攻击。
  4. 可扩展性: 这种模式非常灵活,你可以轻松地添加更多可选的认证方案(如API Key认证),只需为每个方案创建非阻塞依赖,并在auth_user中进行组合。

通过遵循这些原则,你可以构建出既安全又灵活的FastAPI认证系统。


# word  # session  # ai  # 常见问题  # red  # asic  # fastapi  # try  # Token  # 接口  # 对象  # 数据库  # 抛出  # 两种  # 客户端  # 在这个  # 也会  # 你可以  # 使其  # 可选  # 不为  # 将为 


相关文章: 新网站制作渠道有哪些,跪求一个无线渠道比较强的小说网站,我要发表小说?  无锡制作网站公司有哪些,无锡优八网络科技有限公司介绍?  如何通过二级域名建站提升品牌影响力?  如何在Windows服务器上快速搭建网站?  建站DNS解析失败?如何正确配置域名服务器?  建站之星CMS建站配置指南:模板选择与SEO优化技巧  制作网站的网址是什么,请问后缀为.com和.com.cn还有.cn的这三种网站是分别是什么类型的网站?  详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)  北京建设网站制作公司,北京古代建筑博物馆预约官网?  php能控制zigbee模块吗_php通过串口与cc2530 zigbee通信【介绍】  如何用wdcp快速搭建高效网站?  ,想在网上投简历,哪几个网站比较好?  如何用美橙互联一键搭建多站合一网站?  如何在IIS服务器上快速部署高效网站?  如何在建站宝盒中设置产品搜索功能?  c++怎么用jemalloc c++替换默认内存分配器【性能】  ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?  网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?  存储型VPS适合搭建中小型网站吗?  如何在沈阳梯子盘古建站优化SEO排名与功能模块?  企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?  制作表格网站有哪些,线上表格怎么弄?  网页设计与网站制作内容,怎样注册网站?  企业宣传片制作网站有哪些,传媒公司怎么找企业宣传片项目?  如何获取上海专业网站定制建站电话?  如何高效利用200m空间完成建站?  如何通过山东自助建站平台快速注册域名?  网站网页制作电话怎么打,怎样安装和使用钉钉软件免费打电话?  制作网站的软件下载免费,今日头条开宝箱老是需要下载怎么回事?  如何访问已购建站主机并解决登录问题?  沈阳制作网站公司排名,沈阳装饰协会官方网站?  建站之星如何快速生成多端适配网站?  Python多线程使用规范_线程安全解析【教程】  大连网站制作公司哪家好一点,大连买房网站哪个好?  公司网站制作费用多少,为公司建立一个网站需要哪些费用?  南平网站制作公司,2025年南平市事业单位报名时间?  如何选择长沙网站建站模板?H5响应式与品牌定制哪个更优?  建站之星后台密码遗忘或太弱?如何重置与强化?  免费公司网站制作软件,如何申请免费主页空间做自己的网站?  天津个人网站制作公司,天津网约车驾驶员从业资格证官网?  如何通过远程VPS快速搭建个人网站?  广州商城建站系统开发成本与周期如何控制?  定制建站策划方案_专业建站与网站建设方案一站式指南  如何在宝塔面板创建新站点?  建站之星安装提示数据库无法连接如何解决?  网站建设设计制作营销公司南阳,如何策划设计和建设网站?  广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?  建站之星如何配置系统实现高效建站?  如何在IIS中新建站点并配置端口与物理路径?  网站制作报价单模板图片,小松挖机官方网站报价? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。