美文网首页
nginx中配置跨域

nginx中配置跨域

作者: 阳光_8af8 | 来源:发表于2020-02-13 14:30 被阅读0次

    前后端分离的项目中会碰到跨域的问题。在应用的日志中当出现403跨域错误的时候 No 'Access-Control-Allow-Origin' header is present on the requested resource,需要给Nginx服务器配置响应的header参数:

    1. 范围较大时,在nginx的站点配置文件中添加如下:

       add_header 'Access-Control-Allow-Origin' '*';
       add_header 'Access-Control-Allow_Credentials' 'true';
       add_header 'Access-Control-Allow-Headers' 'Authorization,Accept,Origin,DNT,X-CustomHeader,Keep-Alive,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Content-Range,Range';
       add_header 'Access-Control-Allow-Methods' 'GET,POST,OPTIONS,PUT,DELETE,PATCH';
    

    2. 如果对谓词有限制,添加如下即可:

        add_header Access-Control-Allow-Origin *;
        add_header Access-Control-Allow-Methods 'GET, POST, OPTIONS';
        add_header Access-Control-Allow-Headers 'DNT,X-Mx-ReqToken,Keep-Alive,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Authorization';
    
        if ($request_method = 'OPTIONS') {
            return 204;
        }
    

    3. 配置项意义

    (1)Access-Control-Allow-Origin:服务器默认是不被允许跨域的。给Nginx服务器配置Access-Control-Allow-Origin *后,表示服务器可以接受所有的请求源(Origin),即接受所有跨域的请求。
    (2)Access-Control-Allow_Credentials:是服务端下发到客户端的 response 中头部字段,意义是允许客户端携带验证信息,例如 cookie 之类的。这样客户端在发起跨域请求的时候,不就可以携带允许的头,还可以携带验证信息的头。
    (3)Access-Control-Allow-Headers :是为了防止出现以下错误,Request header field Content-Type is not allowed by Access-Control-Allow-Headers in preflight response.
    这个错误表示当前请求Content-Type的值不被支持。其实是程序发起了"application/json"的类型请求导致的。
    (4)Access-Control-Allow-Methods :允许跨域的谓词(GET,POST,OPTIONS,PUT,DELETE,PATCH,常用的是前面三个)。是为了防止出现以下错误:Content-Type is not allowed by Access-Control-Allow-Headers in preflight response.
    (5)给OPTIONS 添加 204的返回,是为了处理在发送POST请求时Nginx依然拒绝访问的错误;
    发送"预检请求"时,需要用到方法 OPTIONS ,所以服务器需要允许该方法。OPTIONS请求旨在发送一种“探测”请求以确定针对某个目标地址的请求必须具有怎样的约束.

    4. 预检请求

    CROS,全称是跨域资源共享 (Cross-origin resource sharing),是一个W3C标准;它的提出就是为了解决跨域请求的。

    跨域资源共享(CORS)标准新增了一组 HTTP 首部字段,允许服务器声明哪些源站有权限访问哪些资源。另外,规范要求,对那些可能对服务器数据产生副作用的HTTP 请求方法(特别是 GET 以外的 HTTP 请求,或者搭配某些 MIME 类型的 POST 请求),浏览器必须首先使用 OPTIONS 方法发起一个预检请求(preflight request),从而获知服务端是否允许该跨域请求。服务器确认允许之后,才发起实际的 HTTP 请求。在预检请求的返回中,服务器端也可以通知客户端,是否需要携带身份凭证(包括 Cookies 和 HTTP 认证相关数据)。

    Content-Type字段的类型为application/json的请求就是上面所说的搭配某些 MIME 类型的 POST 请求,CORS规定,Content-Type不属于以下MIME类型的,都属于预检请求:

    application/x-www-form-urlencoded
    multipart/form-data
    text/plain
    

    因此application/json的请求 会在正式通信之前,增加一次"预检"请求,这次"预检"请求会带上头部信息 Access-Control-Request-Headers: Content-Type:

    OPTIONS /api/test HTTP/1.1
    Origin: http://foo.example
    Access-Control-Request-Method: POST
    Access-Control-Request-Headers: Content-Type
    ......
    服务器回应时,返回的头部信息如果不包含Access-Control-Allow-Headers: Content-Type则表示不接受非默认的的Content-Type。即出现以下错误:Request header field Content-Type is not allowed by Access-Control-Allow-Headers in preflight response.

    5 两种请求

    浏览器将CORS请求分成两类:简单请求(simple request)和非简单请求(not-so-simple request)。
    (1)只要同时满足以下两大条件,就属于简单请求。
    a. 请求方法是以下三种方法之一:
    HEAD
    GET
    POST
    b.HTTP的头信息不超出以下几种字段:
    Accept
    Accept-Language
    Content-Language
    Last-Event-ID
    Content-Type:只限于三个值application/x-www-form-urlencoded、multipart/form-data、text/plain

    (2)凡是不同时满足上面两个条件,就属于非简单请求。

    6请求流程

    (1)简单
    对于简单请求,浏览器直接发出CORS请求。具体来说,就是在头信息之中,增加一个Origin字段。

    (2)非简单
    非简单请求是那种对服务器有特殊要求的请求,比如请求方法是PUT或DELETE,或者Content-Type字段的类型是application/json。非简单请求的CORS请求,会在正式通信之前,增加一次HTTP查询请求,称为"预检"请求(preflight)。

    浏览器先询问服务器,当前网页所在的域名是否在服务器的许可名单之中,以及可以使用哪些HTTP动词和头信息字段。只有得到肯定答复,浏览器才会发出正式的XMLHttpRequest请求,否则就报错。

    参考地址:http://www.ruanyifeng.com/blog/2016/04/cors.html

    相关文章

      网友评论

          本文标题:nginx中配置跨域

          本文链接:https://www.haomeiwen.com/subject/lsmkfhtx.html