解决 Vercel + Cloudflare 部署中的 ERR_TOO_MANY_REDIRECTS 错误

讨论通过 Vercel 和 Cloudflare 部署网站时,报出的“ERR_TOO_MANY_REDIRECTS”错误及其解决方法。

我在部署本网站时,使用了 Vercel 作为前端托管服务,并将域名解析交给了 Cloudflare。初步配置完成后,满怀期待地访问了自己的网站,结果迎面而来的却是浏览器的报错提示:ERR_TOO_MANY_REDIRECTS

我非常困惑,仅仅将域名绑定到 Vercel 并在 Cloudflare 上做了域名解析,且并未做任何复杂的配置,何来的重定向次数过多?在网上搜集了一些资料,总结起来,这个问题是由于 Cloudflare 在“灵活”模式下通过 HTTP 与 Vercel 通信,而 Vercel 会将所有 HTTP 请求重定向到 HTTPS,导致重定向循环而造成的。

从 SSL/TLS 配置谈起

Cloudflare 提供了几种 SSL/TLS 的配置模式,其中最常见的模式包括“灵活”(Flexible)和“完全”(Full)。我经历的报错正是因为 SSL/TLS 设置不当引起的。以下是这几种模式主要特点:

  • 灵活模式(Flexible): 在这种模式下,Cloudflare 与客户端的连接是通过 HTTPS(SSL 加密)完成的,但是 Cloudflare 与源服务器(例如 Vercel)之间的连接却是使用 HTTP。这意味着从用户访问网站到 Cloudflare 是加密的,但从 Cloudflare 到 Vercel 是未加密的。
  • 完全模式(Full): Cloudflare 与客户端以及 Cloudflare 与源服务器之间都使用 HTTPS 进行通信,但是并不验证源服务器的 SSL 证书是否合法。
  • 完全(严格)模式(Full (Strict)): 同样使用 HTTPS,但要求 Cloudflare 验证源服务器的 SSL 证书是有效的、可信的。 在部署网站时,默认情况下 Cloudflare 的 SSL/TLS 设置可能是“灵活”。这意味着,当客户端通过 HTTPS 访问网站时,Cloudflare 会将请求解密后以 HTTP 的形式传递给 Vercel。而 Vercel 会自动将所有的 HTTP 请求重定向到 HTTPS。这一来一回便产生了一个问题:Vercel 将请求重定向到 HTTPS,而 Cloudflare 接收到 HTTPS 请求后再次转发,形成了一个自我循环。这时,浏览器会检测到这一过多的重定向,并抛出 ERR_TOO_MANY_REDIRECTS 错误。

报错原因

当 Vercel 收到 HTTP 请求时,它会返回 308 状态码,告知客户端应将请求重定向到 HTTPS URL。HTTP 308 是一种永久性重定向,意味着客户端必须使用 HTTPS 协议进行后续请求。

当客户端(即用户的浏览器)最初通过 HTTPS 访问 https://brume.top 时,Cloudflare 在灵活模式下将请求以 HTTP 方式传给 Vercel。Vercel 接收到 HTTP 请求后,按照其默认行为,发出 308 永久重定向,强制要求将请求重定向至 HTTPS 版本的相同 URL。Cloudflare 将此 HTTPS 重定向信息再次发送给客户端,而客户端早已处于 HTTPS 状态,因此这个重定向实际上是无效的。浏览器一再收到相同的重定向指令,形成了一个闭环,导致浏览器认为重定向次数过多,最终返回 ERR_TOO_MANY_REDIRECTS 错误。

此外,对于这个所谓的“灵活模式”,从安全性角度看,它虽然表面上为用户提供了 HTTPS 访问的体验,但在 Cloudflare 和源服务器之间的通信仍然是明文的 HTTP,这使得数据在传输过程中存在被窃听或篡改的风险。因此,严格意义上讲,“灵活模式”并不是一个安全的方案。

解决方案

要解决这个问题,只需更改 Cloudflare 的 SSL/TLS 配置,将其从“灵活”改为“完全(严格)”,确保 Cloudflare 与 Vercel 之间的通信使用的是 HTTPS 协议即可。 此时,Cloudflare 将强制会以 HTTPS 方式与 Vercel 通信,避免了重定向循环。

Licensed under CC BY-NC-SA 4.0