一篇搞定面试中的跨域问题

什么是CORS(跨源资源共享)?

CORS(Cross-Origin Resource Sharing)是一种机制,允许网页从不同的域访问服务器上的资源。

在同源策略下,浏览器限制了跨域访问,CORS允许服务器指定哪些源可以访问其资源。

同源策略(Same-origin policy)

同源策略在web应用安全模型中是一个重要的概念。在这个策略下,浏览器允许第一个网页中包含的脚本可以获取第二个网页的数据,前提是这两个网页在同一个源下。

同源:需要URI、主机名、端口都相同。

这个策略可以防止一个网页上的恶意脚本通过DOM获取其他网页的敏感数据。

需要牢记的一点就是同源策略只应用于脚本,这意味着像images、css和其他动态加载的脚本 可以通过对应的标签跨域访问资源。

是否同源的规则

同源需要满足相同的协议(scheme),相同的主机名(host),相同的端口号(port)

Compared URL Outcome Reason
http://www.example.com/dir/page2.html Success Same scheme, host and port
http://www.example.com/dir2/other.html Success Same scheme, host and port
http://username:password@www.example.com/dir2/other.html Success Same scheme, host and port
http://www.example.com:81/dir/other.html Failure Same scheme and host but different port
https://www.example.com/dir/other.html Failure Different scheme
http://en.example.com/dir/other.html Failure Different host
http://example.com/dir/other.html Failure Different host (exact match required)
http://v2.www.example.com/dir/other.html Failure Different host (exact match required)
http://www.example.com:80/dir/other.html Depends Port explicit. Depends on implementation in browser.

::: tip
scheme: http https
:::

CORS存在的意义

跨域资源共享(CORS) 是一种机制,它使用额外的 HTTP 头来告诉浏览器 让运行在一个 origin (domain) 上的Web应用被准许访问来自不同源服务器上的指定的资源。

出于安全性,浏览器限制脚本内发起的跨源HTTP请求。 例如,XMLHttpRequest 和 Fetch API 遵循同源策略。这意味着使用这些 API 的 Web 应用程序只能从加载应用程序的同一个域请求 HTTP 资源,除非响应报文包含了正确 CORS 响应头。

跨源域资源共享(CORS)机制允许 Web 应用服务器进行跨源访问控制,从而使跨源数据传输得以安全进行。

CORS的工作原理是什么?

当浏览器发起跨域请求时,会在请求头中添加Origin字段,指示请求的源。服务器接收到请求后,会在响应头中添加Access-Control-Allow-Origin字段,指示允许访问的源。如果服务器允许该源访问资源,浏览器会将响应返回给客户端。

CORS中的预检请求是什么?为什么需要预检请求?

预检请求是浏览器在发送跨域请求前发送的一种OPTIONS请求,用于检查服务器是否支持跨域请求。预检请求中包含一些额外的头信息,如Access-Control-Request-Method、Access-Control-Request-Headers等。

预检请求的目的是确保服务器支持跨域请求,避免跨域请求对服务器造成安全风险。只有在服务器返回允许的响应头后,浏览器才会发送实际的跨域请求。

了解了上面的内容,我们解决浏览器控制台的跨域问题,一般有两个方向:

  1. 后端服务设置允许跨域访问
  2. 前端通过代理访问资源(开发阶段使用)

如何在服务器端配置CORS?

在服务器端配置CORS通常需要在响应头中添加一些字段,如Access-Control-Allow-Origin、Access-Control-Allow-Methods、Access-Control-Allow-Headers等。通过配置这些字段,可以控制允许的源、请求方法和请求头。

用 Node.js 的一个框架 koa 来举例,解决跨域使用 koa-cors 非常简单,如下:

var koa = require('koa');
var route = require('koa-route');
var cors = require('koa-cors');
var app = koa();

app.use(cors());

app.use(route.get('/', function() {
  this.body = { msg: 'Hello World!' };
}));

app.listen(3000);

这个中间件大概做了这样的事情:

module.exports = () => {
  return async function(ctx, next) {
    ctx.set('Content-Type', 'application/json');
    ctx.set('Access-Control-Allow-Origin', '*');
    ctx.set('Access-Control-Allow-Methods', 'GET, POST, PATCH, DELETE, PUT');
    ctx.set('Access-Control-Allow-Headers', 'X-Requested-With, content-type, X-Authorization, X-uuid');
    ctx.json = json.bind(ctx);
    ctx.halt = halt.bind(ctx);
    try {
        await next();
    } catch (e) {
        return ctx.halt(e.code, e.message);
    }
  };
};

这样前端收到的响应会是下面的样子:

HTTP/1.1 200 OK
Date: Mon, 01 Dec 2008 00:23:53 GMT
Server: Apache/2
Access-Control-Allow-Origin: *
Keep-Alive: timeout=2, max=100
Connection: Keep-Alive
Transfer-Encoding: chunked
Content-Type: application/xml

本例中,服务端返回的 Access-Control-Allow-Origin: * 表明,该资源可以被 任意 外域访问。

前端代理

如果使用了webpack 那么配置一个代理就很容易,通过代理模拟出和服务端同源的请求。

//webpack.config.js
module.exports = {
  //...
  devServer: {
    proxy: {
      '/api': {
        target: 'http://localhost:3000',
        pathRewrite: { '^/api': '' },
      },
    },
  },
};

这样前端发起的请求包含/api的路径就会被代理到 http://localhost:3000,并且会把/api替换为空。如果你的接口地址本来就包括/api,那只要把pathRewrite: { '^/api': '' }去掉即可。

这个devServer代理使用功能强大的 http-proxy-middleware 软件包。

http-proxy-middleware 内部又使用了 node-http-proxy

服务端代理

服务端代理是一种解决跨域访问的方法,通过在服务端发起请求并将结果返回给客户端,绕过了浏览器的同源策略限制。通常可以使用Node.js、Java、Python等后端语言来实现服务端代理。

以下是一个使用Node.js实现服务端代理的示例代码:

const express = require('express');
const request = require('request');

const app = express();

app.get('/proxy', (req, res) => {
  const url = req.query.url;
  request(url, (error, response, body) => {
    if (!error && response.statusCode === 200) {
      res.send(body);
    } else {
      res.status(500).send('Error');
    }
  });
});

app.listen(3000, () => {
  console.log('Server is running on port 3000');
});

在上面的示例中,当客户端发送请求到/proxy接口时,服务端会将请求转发到指定的URL,并将结果返回给客户端。

总结

总的来说,跨域资源共享是一个常见的Web开发问题,了解跨域资源共享的原理和解决方法对于开发安全可靠的Web应用程序非常重要。

参考链接

跨源资源共享(CORS)

你可能感兴趣的:(cors前端面试问题)