FastAPI 多种认证方式(任选其一)实现指南

FastAPI 多种认证方式(任选其一)实现指南

本教程详细阐述了如何在 fastapi 中实现多种认证机制(如 basic auth 和 jwt auth),并允许客户端任选其一进行认证。核心方法是修改各个认证依赖项,使其在认证失败时返回 `none` 而非立即抛出异常,从而使一个组合认证依赖能够基于“或”逻辑判断任一认证是否成功,最终实现灵活的多重认证支持。

在构建现代 Web API 时,支持多种认证方式以满足不同客户端或场景的需求是常见的实践。例如,您可能希望同时支持传统的 HTTP Basic 认证和基于令牌的 JWT 认证。然而,在 FastAPI 中直接将多个认证依赖项组合使用时,默认行为可能会导致问题:FastAPI 会强制所有依赖项都通过验证,如果其中一个依赖项在解析阶段抛出 HTTPException,那么后续的依赖项将不会被执行,从而无法实现“任选其一”的逻辑。

理解 FastAPI 依赖注入的默认行为

FastAPI 的依赖注入系统在处理多个 Depends 依赖时,遵循严格的顺序和错误处理机制。当一个依赖项(例如,一个认证函数)内部抛出 HTTPException 时,FastAPI 会立即捕获该异常并返回相应的 HTTP 响应,而不会继续执行后续的依赖项或路由处理函数。这意味着,如果您尝试通过一个自定义的组合依赖函数来包装多个认证依赖,并期望它们能像逻辑“或”一样工作,那么默认情况下是行不通的。

例如,以下尝试实现“或”逻辑的组合认证依赖:

# 假设 jwt_logged_user 和 basic_logged_user 在认证失败时会抛出 HTTPExceptiondef auth_user(jwt_auth: Any = Depends(jwt_logged_user),              basic_auth: Any = Depends(basic_logged_user)):    if jwt_auth or basic_auth:        return jwt_auth or basic_auth    raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail='Invalid Credentials')

如果 jwt_logged_user 认证失败并抛出异常,那么 basic_auth 依赖将永远不会被解析,auth_user 函数体也永远不会执行,从而无法检查 Basic Auth 是否有效。

核心解决方案:优雅地处理认证失败

要实现“任选其一”的认证逻辑,关键在于修改各个独立的认证依赖项,使其在认证失败时不再立即抛出 HTTPException,而是返回一个指示失败的值(例如 None)。这样,组合认证依赖就可以接收到所有独立认证的结果,并根据这些结果进行逻辑判断。

1. 修改 HTTP Basic 认证依赖

对于 HTTP Basic 认证,FastAPI 提供了 HTTPBasic 类。通过设置 auto_error=False,可以禁用其在认证失败时自动抛出 HTTPException 的行为。

首先,更新 HTTPBasic 实例:

from fastapi.security import HTTPBasic, HTTPBasicCredentialsfrom typing import Annotated, Optionalimport secrets# ... 其他导入 ...# 将 auto_error 设置为 Falsesecurity = HTTPBasic(auto_error=False)def basic_logged_user(credentials: Annotated[Optional[HTTPBasicCredentials], Depends(security)]):    """    处理 HTTP Basic 认证。    如果认证失败,返回 None 而非抛出异常。    """    if credentials is None:        # 没有提供 Basic 认证凭据,或者凭据格式不正确 (由 auto_error=False 处理)        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

关键点:

security = HTTPBasic(auto_error=False):禁用 HTTPBasic 在凭据缺失或格式错误时自动抛出 401 异常。credentials: Annotated[Optional[HTTPBasicCredentials], Depends(security)]:声明 credentials 为 Optional 类型,因为 security 在 auto_error=False 时可能返回 None。if credentials is None::检查是否提供了凭据。if not (is_correct_username and is_correct_password): return None:在自定义逻辑中,如果凭据无效,也返回 None。

2. 修改 JWT 认证依赖

对于 JWT 认证,通常会使用 OAuth2PasswordBearer。同样,通过设置 auto_error=False 来禁用其自动错误处理。此外,您的令牌验证函数 (utils.verify_token) 也应该被修改,以在验证失败时返回 None 或被 try-except 块包裹。

假设 utils.OAuth2_scheme 是 OAuth2PasswordBearer 的实例,并且 utils.verify_token 函数负责验证 JWT。

from fastapi.security import OAuth2PasswordBearerfrom sqlalchemy.orm import Sessionfrom typing import Annotated, Optionalfrom jose import JWTError, jwt # 假设您使用 jose 库from pydantic import BaseModel# ... 其他导入 ...# 假设 utils 模块包含 OAuth2_scheme 和 verify_token# utils.OAuth2_scheme 应该被初始化为 auto_error=False# 例如:# class TokenData(BaseModel):#     username: Optional[str] = None# class Utils:#     OAuth2_scheme = OAuth2PasswordBearer(tokenUrl="token", auto_error=False)#     SECRET_KEY = "your-secret-key" # 替换为实际的密钥#     ALGORITHM = "HS256"#     def verify_token(self, token: str) -> dict:#         try:#             payload = jwt.decode(token, self.SECRET_KEY, algorithms=[self.ALGORITHM])#             username: str = payload.get("sub")#             if username is None:#                 raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Invalid token payload")#             return {"username": username}#         except JWTError:#             raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Could not validate credentials")# utils = Utils()def jwt_logged_user(token: Annotated[Optional[str], Depends(utils.OAuth2_scheme)],                    db: Session = Depends(db_session)):    """    处理 JWT 认证。    如果认证失败,返回 None 而非抛出异常。    """    if token is None:        # 没有提供 JWT 令牌,或者令牌格式不正确 (由 auto_error=False 处理)        return None    try:        # 尝试验证令牌,并从 payload 中获取用户名        # 假设 utils.verify_token 在验证失败时会抛出 HTTPException 或 JWTError        payload = utils.verify_token(token) # 假设返回一个包含用户信息的字典        username = payload.get("username") # 根据实际 payload 结构调整        if username is None:            return None # 令牌有效但没有找到用户名信息        # 查询数据库以验证用户是否存在        user = db.query(User).filter(User.username == username).first()        if user is None:            return None # 用户不存在        # 认证成功,返回用户名        return user.username    except HTTPException:        # 捕获 verify_token 内部抛出的 HTTPException        return None    except JWTError: # 捕获 jose 库的 JWT 错误        return None    except Exception:        # 捕获其他未知错误        return None

关键点:

utils.OAuth2_scheme = OAuth2PasswordBearer(tokenUrl=”token”, auto_error=False):确保 OAuth2PasswordBearer 实例设置了 auto_error=False。token: Annotated[Optional[str], Depends(utils.OAuth2_scheme)]:声明 token 为 Optional 类型。if token is None::检查是否提供了令牌。try…except 块:包裹 utils.verify_token 调用,捕获可能抛出的异常,并在捕获到异常时返回 None。return user.username:认证成功后返回用户名,与 Basic 认证的返回类型保持一致。

3. 实现组合认证逻辑

现在,当 basic_logged_user 和 jwt_logged_user 认证失败时都返回 None,我们可以创建一个新的组合依赖项来检查任一认证是否成功。

from fastapi import HTTPException, status, Dependsfrom typing import Annotated, Optional, Union# ... 其他导入 ...def auth_user(jwt_username: Annotated[Optional[str], Depends(jwt_logged_user)],              basic_username: Annotated[Optional[str], Depends(basic_logged_user)]) -> str:    """    组合认证依赖,允许通过 JWT 或 Basic Auth 中的任一种进行认证。    如果任一认证成功,返回对应的用户名。    如果两种认证都失败,则抛出 401 异常。    """    if jwt_username:        # JWT 认证成功        return jwt_username    if basic_username:        # Basic Auth 认证成功        return basic_username    # 如果两种认证都失败,则抛出未授权异常    raise HTTPException(        status_code=status.HTTP_401_UNAUTHORIZED,        detail='Invalid Credentials',        headers={"WWW-Authenticate": "Basic, Bearer"}, # 提示支持的认证方式    )# 路由使用组合认证依赖@router.get("/users/")async def get_users(db: Session = Depends(db_session),                     logged_username: Annotated[str, Depends(auth_user)]):    """    获取用户列表,需要通过 JWT 或 Basic Auth 认证。    """    # 假设您需要根据用户名进行某些操作,例如权限检查    print(f"Authenticated user: {logged_username}")     query_users = db.query(User).all()    return query_users

关键点:

jwt_username: Annotated[Optional[str], Depends(jwt_logged_user)] 和 basic_username: Annotated[Optional[str], Depends(basic_logged_user)]:接收两个子认证依赖的结果,它们现在可以是 None。if jwt_username: return jwt_username:如果 JWT 认证成功(返回了用户名),则直接返回该用户名。if basic_username: return basic_username:如果 JWT 认证失败但 Basic Auth 成功,则返回 Basic Auth 的用户名。raise HTTPException(…):只有当所有子认证都失败时,才由 auth_user 统一抛出 401 异常。这样确保了只有在没有任何有效凭据的情况下才拒绝访问。headers={“WWW-Authenticate”: “Basic, Bearer”}:在 401 响应头中明确告知客户端支持的认证方式。

完整示例代码结构

为了更好地理解,以下是一个整合了上述修改的 FastAPI 应用片段:

from fastapi import APIRouter, Depends, HTTPException, statusfrom fastapi.security import HTTPBasic, HTTPBasicCredentials, OAuth2PasswordBearerfrom sqlalchemy.orm import Sessionfrom typing import Annotated, Optional, Unionimport secretsfrom jose import JWTError, jwt # 假设使用 jose 库# 假设这些是您的配置和模型class Settings:    SESSION_LOGIN_USER = "admin"    SESSION_LOGIN_PASS = "securepassword"    JWT_SECRET_KEY = "your-super-secret-jwt-key" # 替换为实际的密钥    JWT_ALGORITHM = "HS256"settings = Settings()# 假设 db_session 是您的数据库会话依赖# 假设 User 是您的 SQLAlchemy 用户模型class User: # 简化示例    def __init__(self, username):        self.username = username    def __repr__(self):        return f""class MockDBSession: # 模拟数据库会话    def query(self, model):        return self    def filter(self, *args, **kwargs):        return self    def first(self):        return User("testuser") # 模拟返回一个用户    def all(self):        return [User("user1"), User("user2")]def get_db():    db = MockDBSession()    try:        yield db    finally:        pass # 实际应用中会关闭会话db_session = Depends(get_db)# JWT 相关的工具类class Utils:    OAuth2_scheme = OAuth2PasswordBearer(tokenUrl="/token", auto_error=False)    SECRET_KEY = settings.JWT_SECRET_KEY    ALGORITHM = settings.JWT_ALGORITHM    def verify_token(self, token: str) -> dict:        try:            payload = jwt.decode(token, self.SECRET_KEY, algorithms=[self.ALGORITHM])            username: str = payload.get("sub")            if username is None:                raise JWTError("Invalid token payload: no subject")            return {"username": username}        except JWTError as e:            # 在这里可以记录错误,但不再抛出 HTTPException            raise e # 重新抛出 JWTError,由 jwt_logged_user 捕获        except Exception as e:            raise eutils = Utils()router = APIRouter()# --- 独立认证依赖项 ---# Basic Authsecurity = HTTPBasic(auto_error=False)def basic_logged_user(credentials: Annotated[Optional[HTTPBasicCredentials], Depends(security)]) -> Optional[str]:    if credentials is None:        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):        return None    return credentials.username# JWT Authdef jwt_logged_user(token: Annotated[Optional[str], Depends(utils.OAuth2_scheme)],                    db: Session = Depends(db_session)) -> Optional[str]:    if token is None:        return None    try:        payload = utils.verify_token(token)        username = payload.get("username")        if username is None:            return None        user = db.query(User).filter(User.username == username).first()        if user is None:            return None        return user.username    except JWTError:        return None    except Exception:        return None# --- 组合认证依赖项 ---def auth_user(jwt_username: Annotated[Optional[str], Depends(jwt_logged_user)],              basic_username: Annotated[Optional[str], Depends(basic_logged_user)]) -> str:    if jwt_username:        return jwt_username    if basic_username:        return basic_username    raise HTTPException(        status_code=status.

以上就是FastAPI 多种认证方式(任选其一)实现指南的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1382815.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月15日 00:34:04
下一篇 2025年12月15日 00:34:14

相关推荐

  • 为什么Golang的GC会突然卡顿 详解GC调优参数与平滑回收策略

    golang gc突然卡顿主要由对象分配速率过高、堆内存增长过快、stw阶段耗时增加及系统资源争抢引起,优化方法包括调整gogc、使用sync.pool减少分配频率等。具体而言:1. 高并发下频繁创建临时对象导致gc频繁触发;2. 堆增长过快引发gc滞后;3. stw阶段因堆大而延长停顿时间;4. …

    2025年12月15日 好文分享
    000
  • Go语言:如何正确初始化自定义基本类型

    本文将详细介绍Go语言中自定义基本类型(如基于int的自定义类型)的初始化方法。不同于make函数,这类自定义类型应像其底层基本类型一样直接进行初始化,包括声明时赋值和类型转换两种常见方式。文章将通过代码示例,清晰展示其用法,并解释make函数不适用于此场景的原因,帮助读者掌握Go语言中自定义类型的…

    2025年12月15日
    000
  • Go语言中如何向函数传递数组指针及其应用与限制

    本文深入探讨Go语言中向函数传递数组指针的方法。我们将详细介绍其语法、实现细节,并结合实际场景(如从磁盘加载数据)进行说明。重点阐述Go语言中数组大小作为类型一部分的特性,这如何限制了数组指针的通用性。同时,文章将对比数组指针与切片(slice)的优劣,并强调在多数情况下,切片是更灵活和推荐的选择。…

    2025年12月15日
    000
  • Go语言:理解与使用数组指针作为函数参数

    本文深入探讨了Go语言中如何将数组指针作为函数参数传递,并阐明了数组大小作为类型一部分的关键特性及其对函数签名的影响。我们将通过示例代码展示其用法,并对比分析了数组指针与更常用、更灵活的切片(slice)在参数传递上的异同,旨在帮助开发者理解Go语言中处理集合类型数据的最佳实践。 如何向函数传递数组…

    2025年12月15日
    000
  • Go语言中自定义整型(int)的初始化方法详解

    本文详细介绍了Go语言中自定义整型(如type Num int)的初始化方法。不同于内置复合类型,自定义基础类型应通过直接赋值或类型转换进行初始化,其方式与底层类型保持一致。文章将明确指出make函数不适用于此类初始化,并通过示例代码演示正确的初始化实践,帮助开发者理解Go语言的类型系统特性。 Go…

    2025年12月15日
    000
  • Go语言中数组指针的传递与使用:深入理解其特性与局限

    本文深入探讨Go语言中如何传递数组指针,包括其语法、在函数中接收和使用的方法。重点阐述了数组指针的一个核心局限:数组大小是其类型的一部分,导致函数签名必须与特定大小的数组精确匹配。文章对比了数组指针与切片(Slic++e)的适用场景,并强调了在Go语言中,切片通常是更灵活、更推荐的数据传递方式,同时…

    2025年12月15日
    000
  • Go 语言自定义整型类型初始化详解

    Go 语言中,自定义整型类型(如 type Num int)的初始化方法与其底层基本类型(如 int)相同。可以通过直接赋值或类型转换的方式进行初始化,例如 var myNum Num = 7 或 anotherNum := Num(42)。需要注意的是,Go 语言内置的 make 函数仅用于初始化…

    2025年12月15日
    000
  • Go语言中基于通道的并发注册中心设计模式

    本文探讨Go语言中如何利用通道(channel)实现并发安全的注册中心(Registry)或任务管理器,以解决共享状态的序列化访问问题。通过分析初始设计中面临的样板代码和错误处理复杂性,文章提出了一种更通用、可扩展的基于接口和单一请求通道的解决方案,并详细阐述了如何优雅地处理并发操作的返回值和错误,…

    2025年12月15日
    000
  • Go语言中传递数组指针:教程与最佳实践

    本文旨在讲解如何在Go语言中传递数组指针,并探讨使用数组指针与切片的差异。我们将通过示例代码展示如何声明、传递和使用数组指针,并分析其适用场景和潜在问题,帮助开发者更好地理解和运用这一特性。 在Go语言中,数组是一种固定长度的数据结构,而切片则提供了更灵活的动态数组功能。虽然通常推荐使用切片,但在某…

    2025年12月15日
    000
  • Go语言中函数参数传递:使用指向数组的指针

    本文介绍了在Go语言中如何将数组的指针作为参数传递给函数。虽然Go语言中切片更为常用,但了解数组指针的传递方式仍然具有一定的价值。本文将详细讲解数组指针的声明、传递以及在函数内部的使用方法,并强调使用数组指针时需要注意的问题。 数组指针的声明和传递 在Go语言中,数组的大小是数组类型的一部分。这意味…

    2025年12月15日
    000
  • Go 语言中 Nil 指针比较的正确处理方式

    Go 语言中 Nil 指针比较的机制和处理方法至关重要。Nil 指针解引用会导致程序崩溃,因此理解其背后的原理并掌握避免此类错误的技巧是每个 Go 开发者必备的技能。本文将深入探讨 Nil 指针的特性,并提供实用指南和示例代码,帮助开发者编写更健壮的 Go 程序。 Nil 指针解引用错误 在 Go …

    2025年12月15日
    000
  • Go 语言中 Nil 指针比较的处理与避免

    第一段引用上面的摘要: 本文旨在深入探讨 Go 语言中 nil 指针比较时可能出现的问题,并提供避免运行时错误的实用方法。我们将分析 nil 指针解引用的错误原因,并提供通过显式 nil 检查来确保代码健壮性的策略。通过本文,开发者可以更好地理解 Go 语言的 nil 指针处理机制,编写出更安全可靠…

    2025年12月15日
    000
  • Go 中 nil 指针比较:避免运行时错误

    本文旨在深入探讨 Go 语言中 nil 指针比较的问题,解释为何直接比较 nil 指针会导致运行时错误,并提供避免此类错误的有效方法。我们将通过示例代码和详细分析,帮助开发者理解 nil 指针的本质,并掌握在 Go 语言中安全处理指针的最佳实践。 在 Go 语言中,尝试访问 nil 指针的成员会导致…

    2025年12月15日
    000
  • 在 Go 中整合 C 和 Python 代码实现 Markdown 解析

    本文旨在指导开发者如何在 Go 语言中利用 CGO 和 go-python 整合 C 和 Python 代码,以实现 Markdown 文本到 HTML 的转换。文章将重点介绍使用 CGO 封装 C 语言编写的 Markdown 解析库,并简要提及 go-python 的使用场景,同时推荐使用纯 G…

    2025年12月15日
    000
  • Golang模块缓存机制如何工作 解析Golang本地缓存的运行原理

    golang模块缓存是go工具链用于存储已下载依赖模块的本地目录,以提升构建效率。其作用包括避免重复下载相同版本模块、校验模块完整性并支持快速复用;默认路径为$gopath/pkg/mod;每个模块按模块路径和版本号组织为独立目录,且缓存内容不可变;可通过go clean -modcache查看或清…

    2025年12月15日 好文分享
    000
  • 如何通过反射获取Golang方法的注释 分析AST与反射的结合使用

    要通过反射获取 golang 方法的注释,需解析源码 ast 并结合反射 api。1. 使用 go/parser 解析源代码为 ast;2. 遍历 ast 查找 *ast.funcdecl 节点以定位目标方法;3. 从 doc 字段提取注释;4. 利用 reflect.typeof 和 method…

    2025年12月15日 好文分享
    000
  • Golang跨语言调用:解决CGO内存管理问题

    c++go内存管理需注意跨语言内存分配与释放。1. go分配,c使用:优先在go侧分配内存并传递指针给c/c++,如用c.gobytes将c内存复制到go slice后释放c内存;2. c分配,go使用后释放:使用defer确保释放c分配的内存,如defer c.free_string(cresul…

    2025年12月15日 好文分享
    000
  • Golang程序启动慢 如何减少初始化时间

    优化golang程序启动慢的核心方法是延迟非必要逻辑执行和优化早期加载内容,具体包括:1. 使用延迟初始化(如sync.once)将非关键组件的初始化推迟到首次使用时;2. 避免在init函数中执行耗时操作,将复杂初始化移至main函数或统一流程中;3. 对无依赖关系的模块进行并行初始化,利用gor…

    2025年12月15日 好文分享
    000
  • Golang的select语句如何处理多路channel 演示非阻塞通信的实现方式

    golang的select语句能同时监听多个channel并随机选择准备好的分支执行,从而实现非阻塞通信。解决方案:1. select语句通过case监听多个channel操作,哪个channel先准备好就执行哪个;2. 使用default分支实现非阻塞,在所有channel未准备好时立即执行默认操…

    2025年12月15日 好文分享
    000
  • 如何在云服务器上快速部署Golang环境 分享一键脚本与优化建议

    选择合适的云服务器配置需考虑cpu、内存、存储类型和网络带宽。1. cpu密集型应用应选高主频配置;2. 并发需求大时需足够内存;3. ssd硬盘提升i/o性能;4. 充足带宽保障数据传输。初期可选适中配置,后续根据实际运行情况调整,如cpu占用过高则升级cpu。 在云服务器上快速部署Golang环…

    2025年12月15日 好文分享
    000

发表回复

登录后才能评论
关注微信