[译] 如何且为何要在单页应用中防止跨域资源共享

  • 原文地址:How and Why You Should Avoid CORS in Single Page Apps
  • 原文作者:Ashan Fernando
  • 译文出自:掘金翻译计划
  • 本文永久链接:https://github.com/xitu/gold-miner/blob/master/article/2020/how-and-why-you-should-avoid-cors-in-single-page-apps.md
  • 译者:ZiXYu
  • 校对者:Xiny

如何且为何要在单页应用中防止跨域资源共享

在过去的十年中,单页应用成为了开发一个 web 应用的常规技术手段。如今的一些框架,比如 Angular,React 和 Vue 统治着前端开发的领域,它们为这些 web 应用提供了底层平台。好消息是,它一般从同一个域名提供前端服务和后端 API 服务。但是某些情况下,我们提供前端服务的地址 (例如 web.myapp.com) 和提供后端服务的地址 (例如 api.myapp.com) 是两个不同的子域名。某些时候,我们会仅在开发环境下允许跨域访问后端 API 服务。

image.png

跨域资源共享 (CORS) 是一种在浏览器中实现的机制,用于允许或拒绝某些其它域名发送到你 web 应用的请求。 通过 CORS,web 浏览器和 web 服务端就确认资源是否可以被访问的标准协议达成一致。所以记住了,实现了 CORS 并不意味着某些爬虫机器人或者其它的机制就不能访问你的服务端资源。

你需要 CORS 吗?

但是我们需要在我们的 web 应用中允许 CORS 吗?我想说在大多数情况下,你并不需要操心 CORS。然而,在某些特殊情况下,例如在你的主 web 应用之外允许嵌入一个页面 (比如表单,视频),那么这里你就需要考虑启用 CORS 了。不过,你可以控制仅启用 CORS 的一部分功能来满足特定的需求。

CORS 带来的问题

CORS 带来最主要的问题是会影响到 web 应用的性能。当你的前端应用向一个不同的域名发送了一个 HTTP 请求,浏览器将会发送一个额外的预检 HTTP 请求以确认这个服务器是否接受来自发送域名的请求。

所以对于每个由前端触发的 HTTP 请求,浏览器都需要发送 2 个 HTTP 请求,从而增加了总体的响应时间。在大多数情况下,这个额外的延迟是在 web 应用中可被感知的,同时也降低了用户体验。

单页应用中的 CORS

在单页应用中,CORS 的使用更为显著。当 web 应用仅使用 HTTP headers (Accept, Accept-Language, DPR, Downlink, Save-Data, Viewport-Width, Width, Content-Language, Content-Type (除 application/x-www-form-urlencoded, multipart/form-data, text/plan 之外)) 和 HTTP methods (GET, HEAD, POST) 来调用后端 API 时,浏览器是不会考虑 CORS 的。在你的单页应用中,你也可能需要使用其它的 HTTP header 和 HTTP methods。

在这些应用中,我们在前端应用中把用于操作服务端的后端 API 地址定义为一个变量。除此之外,我们可能甚至需要在开发环境允许 CORS,因为前端和后端 API 的开发服务端可能运行在两个不同的端口。同时开发环境也可能影响你在生产环境的设置,在生产环境中你可能将前端和后端 API 部署在两个不同的子域名中。

但是我们需要朝着这个方向走吗?让我们来看看如何在开发和生产环境都避免使用 CORS 的方法吧。

避免在开发环境使用 CORS

现在,我们为前端选择的开发服务端大多数都使用 NodeJS。而绝大多数的 Node 服务端支持代理设置。除此之外,像 Angular,React,Vue 这样的框架都随着 Webpack 的开发服务端内置了对代理设置的支持。

那么代理设置到底能做什么?

让我们假设一下我们的前端应用运行在 http://localhost:4200 ,同时后端 API 服务运行在 http://localhost:3000/api/。你的前端应用需要存储后端 API URL 的地址和端口从而在本地运行。同时,你也必须在后端 API 服务中开启 CORS 以支持运行在 http://locahost:4200 的前端服务访问。

通过在前端开发服务端上配置代理设置,我们可以避免上述的所有麻烦。当你使用了代理时,你只需要在你的前端应用中存储相对路径 (/api)。当在本地运行前端应用时,你的前端服务会尝试使用同样的域名和端口 (http://localhost:4200/api/) 来访问后端服务,浏览器并不会担心任何有关 CORS 的问题。

在这个阶段,代理就开始发挥它的作用了。在代理设置中,你可以设置在你的前端开发服务中将所有来自 /api 的请求转发到 http://localhost:3000

因为你的开发服务器是和后端 API 通讯的中间人,它可以安全的避免 CORS。下面的例子展示了你可以在 Webpack 开发服务端 添加的代理配置。


module.exports = {

  //...

  devServer: {

    proxy: {

      '/api': 'http://localhost:3000'

    }

  }

};

作为一种替代的方案,如果你不想在你的前端应用中为后端 API 使用相对路径,你可以使用特殊的方法启动浏览器以禁用 CORS 进行本地测试。可参考 禁用 CORS 启动 Chrome 浏览器。

在生产环境避免 CORS

在生产环境中,除非你的前后端都运行在同一个 web 服务器中,你就需要在它们的前面设置一个网关或代理从而从同一域名提供服务。在某些情况下,如果你的负载均衡可以基于不同的 HTTP 路径路由到不同的端点,那么它也就足够使用了。

和开发服务器的代理类似,网关,代理或者负载均衡都是依据我们提供的配置来匹配请求中接收到的 HTTP 路径从而进行路由的。下面的列表包括一部分支持基于 URL 路径路由的流行网关、代理和负载均衡。

  • NGINX

  • Traefik

  • AWS CloudFront

  • AWS 应用负载均衡

  • Azure 应用网关

此外,仅允许相同来源的访问来增强你的后端 API 也是十分必要的。

总结

综上所述,我希望你能明白 CORS 的使用和在你的单页应用中避免使用它的必要性。尽快看起来在你的开发和生产环境设置代理好像很麻烦,但是它会比你想象的更容易。

例如,在 Angular,React 和 Vue 中设置一个开发服务器的代理,只需要在 Webpack 的设置文件中增加几行代码就可以把你的请求代理的后端 API 从而避免 CORS。这同样适用于生产环境,因为有非常健全的方法来实现基于 URL 路径的路由。

然而,你必须为你的后端 API 设置一套合适的路径转换规则,以防止每次你新增加一个端点就必须去更新代理设置。比如,如果你使用了一个基本路径 (例如 \api\),那么将所有包含这个基本路径的请求都转发到后端 API ,同时将其它的 HTTP 请求回退到前端资源是更容易的。

最后,我想要再强调一下,如果你不存在任何的需要来使用 CORS,那么在你的开发和生产环境都只需要让你的后端 API 允许同源访问。从我的经验来说,这会在节省大量的时间,同时避免很多潜在的风险。

了解更多

使用 Hooks 在 React 中获取数据

每个程序猿都应该遵守的代码准则

如何在本地设置一个私有化 NPM 源

如果发现译文存在错误或其他需要改进的地方,欢迎到 掘金翻译计划 对译文进行修改并 PR,也可获得相应奖励积分。文章开头的 本文永久链接 即为本文在 GitHub 上的 MarkDown 链接。


掘金翻译计划 是一个翻译优质互联网技术文章的社区,文章来源为 掘金 上的英文分享文章。内容覆盖 Android、iOS、前端、后端、区块链、产品、设计、人工智能等领域,想要查看更多优质译文请持续关注 掘金翻译计划、官方微博、知乎专栏。

你可能感兴趣的:([译] 如何且为何要在单页应用中防止跨域资源共享)