ChatGPT解决这个技术问题 Extra ChatGPT

为什么 OAuth v2 既有访问令牌又有刷新令牌?

OAuth 2.0 协议草案的第 4.2 节指出,授权服务器可以返回 access_token(用于通过资源验证自己)和 refresh_token,后者纯粹用于创建新的 access_token

https://www.rfc-editor.org/rfc/rfc6749#section-4.2

为什么两者都有?为什么不让 access_tokenrefresh_token 一样长而没有 refresh_token


s
sleske

Catchdave 提供的讨论链接有另一个由 Dick Hardt 制作的 valid point (original, dead link),我认为除了上面写的内容之外,这里值得一提:

我对刷新令牌的回忆是为了安全和撤销。 <...> 撤销:如果访问令牌是自包含的,则可以通过不发布新的访问令牌来撤销授权。资源不需要查询授权服务器来查看访问令牌是否有效。这简化了访问令牌验证,并且更容易扩展和支持多个授权服务器。当访问令牌有效但授权被撤销时,存在一个时间窗口。

实际上,在资源服务器和授权服务器是同一个实体的情况下,并且用户和它们之间的连接(通常)同样安全,将刷新令牌与访问令牌分开没有多大意义。

尽管如引用中所述,刷新令牌的另一个作用是确保用户可以随时撤销访问令牌(例如,通过其配置文件中的 Web 界面),同时保持系统可扩展性.

通常,令牌可以是指向服务器数据库中特定记录的随机标识符,也可以是它们本身包含所有信息(当然,必须对这些信息进行签名,例如使用 MAC)。

具有长期访问令牌的系统应该如何工作

服务器允许客户端通过发出令牌来访问预定义范围内的用户数据。由于我们想保持令牌可撤销,我们必须将令牌连同设置或取消设置的“已撤销”标志一起存储在数据库中(否则,您将如何使用自包含令牌来做到这一点?)数据库可以包含多达 { 1}记录。然后,每个 API 请求都必须访问数据库。尽管执行 O(1) 对此类数据库进行查询非常简单,但单点故障本身会对系统的可伸缩性和性能产生负面影响。

具有长期刷新令牌和短期访问令牌的系统应该如何工作

这里我们发出两个密钥:随机刷新令牌与数据库中的相应记录,以及签名的自包含访问令牌,其中包含过期时间戳字段。

由于访问令牌是自包含的,我们根本不需要访问数据库来检查其有效性。我们所要做的就是解码令牌并验证签名和时间戳。

尽管如此,我们仍然要保留刷新令牌的数据库,但是对这个数据库的请求次数通常由访问令牌的生命周期定义(生命周期越长,访问率越低)。

为了撤销特定用户对 Client 的访问权限,我们应该将相应的刷新令牌标记为“已撤销”(或完全删除)并停止发布新的访问令牌。很明显,虽然有一个窗口在此期间刷新令牌已被撤销,但其访问令牌可能仍然有效。

权衡取舍

刷新令牌部分消除了 Access Token 数据库的 SPoF(单点故障),但它们也有一些明显的缺点。

窗户”。事件“用户撤销访问”和“访问被保证被撤销”之间的时间范围。客户端逻辑的复杂性。没有刷新令牌 发送带有访问令牌的 API 请求 如果访问令牌无效,失败并要求用户使用刷新令牌重新进行身份验证 发送带有访问令牌的 API 请求 如果访问令牌无效,如果刷新请求通过,请尝试使用刷新令牌更新它,更新访问令牌并重新发送初始 API 请求 如果刷新请求失败,要求用户重新认证

我希望这个答案确实有意义,并帮助某人做出更深思熟虑的决定。我还想指出,一些著名的 OAuth2 提供者,包括 github 和foursquare 采用了没有刷新令牌的协议,并且似乎对此很满意。


@RomannImankulov如果我理解正确刷新令牌,我们可以保存到数据库中并在我们想要撤销访问权限的任何时候删除它们,那么为什么不自己保存访问令牌呢?
@kosnkov 我的帖子的简短版本是,如果您将访问令牌保存在数据库中,您会在每次请求 API 时访问数据库(在您的特定情况下这可能是也可能不是问题)。如果您保存刷新令牌并保持访问令牌“自包含”,则仅在客户端决定刷新访问令牌时才访问数据库。
就我个人而言,我不喜欢这种不访问数据库来获得性能的方法,如果它会危及安全性(即使只是在窗口的时间跨度内)。如有必要,我们应该能够立即撤销 access_token,因为我们几乎总是在处理敏感的用户信息(否则我们可能一开始就不会使用 OAuth)。我想知道 Facebook 和 Google 等大公司使用哪种方法。
c
catchdave

刷新令牌的想法是,如果访问令牌被泄露,因为它是短暂的,攻击者有一个有限的窗口来滥用它。

刷新令牌,如果被泄露,将毫无用处,因为攻击者除了刷新令牌之外还需要客户端 ID 和密码才能获得访问令牌。

话虽如此,因为对授权服务器和资源服务器的每次调用都是通过 SSL 完成的——包括当他们请求访问/刷新令牌时的原始客户端 ID 和秘密——我不确定访问令牌是如何的“比长期刷新令牌和客户端ID/秘密组合更容易妥协”。

这当然与您不控制授权和资源服务器的实现不同。

这是一个讨论刷新令牌使用的好帖子:OAuth Archives

上面的引用,谈论刷新令牌的安全目的:

刷新令牌...减轻长期 access_token 泄漏的风险(在不安全的资源服务器上的日志文件中查询参数、测试版或编码不佳的资源服务器应用程序、将 access_token 放在非 https 站点上的 JS SDK 客户端饼干等)


Catchdave 是对的,但我想我会补充一点,自从他最初的回复以来,事情已经发生了变化。 SSL 的使用现在是可选的(当 catchdave 回答时,这可能仍在争论中)。例如,MAC 令牌(目前正在开发中)提供了使用私钥对请求进行签名的能力,因此不需要 SSL。因此,刷新令牌变得非常重要,因为您希望拥有短暂的 mac 令牌。
“如果遭到破坏,刷新令牌将毫无用处,因为除了刷新令牌之外,攻击者还需要客户端 ID 和密码才能获得访问令牌。”但是客户端 ID 和密码也存储在设备中,不是吗?因此,有权访问设备的攻击者可以获得它们。那么为什么?在这里,github.com/auth0/lock/wiki/Using-a-Refresh-Token,写的是丢失刷新令牌意味着,他可以请求任意数量的身份验证令牌,可能不在 google 场景中,但是如果我正在实现自己的 oauth2 服务器呢?
“除了刷新令牌之外,攻击者还需要客户端 ID 和密码才能获得访问令牌”:那么使用刷新令牌和简单地重新登录有什么区别?
第三方可以使用刷新令牌来更新访问令牌,而无需了解用户凭据。
@KevinWheeler 不,客户端 ID 和密码是 OAuth 客户端的凭据,而不是用户的凭据。在谈论 OAuth 时,“客户端”通常是与授权或资源 API 服务器(例如 facebook auth 提供程序)交互的服务器(例如 stackoverflow Web 服务器)。用户的凭据仅在用户和 OAuth API 服务器之间传递,客户端永远不知道。客户端密码仅从客户端传递到 OAuth API 服务器,用户永远不知道。
A
Andrew Tobilko

尽管上面给出了所有很好的答案,但我作为一名安全硕士生和程序员,之前在 eBay 工作过,当我研究买家保护和欺诈时,我可以说分离访问令牌和刷新令牌在骚扰频繁使用用户名的用户之间取得了最佳平衡/password 输入并保持授权以撤销对潜在滥用服务的访问权限。

想想这样的场景。您发出一个 3600 秒的访问令牌的用户,并且刷新令牌的时间要长得多一天。

用户是一个很好的用户,他在家并且在他的 iPhone 上打开/关闭您的网站购物和搜索。他的 IP 地址不会改变,并且在您的服务器上的负载非常低。就像每分钟 3-5 个页面请求一样。当他在访问令牌上的 3600 秒结束时,他需要一个带有刷新令牌的新令牌。我们在服务器端检查他的活动历史和 IP 地址,认为他是一个人并且表现自己。我们授予他新的访问令牌以继续使用我们的服务。用户无需再次输入用户名/密码,直到他达到刷新令牌本身的一天生命周期。用户是粗心的用户。他住在美国纽约,他的病毒程序被关闭,并在波兰被黑客入侵。当黑客获得访问令牌和刷新令牌时,他试图冒充用户并使用我们的服务。但是在短期访问令牌过期后,当黑客尝试刷新访问令牌时,我们在服务器上注意到用户行为历史中的 IP 发生了巨大变化(嘿,这家伙在美国登录,现在在波兰刷新访问仅 3600 秒后 ???)。我们终止刷新过程,使刷新令牌本身无效并提示再次输入用户名/密码。该用户是恶意用户。他打算通过使用机器人每分钟调用 1000 次我们的 API 来滥用我们的服务。他可以很好地做到这一点,直到 3600 秒后,当他尝试刷新访问令牌时,我们注意到他的行为并认为他可能不是人类。我们拒绝并终止刷新过程,并要求他再次输入用户名/密码。这可能会破坏他的机器人的自动流程。至少让他不舒服。

当我们尝试平衡我们的工作、用户体验和令牌被盗的潜在风险时,您可以看到刷新令牌的效果非常好。服务器端的看门狗可以检查IP变化,api调用频率,以确定用户是否为好用户。

换句话说,您还可以尝试通过在每个 api 调用上实施基本 IP 看门狗或任何其他措施来限制被盗令牌/滥用服务的损害控制。但这很昂贵,因为您必须读取和写入有关用户的记录,并且会减慢服务器的响应速度。


这些是一些很棒的政策和想法,但是我在您的答案中看不到任何本质上需要使用刷新令牌的内容。只需访问令牌即可实现所有这些功能。
@Evert,同时使用访问令牌和刷新令牌的好处之一是访问令牌可以是短暂的,因此在不检查最初发布它们的服务器的情况下无条件地信任它们并不是太大的安全妥协。这可以让您扩展您的基础架构,以便它的非关键部分可以信任存储在(签名)令牌中的信息,而无需直接访问用户的帐户信息。
@Avi Cherry - 是的,访问令牌可以是短暂的,如果用户仍然被认为是有效的,它也可以被刷新。不需要刷新令牌即可。
我相信这个答案假定我们从不希望资源服务器自己进行高级访问控制(例如,针对各种数据库检查 IP 活动等),而是他们只能依靠完全隔离地验证访问令牌。虽然这在规模上可能很明显(出于性能原因),但鉴于其他帖子和评论中的混乱,这对这里的每个人来说显然并不明显。这是一篇包含很好信息的好帖子,但我觉得它大大错过了原始问题的重点。我建议至少明确上述假设。
@RickJolly关于“访问令牌可以是短暂的,如果用户仍然被认为是有效的,它也可以被刷新” - 我认为回答者得到的是客户可能在 ebay 上浏览 - 比如说鞋子 - 24 小时内 3 或 4 次。登录必须全天保持“有效”,否则用户必须继续登录会感到非常沮丧。如果他们在移动设备上,那么如果用户在该浏览器选项卡中未处于活动状态,则不会“刷新”任何内容.但他获取成本较高的访问令牌只需要刷新 3 或 4 次。
P
Pang

这些答案都没有解决刷新令牌存在的核心原因。显然,您始终可以通过将客户端凭据发送到身份验证服务器来获得新的访问令牌/刷新令牌对 - 这就是您首先获取它们的方式。

因此,刷新令牌的唯一目的是限制使用通过线路发送到身份验证服务的客户端凭据。访问令牌的 TTL 越短,必须使用客户端凭据来获取新访问令牌的次数就越多,因此攻击者破坏客户端凭据的机会就越大(尽管这可能非常困难无论如何,如果使用非对称加密来发送它们)。因此,如果您有一次性刷新令牌,则可以在不影响客户端凭据的情况下将访问令牌的 TTL 设置为任意小。


这很有趣,因为在 Google 的情况下,当您请求刷新令牌时,您还发送了客户端 ID 和客户端密码。所以不管怎样,你每小时都在妥协。
“唯一目的” - 不洗。使 access-token 的 TTL 与想象中的 refresh-token 的 TTL 相同。
由于客户端凭据与刷新令牌一起发送的标准 requires,因此此答案的前提是错误的。 “因为刷新令牌通常是用于请求额外访问令牌的持久凭证……客户端必须通过授权服务器进行身份验证。”另请参阅@Rots 的评论。
A)我认为您混淆了客户端机密和用户机密。客户端机密永远不会从用户设备发送,只会从访问后端应用程序发送到数据提供后端应用程序。 B) 允许为公共客户端(不能保密的客户端,例如本机或 JavaScript 应用程序)授予密码的 oAuth 服务器也将为该公共客户端提供刷新令牌授权,因此您不需要刷新令牌时发送客户端密码。 C) refresh-token 在检查用户的有效性时为后端提供了一个“hart-beat”!
这个答案是错误的,因为 Andreas Lundgren 指出
C
Community

为了消除一些混淆,您必须了解 client secret用户密码 的角色,它们是非常不同的。

客户端是由服务器支持的应用程序/网站/程序/...,它希望通过使用第三方身份验证服务来对用户进行身份验证。客户端密码是一个(随机)字符串,此客户端和身份验证服务器都知道。使用这个秘密,客户端可以通过身份验证服务器识别自己,接收请求访问令牌的授权。

要获取初始访问令牌和刷新令牌,需要的是:

用户标识

用户密码

客户标识

客户秘密

然而,为了获得刷新的访问令牌,客户端使用以下信息:

客户标识

客户秘密

刷新令牌

这清楚地表明了区别:在刷新时,客户端通过使用其客户端密钥接收刷新访问令牌的授权,因此可以使用刷新令牌而不是用户 ID + 密码重新验证用户。这有效地防止了用户必须重新输入他/她的密码。

这也表明丢失刷新令牌是没有问题的,因为客户端 ID 和机密是未知的。它还表明,保持客户端 ID 和客户端机密是至关重要的。


m
mfaani

这个答案是在两位高级开发人员(John Brayton 和 David Jennes)的帮助下完成的。

使用刷新令牌的主要原因是减少攻击面。

假设没有刷新键,让我们看一下这个例子:

一栋楼有80扇门。所有的门都是用同一把钥匙打开的。密钥每 30 分钟更换一次。在 30 分钟结束时,我必须将旧钥匙交给钥匙制造商并获得一把新钥匙。

如果我是黑客并拿到了你的钥匙,那么在 30 分钟结束时,我会将它快递给钥匙制造商并获得一个新钥匙。我将能够不断地打开所有的门,不管钥匙怎么换。

问题:在这 30 分钟内,我有多少次攻击钥匙的机会?每次您使用密钥时,我都有 80 次黑客攻击机会(将其视为发出网络请求并传递访问令牌以识别您自己)。这就是 80X 攻击面。

现在让我们看一下相同的示例,但这次我们假设有一个刷新键。

一栋楼有80扇门。所有的门都是用同一把钥匙打开的。密钥每 30 分钟更换一次。要获取新密钥,我无法传递旧的访问令牌。我必须只通过刷新键。

如果我是黑客并拿到了你的密钥,我可以使用它 30 分钟,但在 30 分钟结束时将它发送给密钥制造者没有任何价值。如果我这样做了,那么密钥制造者只会说“此令牌已过期。您需要刷新令牌。”为了能够扩展我的破解,我必须将信使破解给钥匙制造商。信使有一个独特的密钥(将其视为刷新令牌)。

问题:在这 30 分钟内,我有多少次针对刷新键的黑客攻击机会? 80?不,我只有 1 次黑客攻击机会。在此期间,快递员与钥匙制造商沟通。这就是 1X 攻击面。我确实有 80 次攻击密钥的机会,但 30 分钟后它们都不好。

服务器将根据凭据和(通常)JWT 的签名来验证访问令牌。

访问令牌泄漏是不好的,但一旦过期,它对攻击者就不再有用了。刷新令牌泄漏要糟糕得多,但可能不太可能。 (我认为存在质疑刷新令牌泄漏的可能性是否远低于访问令牌泄漏的可能性,但这就是想法。)

关键是访问令牌会添加到您发出的每个请求中,而刷新令牌仅在刷新流程期间使用,因此 MITM 看到令牌的机会较小

频率有助于攻击者。 Heartbleed 之类的 SSL 中的潜在安全漏洞、客户端中的潜在安全漏洞以及服务器中的潜在安全漏洞都使泄漏成为可能。

此外,如果授权服务器与处理其他客户端请求的应用程序服务器分开,那么该应用程序服务器将永远不会看到刷新令牌。它只会看到不会存在更长时间的访问令牌。

分隔有利于安全。

最后但并非最不重要的刷新令牌可以旋转。意思是“每次客户端请求将刷新令牌交换为新的访问令牌时,都会返回一个新的刷新令牌。”。随着刷新令牌的不断交换和失效,威胁降低了。举个例子:令牌通常在 TTL 通常一个小时后过期。

刷新令牌并非总是如此,但经常在使用时被撤销并发行新令牌。这意味着如果您遇到网络故障,当您检索新的刷新令牌时,下次您发送该刷新令牌时,它会被视为已撤销,您必须登录。

有关轮换的更多信息,请参阅 herehere

概括

降低频率

区隔

令牌的轮换(更快的失效)和更精细的管理(到期时间或发出的请求数)。

所有这些都有助于减轻威胁

有关这方面的另一种看法,请参阅 this awesome answer

什么刷新令牌不是关于?

通过刷新令牌更新/撤销访问级别的能力是选择使用刷新令牌的副产品,否则独立访问令牌可能会被撤销或在其过期并且用户获得新令牌时修改其访问级别


刷新令牌也可能无效,在这种情况下,该人需要在获得新的刷新密钥之前向快递员表明自己的身份。为了让这个刷新密钥更加安全,您可以实施所谓的“刷新令牌轮换”,每次询问访问令牌时,也会给出一个新的刷新密钥。如果您或黑客使用旧的刷新密钥去找快递公司,那么快递公司也会使最新的新刷新密钥失效,并且没有人再获得新的访问令牌。
为什么攻击者获取刷新令牌比获取访问令牌更难?使用 HTTPS 可以在传输过程中提供保护,但如果是 SPA,我需要将它们都存储在浏览器中。因此,攻击者可以同时强化它们。我错过了什么吗?
@Vmxes 在这种情况下,我认为它们没有什么不同。但是从浏览器、路由器、ISP、VPN 等之间的传输层上下文来看,访问令牌每小时可以传递 1000 次,而刷新令牌只能传递一次。
在您的示例中,您使用旧令牌(旧密钥)来获取新令牌(新密钥)。如果没有刷新令牌机制,这不是您获得访问令牌的方式:每次需要新的访问令牌时,您都必须发回密码。
同时拥有 Access 和 Refresh 令牌意味着攻击者有更多机会通过猜测其中之一来访问用户的帐户。如果不是这种情况,如果攻击者能够从客户端破解您的访问令牌,为什么攻击者将无法破解您的刷新令牌。
M
Manicode

此答案来自 Justin Richer 通过 OAuth 2 标准正文电子邮件列表。这是在他的许可下发布的。

刷新令牌的生命周期取决于(AS)授权服务器——它们可以过期、被撤销等。刷新令牌和访问令牌之间的区别在于受众:刷新令牌只返回到授权服务器,访问令牌转到(RS)资源服务器。

此外,仅获取访问令牌并不意味着用户已登录。事实上,用户甚至可能不再存在,这实际上是刷新令牌的预期用例。刷新访问令牌将使您代表用户访问 API,它不会告诉您用户是否在那里。

OpenID Connect 不仅为您提供来自访问令牌的用户信息,它还为您提供 ID 令牌。这是针对客户端本身而不是 AS 或 RS 的单独数据片段。在 OIDC 中,如果您可以获得新的 ID 令牌,您应该只考虑某人实际通过协议“登录”。刷新它可能还不够。

如需更多信息,请阅读http://oauth.net/articles/authentication/


这似乎与 OpenID Connect 和身份验证有关,所以我看不出这是如何回答问题的,这是关于刷新令牌的动机。
P
Phil

客户端可能会以多种方式受到损害。例如,可以克隆手机。访问令牌过期意味着客户端被迫重新向授权服务器进行身份验证。在重新认证期间,授权服务器可以检查其他特性(IOW 执行自适应访问管理)。

刷新令牌只允许客户端重新认证,而重新授权强制与用户进行对话,许多人表示他们不愿意这样做。

刷新令牌基本上适用于普通网站可能会选择在一小时左右后定期重新验证用户的地方(例如银行网站)。由于大多数社交网站不会重新验证 Web 用户,因此目前使用率不高,那么他们为什么要重新验证客户端呢?


“刷新令牌只允许客户端重新认证......”是这里的一个重要方面。
b
bitcoder

为了进一步简化 B T 的回答:当您通常不希望用户再次输入凭据但仍希望能够撤销权限时使用刷新令牌(通过撤销刷新令牌)

您不能撤销访问令牌,只能撤销刷新令牌。


您可以撤销访问令牌,这将需要再次登录以获得另一个访问令牌或使用刷新令牌来获取另一个访问令牌。如果刷新令牌无效,则用户必须重新进行身份验证才能获取新的访问令牌以及新的刷新令牌。
我不同意。访问令牌由身份验证服务器颁发,用到期日期签名,然后发送给客户端。当客户端将该令牌发送到资源服务器时,资源服务器不会联系身份验证服务器来验证令牌;它只查看(已签名且未篡改的)令牌中的到期日期。因此,无论您在身份验证服务器上做什么来尝试“撤销”,资源服务器都不关心。有些人将客户端注销称为撤销(即客户端删除其令牌)但恕我直言这是误导性术语 - 我们希望在服务器而不是客户端“撤销”令牌
并不是说您不能编写自定义代码来忽略某些令牌(如此处 stackoverflow.com/questions/22708046/…),但这样做可能涉及每次客户端调用时从资源服务器到 oauth 服务器/db 的一些网络旅行。您可以通过使用刷新令牌来避免这些调用,我认为这更符合 oauth 作者的意图。
P
Pang

为什么不让 access_token 与 refresh_token 一样长而没有 refresh_token 呢?

除了其他人提供的很好的答案之外,我们使用刷新令牌还有另一个原因,它与声明有关。

每个令牌都包含声明,其中可以包括用户名、他们的角色或创建声明的提供者等任何内容。随着令牌的刷新,这些声明也会更新。

如果我们更频繁地刷新令牌,我们显然会给我们的身份服务带来更大的压力;但是,我们正在获得更准确和最新的声明。


将此类“声明”放入访问令牌中将是一种不寻常的不良做法。如 the specification 中所述,访问令牌“通常对客户端不透明”。您是否有执行此操作的 OAuth 提供程序的示例?
@heymega 当用户角色从 ADMIN 降级到 REGULAR_USER 期望是用户角色需要立即撤销,而不是在 access_token 过期时。因此,看起来每次请求都访问数据库是不可避免的。
@svlada 我想这将是应用程序将实体从 ADMIN 降级到 REGULAR_USER 的情况(在同一过程中)也需要撤销适当的令牌。即,如果我们知道索赔将会改变,我们不会等待到期,我们会立即撤销
A
Anbuselvan Rocky

假设您使 access_token 持续很长时间,并且没有 refresh_token,那么在一天之内,黑客得到了这个 access_token,他可以访问所有受保护的资源!

但是如果你有refresh_tokenaccess_token的存活时间很短,所以黑客很难破解你的access_token,因为它会在短时间内失效。 Access_token 只能通过使用 refresh_tokenclient_idclient_secret 来检索,而黑客没有。


“不仅使用 refresh_token,还使用 client_id 和 client_secret,黑客没有这些。” 1.假设它只是访问令牌,那么黑客是否还需要client_id和client_secret? 2.如果一个黑客是一个好的黑客,那么他也可以破解client_id和client_secret。不管那部分,破解额外的东西与比较无关,因为如果很难破解,那么仅使用访问令牌的情况也很难破解……长话短说,你不是在比较相同的情况。你正在混合它们
K
Kraming

授权服务器保留刷新令牌。访问令牌是自包含的,因此资源服务器可以在不存储它的情况下对其进行验证,从而在验证时节省检索工作。讨论中缺少的另一点来自 rfc6749#page-55

“例如,授权服务器可以使用刷新令牌轮换,其中每个访问令牌刷新响应都会发出一个新的刷新令牌。之前的刷新令牌无效但由授权服务器保留。如果刷新令牌被泄露并随后被攻击者和合法客户端,其中之一将提供一个无效的刷新令牌,这将通知授权服务器违规行为。”

我认为使用刷新令牌的全部意义在于,即使攻击者以某种方式设法获得刷新令牌、客户端 ID 和秘密组合。如果每个刷新请求都导致新的访问令牌和刷新令牌,则可以跟踪从攻击者获取新访问令牌的后续调用。


我认为这是非常重要的一点 :-) 它也 - 在某种程度上 - 使这里的论点 auth0.com/docs/tokens/refresh-token/current#restrictions A Single-Page Application (normally implementing Single-Page Login Flow) should not under any circumstances get a Refresh Token. The reason for that is the sensitivity of this piece of information. You can think of it as user credentials, since a Refresh Token allows a user to remain authenticated essentially forever. Therefore you cannot have this information in a browser, it must be stored securely. 无效
S
Saptarshi Basu

让我们考虑一个系统,其中每个用户都链接到一个或多个角色,每个角色都链接到一个或多个访问权限。可以缓存此信息以获得更好的 API 性能。但是,用户和角色配置可能会发生变化(例如,可能授予新访问权限或撤销当前访问权限),这些应该反映在缓存中。

我们可以为此目的使用访问和刷新令牌。当使用访问令牌调用 API 时,资源服务器会检查缓存的访问权限。如果有任何新的访问授权,它不会立即反映。一旦访问令牌过期(比如 30 分钟后)并且客户端使用刷新令牌生成新的访问令牌,就可以使用来自数据库的更新的用户访问权限信息来更新缓存。

换句话说,我们可以将昂贵的操作从使用访问令牌的每个 API 调用转移到使用刷新令牌生成访问令牌的事件中。


K
KJ Sudarshan

刷新令牌和访问令牌只是术语。

这个小类比有助于巩固使用访问令牌和刷新令牌背后的基本原理:

假设 Alice 通过邮寄方式向 Bob 发送了一张支票,该支票可以在发行后 1 小时(假设)内兑现,否则银行将不会兑现。但是爱丽丝还在帖子中附上了一张给银行的便条,要求银行接受并兑现支票,以防它有点延迟(在规定的范围内)

当 Bob 收到这张支票时,如果他看到这张支票被篡改(token tampering),他将自己丢弃这张支票。如果没有,他可以把它带到银行兑现。在这里,当银行注意到发行时间已超过 1 小时的时限,但看到 Alice 的签名票据要求银行兑现,以防规定的延迟在可接受的范围内。

看到这张纸条后,银行会尝试验证签名的消息并检查 Alice 是否仍然拥有正确的权限。如果是,银行将支票兑现。 Bob 现在可以向 Alice 确认这一点。

虽然不是非常准确,但这个类比可以帮助您注意处理交易所涉及的不同部分:

爱丽丝(发件人 - 客户端)

Bob(接收者 - 资源服务器)

银行(授权服务器)

验证过程(数据库访问)

检查(访问令牌)

注意(刷新令牌)

主要是,我们希望减少对 Auth Server 以及最终对数据库的 API 调用次数,以优化可扩展性。我们需要在便利性和安全性之间取得适当的平衡。

注意:在链中让 Auth 服务器比资源服务器更早地响应请求当然更常见。


B
Bernard Wiesner

这一切都与扩展和保持您的资源服务器无状态有关。

您的服务器/资源服务器服务器是无状态的,这意味着不检查任何存储以非常快速地响应。通过使用公钥来验证令牌的签名来做到这一点。检查每个请求的 access_token。通过仅检查 access_token 的签名和到期日期,响应非常快并且允许扩展。 access_token 应该有很短的过期时间(几分钟),因为没有办法撤销它,如果它被泄露,损害是有限的。

服务器是无状态的,意味着不检查任何存储以非常快速地响应。通过使用公钥来验证令牌的签名来做到这一点。

检查每个请求的 access_token。

通过仅检查 access_token 的签名和到期日期,响应非常快并且允许扩展。

access_token 应该有很短的过期时间(几分钟),因为没有办法撤销它,如果它被泄露,损害是有限的。

身份验证服务器/OAuth 服务器服务器不是无状态的,但它可以,因为请求要少得多。仅在 access_token 过期时检查 refresh_token。 (例如每 2 分钟)请求率远低于资源服务器。将刷新令牌存储在数据库中并可以撤销它。 refresh_token 可以有很长的过期时间(几周/几个月),如果它被泄露,有一种方法可以撤销它。

服务器不是无状态的,但它可以,因为请求要少得多。

仅在 access_token 过期时检查 refresh_token。 (例如每 2 分钟)

请求率远低于资源服务器。

将刷新令牌存储在数据库中并可以撤销它。

refresh_token 可以有很长的过期时间(几周/几个月),如果它被泄露,有一种方法可以撤销它。

但是有一个重要的注意事项,身份验证服务器的请求要少得多,因此可以处理负载,但是可能存在存储问题,因为它必须存储所有 refresh_tokens,如果用户急剧增加,这可能会成为问题。


A
Anusha Dharmasena

据我了解,如果您需要能够撤消访问权限,刷新令牌只是为了节省性能和成本。

例如1:不实现刷新令牌;只实现长期访问令牌:如果用户滥用服务(例如:不支付订阅费),您需要能够撤销访问令牌=>您需要在每个 API 调用上检查访问令牌的有效性需要访问令牌,这会很慢,因为它需要数据库查找(缓存可以提供帮助,但这更复杂)。

例 2:实现刷新令牌和短期访问令牌:如果用户滥用服务(例如:不支付订阅费),您需要能够撤销访问令牌 => 短期访问令牌将在短期内过期白色(例如 1 小时)并且用户将需要获取新的访问令牌,因此我们不需要对需要访问令牌的每个 API 调用进行验证。您只需要在从刷新令牌生成访问令牌时验证用户。对于不良用户,如果无法生成访问令牌,您可以注销该用户。当用户尝试重新登录时,验证将再次运行并返回错误。


C
Community

首先,客户端通过授权授权向授权服务器进行身份验证。然后,客户端通过提供访问令牌向资源服务器请求受保护的资源。资源服务器验证访问令牌并提供受保护的资源。客户端通过授予访问令牌向资源服务器发出受保护的资源请求,资源服务器在该令牌中对其进行验证并服务于请求(如果有效)。此步骤不断重复,直到访问令牌过期。如果访问令牌过期,客户端向授权服务器进行身份验证,并通过提供刷新令牌来请求新的访问令牌。如果访问令牌无效,则资源服务器将无效令牌错误响应发送回客户端。客户端通过授予刷新令牌与授权服务器进行身份验证。然后,授权服务器通过对客户端进行身份验证来验证刷新令牌,并在其有效时发出新的访问令牌。


这实际上并没有提到刷新令牌的来源。我假设第二段应该说 access token + refresh token
r
raisercostin

由于刷新和访问令牌是包含大量语义的术语,因此术语转换可能会有所帮助吗?

Revokable Tokens - 必须与授权服务器检查的令牌可以链接(请参阅 RTR - 刷新令牌轮换)可用于创建 NonRevokable 令牌,但也可以直接使用(当交易量较小且检查不会成为负担时) 可以长期存在,但这取决于用户必须多久使用凭据(用户名/密码)来获取新凭据,这可能会在 RTR 或任何其他可疑行为上失效

可以链接(请参阅 RTR - 刷新令牌轮换)

可用于创建 NonRevokable Tokens,但也可直接使用(当交易量较小且支票不会成为负担时)

可以长寿,但这取决于用户必须多久使用凭据(用户名/密码)才能获得新的凭据

可以在 RTR 或任何其他可疑行为上失效

NonRevokable Tokens - 自包含且不需要与授权服务器检查的令牌对于大数据很有用,分布式服务器/api调用水平扩展应该是短暂的(因为不可撤销)

可用于大数据、分布式服务器/api 调用以水平扩展

应该是短暂的(因为不可撤销)

在 2020 年,刷新令牌也可以存在于浏览器中(最初是为后端系统提供的)成为公认的 - 请参阅 https://pragmaticwebsecurity.com/articles/oauthoidc/refresh-token-protection-implications。因此,焦点从“可刷新性”(在没有用户的情况下后端如何延长对 api 的访问)切换到“可撤销性”。

因此,对我来说,将刷新令牌读取为可撤销令牌并将访问令牌读取为不可撤销令牌(可能是快速过期不可撤销令牌)看起来更安全。

作为关于 2021 年良好实践的旁注,系统始终可以从可撤销令牌开始,并在授权服务器压力增加时转向不可撤销令牌。


a
aderchox

为了理解这个问题的答案,我们需要理解两个重要的点。

第一点是,有时用户的访问令牌可能会在用户不知情的情况下被盗。由于用户不知道攻击,他们将无法手动通知我们。然后,就我们给攻击者完成攻击的时间(机会)而言,例如 15 分钟和一整天之间会有巨大的差异。因此,这就是我们需要每隔“短时间”(例如每 15 分钟)自己“刷新”访问令牌的原因,我们不想推迟很长时间(例如一整天)。因此,OP 在问题中所说的显然不是一个选项(只要刷新令牌,就可以延长访问令牌的到期时间)。

所以我们至少有以下两个选择:

要求每个用户每隔很短的时间重新输入他们的凭据,以便为他们提供新的访问令牌。但显然,这不是一个受欢迎的选择,因为它会打扰用户。

使用刷新令牌。阅读下面的第二点以了解其工作原理(其背后的逻辑)。

要理解的第二点是,因为我们已经将访问令牌与刷新令牌分开,现在刷新令牌可以以“不同的方式”发送,因此我们可以以攻击者的 JavaScript 无法访问的方式发送它(客户端一般代码),例如,使用 httpOnly 标记:

HttpOnly Cookie 是添加到浏览器 cookie 的标记,可防止客户端脚本访问数据。资源

在生成 cookie 时使用 HttpOnly 标志有助于降低客户端脚本访问受保护 cookie 的风险。 HttpOnly cookie 于 2002 年由 Microsoft Internet Explorer 开发人员针对 Internet Explorer 6 SP1 首次实施。来源(谢谢IE!)

因此,尽管攻击者可能仍然能够窃取访问令牌(强烈建议将它们保存在 RAM 中,而不是保存在本地存储等易受攻击的地方),但他们将无法窃取刷新令牌。所以,如果攻击者窃取了一个人的访问令牌,他们将只有很短的时间来滥用它(15 分钟?比一整天好多了!),然后一旦过期,他们就没有有机会自己买一个新鲜的。