Skip to main content
Version: 3.17

openid-connect

描述#

openid-connect 插件支持与 OpenID Connect (OIDC) 身份提供商集成,例如 Keycloak、Auth0、Microsoft Entra ID、Google、Okta 等。它允许 APISIX 在允许或拒绝客户端访问受保护的上游资源之前,先对客户端进行身份验证并从身份提供商获取相关信息。

属性#

名称类型必选项默认值有效值描述
client_idstringOAuth 客户端 ID。
client_secretstringOAuth 客户端密钥。
discoverystringOpenID 提供商的 well-known 发现文档 URL,包含 OP API 端点列表。插件可直接使用发现文档中的端点。你也可以单独配置这些端点,单独配置的值优先于发现文档中提供的端点。
scopestringopenid与认证用户相关信息对应的 OIDC 范围,也称为 claims。用于授权具有适当权限的用户。默认值为 openid,这是 OIDC 返回唯一标识认证用户的 sub claim 所需的范围。可以附加额外的范围并以空格分隔,例如 openid email profile
required_scopesarray[string]访问令牌中必须存在的范围。在 bearer_onlytrue 时与 introspection 端点结合使用。如果缺少任何必需范围,插件将以 403 forbidden 错误拒绝请求。
realmstringapisix由于无效 bearer token 导致 401 未授权请求时,WWW-Authenticate 响应头中的 Realm 值。
bearer_onlybooleanfalse如果为 true,则严格要求请求中携带 bearer 访问令牌进行身份验证。
logout_pathstring/logout触发注销的路径。
post_logout_redirect_uristringlogout_path 收到注销请求后重定向用户的 URL。
redirect_uristring与 OpenID 提供商完成身份验证后的重定向 URI。注意,重定向 URI 不应与请求 URI 相同,而应为请求 URI 的子路径。例如,如果路由的 uri/api/v1/*,则 redirect_uri 可配置为 /api/v1/redirect。如果未配置 redirect_uri,APISIX 将在请求 URI 后追加 /.apisix/redirect 作为 redirect_uri 的值。
timeoutinteger3[1,...]请求超时时间,单位为秒。
ssl_verifybooleantrue如果为 true,则验证 OpenID 提供商的 SSL 证书。注意:该属性的默认值在 APISIX 3.16.0 中从 false 更改为 true,这是一个破坏性变更。如果你从早期版本升级,请确保你的 OpenID 提供商 SSL 证书有效,或显式将其设置为 false 以保持之前的行为。
introspection_endpointstringOpenID 提供商用于内省访问令牌的令牌内省端点 URL。如果未设置,则使用 well-known 发现文档中提供的内省端点作为备选项
introspection_endpoint_auth_methodstringclient_secret_basic令牌内省端点的认证方法。值应为 well-known 发现文档中 introspection_endpoint_auth_methods_supported 授权服务器元数据指定的认证方法之一,例如 client_secret_basicclient_secret_postprivate_key_jwtclient_secret_jwt
token_endpoint_auth_methodstringclient_secret_basic令牌端点的认证方法。值应为 well-known 发现文档中 token_endpoint_auth_methods_supported 授权服务器元数据指定的认证方法之一,例如 client_secret_basicclient_secret_postprivate_key_jwtclient_secret_jwt。如果配置的方法不受支持,则回退到 token_endpoint_auth_methods_supported 数组中的第一个方法。
public_keystring使用非对称算法时用于验证 JWT 签名的公钥。提供此值进行令牌验证将跳过客户端凭证流中的令牌内省。可以以 -----BEGIN PUBLIC KEY-----\n……\n-----END PUBLIC KEY----- 格式传递公钥。
use_jwksbooleanfalse如果为 true 且未设置 public_key,则使用 JWKS 验证 JWT 签名并跳过客户端凭证流中的令牌内省。JWKS 端点从发现文档中解析。
use_pkcebooleanfalse如果为 true,则按照 RFC 7636 定义,在授权码流程中使用 PKCE(Proof Key for Code Exchange)。
token_signing_alg_values_expectedstring用于签署 JWT 的算法,例如 RS256
set_access_token_headerbooleantrue如果为 true,则在请求标头中设置访问令牌。默认情况下,使用 X-Access-Token 标头。
access_token_in_authorization_headerbooleanfalse如果为 true 并且 set_access_token_header 也为 true,则在 Authorization 标头中设置访问令牌。
set_id_token_headerbooleantrue如果为 true 并且 ID 令牌可用,则在 X-ID-Token 请求标头中设置值。
set_userinfo_headerbooleantrue如果为 true 并且用户信息数据可用,则在 X-Userinfo 请求标头中设置值。
set_refresh_token_headerbooleanfalse如果为 true 并且刷新令牌可用,则在 X-Refresh-Token 请求标头中设置值。
sessionobjectbearer_onlyfalse 且插件使用 Authorization Code 流程时使用的 Session 配置。
session.secretstring16 个字符以上bearer_onlyfalse 时,用于 session 加密和 HMAC 运算的密钥。
session.cookie_namestring会话 Cookie 名称。作为 cookie_name 透传到 lua-resty-session 4.x。
session.cookie_pathstringCookie 路径范围。作为 cookie_path 透传到 lua-resty-session。
session.cookie_domainstringCookie 域范围。作为 cookie_domain 透传到 lua-resty-session。
session.cookie_secureboolean若为 true,则设置 Cookie 的 Secure 属性。作为 cookie_secure 透传到 lua-resty-session。
session.cookie_http_onlyboolean若为 true,则设置 Cookie 的 HttpOnly 属性。作为 cookie_http_only 透传到 lua-resty-session。
session.cookie_same_sitestring["Strict", "Lax", "None", "Default"]Cookie 的 SameSite 属性。作为 cookie_same_site 透传到 lua-resty-session。
session.idling_timeoutinteger会话空闲超时时间(秒)。作为 idling_timeout 透传到 lua-resty-session。
session.rolling_timeoutinteger会话滚动超时时间(秒)。作为 rolling_timeout 透传到 lua-resty-session。
session.absolute_timeoutinteger会话绝对生存时间(秒)。作为 absolute_timeout 透传到 lua-resty-session。
session.cookie.lifetimeinteger已弃用。当未设置 session.absolute_timeout 时,运行时会将该值映射到 session.absolute_timeout。请改用 session.absolute_timeout
session.storagestringcookie["cookie", "redis"]会话存储方式。
session.redisobjectstorageredis 时的 Redis 配置。
session.redis.hoststring127.0.0.1Redis 主机。
session.redis.portinteger6379Redis 端口。
session.redis.usernamestringRedis 用户名。
session.redis.passwordstringRedis 密码。
session.redis.databaseinteger0Redis 数据库索引。
session.redis.prefixstringsessionsRedis 键前缀。
session.redis.sslbooleanfalse为 Redis 连接启用 SSL。
session.redis.ssl_verifybooleantrue验证 Redis 连接的 SSL 证书。
session.redis.server_namestring用于 SNI 的 Redis 服务器名称。
session.redis.connect_timeoutinteger1000连接超时时间,单位为毫秒。
session.redis.send_timeoutinteger1000发送超时时间,单位为毫秒。
session.redis.read_timeoutinteger1000读取超时时间,单位为毫秒。
session.redis.keepalive_timeoutinteger10000保活超时时间,单位为毫秒。
session_contentsobject会话内容配置。如果未配置,所有数据将存储在会话中。
session_contents.access_tokenboolean如果为 true,则在会话中存储访问令牌。
session_contents.id_tokenboolean如果为 true,则在会话中存储 ID 令牌。
session_contents.enc_id_tokenboolean如果为 true,则在会话中存储加密的 ID 令牌。
session_contents.userboolean如果为 true,则在会话中存储用户信息。
unauth_actionstringauth["auth", "deny", "pass"]未认证请求的处理方式。设置为 auth 时,重定向到 OpenID 提供商的认证端点。设置为 pass 时,允许请求不经认证通过。设置为 deny 时,返回 401 未认证响应而不启动授权码授权流程。
proxy_optsobjectOpenID 提供商所在代理服务器的配置。
proxy_opts.http_proxystringHTTP 请求的代理服务器地址,例如 http://<proxy_host>:<proxy_port>
proxy_opts.https_proxystringHTTPS 请求的代理服务器地址,例如 http://<proxy_host>:<proxy_port>
proxy_opts.http_proxy_authorizationstringBasic [base64 username:password]http_proxy 一起使用的默认 Proxy-Authorization 头值。可以用自定义 Proxy-Authorization 请求头覆盖。
proxy_opts.https_proxy_authorizationstringBasic [base64 username:password]https_proxy 一起使用的默认 Proxy-Authorization 头值。由于 HTTPS 连接时已完成授权,不能用自定义 Proxy-Authorization 请求头覆盖。
proxy_opts.no_proxystring不需要代理的主机列表,以逗号分隔。
authorization_paramsobject发送到授权端点请求中的额外参数。
client_rsa_private_keystring用于向 OP 签署 JWT 进行身份验证的客户端 RSA 私钥。当 token_endpoint_auth_methodprivate_key_jwt 时必填。
client_rsa_private_key_idstring用于计算已签名 JWT 的客户端 RSA 私钥 ID。当 token_endpoint_auth_methodprivate_key_jwt 时可选。
client_jwt_assertion_expires_ininteger60向 OP 进行身份验证的已签名 JWT 的有效期,单位为秒。在 token_endpoint_auth_methodprivate_key_jwtclient_secret_jwt 时使用。
renew_access_token_on_expirybooleantrue如果为 true,则在访问令牌过期或刷新令牌可用时尝试静默续期。如果令牌续期失败,则重定向用户重新认证。
access_token_expires_ininteger当令牌端点响应中没有 expires_in 属性时,访问令牌的有效期,单位为秒。
refresh_session_intervalinteger无需重新认证即可刷新用户 ID 令牌的时间间隔,单位为秒。未设置时不检查网关向客户端签发的会话的过期时间。
iat_slackinteger120ID 令牌 iat claim 时钟偏差容忍度,单位为秒。
accept_none_algbooleanfalse如果 OpenID 提供商不对其 ID 令牌进行签名(例如签名算法设置为 none),则设置为 true。
accept_unsupported_algbooleantrue如果为 true,则忽略 ID 令牌签名以接受不支持的签名算法。
access_token_expires_leewayinteger0访问令牌续期的过期宽限时间,单位为秒。当设置为大于 0 的值时,令牌续期将在令牌过期前该时间量时进行。这可避免在访问令牌刚到达资源服务器时过期的错误。
force_reauthorizebooleanfalse如果为 true,即使令牌已缓存也执行授权流程。
use_noncebooleanfalse如果为 true,则在授权请求中启用 nonce 参数。
revoke_tokens_on_logoutbooleanfalse如果为 true,则在注销时通知授权服务器之前获取的刷新令牌或访问令牌不再需要。
jwk_expires_ininteger86400JWK 缓存的过期时间,单位为秒。
jwt_verification_cache_ignorebooleanfalse如果为 true,则强制重新验证 bearer 令牌并忽略任何现有的缓存验证结果。
cache_segmentstring缓存段的可选名称,用于分离和区分令牌内省或 JWT 验证使用的缓存。
introspection_intervalinteger0缓存和内省的访问令牌的 TTL,单位为秒。默认值为 0,表示不使用此选项,插件默认使用 introspection_expiry_claim 定义的过期 claim 传递的 TTL。如果 introspection_interval 大于 0 且小于 introspection_expiry_claim 定义的过期 claim 传递的 TTL,则使用 introspection_interval
introspection_expiry_claimstringexp过期 claim 的名称,用于控制缓存和内省的访问令牌的 TTL。
introspection_addon_headersarray[string]用于向内省 HTTP 请求追加额外头值。如果指定的头在原始请求中不存在,则不会追加该值。
claim_validatorobjectJWT claim 验证配置。
claim_validator.issuer.valid_issuersarray[string]受信任的 JWT 颁发者数组。如果未配置,将使用发现端点返回的颁发者。如果两者均不可用,则不验证颁发者。
claim_validator.audienceobject受众 claim 验证配置。
claim_validator.audience.claimstringaud包含受众的 claim 名称。
claim_validator.audience.requiredbooleanfalse如果为 true,则受众 claim 为必填项,claim 名称为 claim 中定义的名称。
claim_validator.audience.match_with_client_idbooleanfalse如果为 true,则要求受众与客户端 ID 匹配。如果受众是字符串,则必须与客户端 ID 完全匹配。如果受众是字符串数组,则至少一个值必须与客户端 ID 匹配。如果未找到匹配,将收到 mismatched audience 错误。OpenID Connect 规范规定了此要求,以确保令牌是为特定客户端颁发的。
claim_schemaobjectOIDC 响应 claim 的 JSON schema。示例:{"type":"object","properties":{"access_token":{"type":"string"}},"required":["access_token"]} - 验证响应包含必填的字符串字段 access_token

注意:schema 中还定义了 encrypt_fields = {"client_secret", "client_rsa_private_key"},这意味着这些字段将在 etcd 中加密存储。详见加密存储字段

此外,你可以使用环境变量或 APISIX Secret 来存储和引用插件属性。APISIX 目前支持两种存储密钥的方式——环境变量和 HashiCorp Vault

例如,使用以下命令设置环境变量:

export KEYCLOAK_CLIENT_SECRET=abc

并在插件配置中引用:

"client_secret": "$ENV://KEYCLOAK_CLIENT_SECRET"

示例#

以下示例展示了如何针对不同场景配置 openid-connect 插件。

note

你可以使用以下命令从 config.yaml 中获取 admin_key 并保存到环境变量:

admin_key=$(yq '.deployment.admin.admin_key[0].key' conf/config.yaml | sed 's/"//g')

授权码流程#

授权码流程在 RFC 6749 第 4.1 节中定义。它涉及将临时授权码换取访问令牌,通常由机密客户端和公共客户端使用。

下图展示了实现授权码流程时不同实体之间的交互:

当传入请求的头中或合适的会话 Cookie 中不包含访问令牌时,插件作为依赖方重定向到授权服务器以继续授权码流程。

认证成功后,插件将令牌保存在会话 Cookie 中,后续请求将使用 Cookie 中存储的令牌。

以下示例创建一个路由,并配置 openid-connect 插件以使用 Keycloak 作为身份提供商的授权码流程:

详见实现授权码授权,获取使用 openid-connect 插件与 Keycloak 集成并使用授权码流程的完整示例。

PKCE (Proof Key for Code Exchange)#

PKCE 在 RFC 7636 中定义。PKCE 通过添加代码挑战和验证器来增强授权码流程,防止授权码截取攻击。

下图展示了实现带 PKCE 的授权码流程时不同实体之间的交互:

要使用 PKCE,在插件配置中将 use_pkce 设置为 true。同时确保已配置 IdP 客户端以使用 PKCE。

详见实现授权码授权,获取使用 openid-connect 插件与 Keycloak 集成并使用带 PKCE 的授权码流程的示例。

客户端凭证流程#

客户端凭证流程在 RFC 6749 第 4.4 节中定义。它涉及客户端使用自身凭证请求访问令牌以访问受保护资源,通常用于机器间认证,不代表特定用户。

下图展示了实现客户端凭证流程时不同实体之间的交互:

详见实现客户端凭证授权,获取使用 openid-connect 插件与 Keycloak 集成并使用客户端凭证流程的示例。

内省流程#

内省流程在 RFC 7662 中定义。它涉及通过查询授权服务器的内省端点来验证访问令牌的有效性和详细信息。

在此流程中,当客户端向资源服务器提供访问令牌时,资源服务器向授权服务器的内省端点发送请求,如果令牌有效,端点将返回令牌详细信息,包括令牌过期时间、关联范围以及令牌所属的用户或客户端等信息。

下图展示了实现带令牌内省的授权码流程时不同实体之间的交互:

详见实现客户端凭证授权,获取使用 openid-connect 插件与 Keycloak 集成并使用带令牌内省的客户端凭证流程的示例。

密码流程#

密码流程在 RFC 6749 第 4.3 节中定义。它专为受信任的应用程序设计,允许它们直接使用用户的用户名和密码获取访问令牌。在此授权类型中,客户端应用程序将用户凭证连同其自身的客户端 ID 和密钥一起发送到授权服务器,授权服务器对用户进行身份验证,如果有效则颁发访问令牌。

尽管效率较高,但此流程仅适用于高度受信任的第一方应用程序,因为它要求应用程序直接处理敏感的用户凭证,如果在第三方场景中使用会带来重大安全风险。

下图展示了实现密码流程时不同实体之间的交互:

详见实现密码授权,获取使用 openid-connect 插件与 Keycloak 集成并使用密码流程的示例。

刷新令牌授权#

刷新令牌授权在 RFC 6749 第 6 节中定义。它使客户端无需用户重新认证,即可使用之前颁发的刷新令牌请求新的访问令牌。此流程通常在访问令牌过期时使用,允许客户端在无需用户干预的情况下保持对资源的持续访问。刷新令牌与访问令牌一起在某些 OAuth 流程中颁发,其生命周期和安全要求取决于授权服务器的配置。

下图展示了实现带刷新令牌的密码流程时不同实体之间的交互:

详见刷新令牌,获取使用 openid-connect 插件与 Keycloak 集成并使用带令牌刷新的密码流程的示例。

用户信息#

OpenID Connect (OIDC) 中的 UserInfo 端点在 OpenID Connect Core 1.0 第 5.3 节中定义。它使客户端能够通过提供有效的访问令牌来检索已认证用户的额外 claim。此端点对于获取用户个人资料信息(如姓名、电子邮件和其他属性)特别有用,这些信息在用户认证后可通过该端点获取。UserInfo 端点返回的数据取决于访问令牌的范围以及授权服务器配置的 claim。

下图展示了在 APISIX 验证用户信息时,不同实体之间的交互过程。

set_userinfo_headertrue(默认值)时,插件在 X-Userinfo 请求头中设置用户信息数据,上游服务可使用该数据进行进一步处理。

故障排除#

本节涵盖使用此插件时常见的一些问题,以帮助你进行故障排查。

APISIX 无法连接到 OpenID 提供商#

如果 APISIX 无法解析或连接到 OpenID 提供商,请检查配置文件 config.yaml 中的 DNS 设置并根据需要进行修改。

未找到会话状态#

如果在使用授权码流程时,日志中出现 500 internal server error 和以下消息,可能有多种原因。

the error request to the redirect_uri path, but there's no session state found

1. 重定向 URI 配置错误#

一个常见的配置错误是将 redirect_uri 配置为与路由 URI 相同。当用户发起访问受保护资源的请求时,请求直接到达重定向 URI,但请求中没有会话 Cookie,导致未找到会话状态的错误。

要正确配置重定向 URI,确保 redirect_uri 与配置了插件的路由匹配,但不完全相同。例如,正确的配置是将路由的 uri 配置为 /api/v1/*,将 redirect_uri 的路径部分配置为 /api/v1/redirect

同时确保 redirect_uri 包含协议,例如 httphttps

2. 缺少会话密钥#

如果你以独立模式部署 APISIX,请确保配置了 session.secret

用户会话以 Cookie 形式存储在浏览器中,并使用会话密钥加密。如果未通过 session.secret 属性配置密钥,则会自动生成密钥并保存到 etcd。但在独立模式下,etcd 不再是配置中心。因此,你应在 YAML 配置中心 apisix.yaml 中为此插件显式配置 session.secret

3. Cookie 未发送或缺失#

检查 SameSite Cookie 属性是否正确设置(即你的应用程序是否需要跨站发送 Cookie),以判断这是否是阻止 Cookie 保存到浏览器 Cookie 存储或从浏览器发送的因素。

4. 上游发送的头太大#

如果你在 APISIX 前面使用 NGINX 代理客户端流量,请检查 NGINX 的 error.log 中是否出现以下错误:

upstream sent too big header while reading response header from upstream

如果是,请尝试将 proxy_buffersproxy_buffer_sizeproxy_busy_buffers_size 调整为更大的值。

另一个选项是配置 session_contents 属性来调整存储在会话中的数据。例如,可以将 session_contents.access_token 设置为 true

5. 客户端密钥无效#

验证 client_secret 是否有效且正确。无效的 client_secret 会导致认证失败,且不会返回令牌并存储在会话中。

6. PKCE IdP 配置#

如果你在授权码流程中启用了 PKCE,请确保你已配置 IdP 客户端以使用 PKCE。例如,在 Keycloak 中,你应在客户端的高级设置中配置 PKCE 挑战方法: