在前后端分离的今天,跨域也成了每个前端工程师都需要了解的基本知识,在各种面试题中的日经话题。这个文章就是想总结一下关于同源策略的前世今生,以及怎么解决它。

同源策略

MDN 中我们可以看到关于同源策略是一个安全机制。详细的说明如下:
同源策略限制了从同一个源加载的文档或脚本如何与来自另一个源的资源进行交互。这是一个用于隔离潜在恶意文件的重要安全机制。

这个机制本身出发点是很好的,但是同源的限制非常严格,url,端口任一不同都会造成跨域错误。

图像 1.png

而且在控制台中你不会发现任何问题。随着前后端分离越来越普遍,这件事就越来越常见。那么它应该如何解决呢?
有个很重要的点,同源策略全称叫《浏览器的同源策略》,它是浏览器内建的一种安全机制。那么我们不要使用浏览器请求就能完美解决问题了。对于前端来说最方便的自然就是 node.js 了。

在开发中使用

现在市面上所有的脚手架都提供了 proxy 的能力,底层基于 http-proxy-middleware, 这个包可以把所有符合正则匹配的请求转发到某个地址,下面是个简单的demo:

  1. var express = require('express');
  2. var proxy = require('http-proxy-middleware');
  3. var app = express();
  4. app.use(
  5. '/api',
  6. proxy({ target: 'http://www.example.org', changeOrigin: true })
  7. );
  8. app.listen(3000);
  9. // http://localhost:3000/api/foo/bar -> http://www.example.org/api/foo/bar

这个配置可以将所有 /api 开头的请求转到到 http://www.example.org/ ,并且附带所有的参数,包括头信息和cookie。有一点需要注意的是,在浏览器控制台里看到的仍然是 http://localhost:3000/api/xxx ,转化的步骤是在 node.js 中完成。

在 Pro 中是用 proxy 更加简单在 config.ts 中配置即可,配置出来可能是这样的:

  1. proxy: {
  2. '/server/api/': {
  3. target: 'https://preview.pro.ant.design/',
  4. changeOrigin: true,
  5. pathRewrite: { '^/server': '' },
  6. },
  7. },

详细的配置建议直接查看 webpack-dev 的配置。

CORS

如果觉得以上改动需要配置比较麻烦,系统又比较简单,无需引入新的复杂度。我们可以使用 CSRF 的方式来允许跨域调用,在 express 中可以这么设置,

  1. res.header('Access-Control-Allow-Origin', '你的项目地址,用*将会带来安全问题');
  2. res.header('Access-Control-Allow-Headers', '*');
  3. res.header('Access-Control-Allow-Methods', '*');
  4. res.header('Content-Type', 'application/json;charset=utf-8');

在别的语言中方法也大同小异,最重要的是 Access-Control-Allow-Origin Access-Control-Allow-Headers Access-Control-Allow-Methods 头的相应设置。

在这里强烈建议每个人通读一下 MDN 的 HTTP访问控制 ,这篇文章可以解决跨域百分之八十的疑惑,图文并茂。

高级用法

在开发中我们可能需要区分多种情况,比如开发环境,测试环境,语法环境,在 Pro 中我们可以通过的环境变量来实现这个需求。

  1. const serveUrlMap = {
  2. dev:"https://dev.pro.ant.design/",
  3. pre:"https://pre.pro.ant.design/",
  4. test:"https://test.pro.ant.design/",
  5. idc:"https://idc.pro.ant.design/",
  6. };
  7. const { SERVE_ENV="idc" } = process.env;
  8. export default {
  9. ...,
  10. proxy:{
  11. '/server/api/': {
  12. target: serveUrlMap[SERVE_ENV],
  13. changeOrigin: true,
  14. pathRewrite: { '^/server': '' },
  15. },
  16. }
  17. }

我们只要在 package.json 中配置好各种快捷命令,就可以做到快速切换。

  1. {
  2. "scripts":{
  3. "start:dev": "cross-env SERVE_ENV=dev umi dev",
  4. "start:pre": "cross-env SERVE_ENV=pre umi dev",
  5. "start:test": "cross-env SERVE_ENV=test umi dev",
  6. }
  7. }

这里值得注意的是 config.ts 的环境为 node.js 的环境,里面是无法使用 dom 和浏览器的相关行为的。所有的配置也以简单为主。