成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

CORS原理及@koa/cors源碼解析

loostudy / 1783人閱讀

摘要:服務(wù)端根據(jù)這個值,決定是否同意本次請求。預(yù)檢請求預(yù)檢請求用的請求方法是,表示這個請求是用來詢問的。該字段也可以設(shè)為星號,表示同意任意跨源請求。這是為了避免多次預(yù)檢請求。

首發(fā)于個人博客
目錄

跨域

簡單請求和復(fù)雜請求

服務(wù)端如何設(shè)置CORS

@koa/cors是怎么實現(xiàn)的

跨域 為什么會有跨域問題?

這是瀏覽器的同源策略所造成的,同源策略限制了從同一個源加載的文檔或腳本如何與來自另一個源的資源進行交互。這是一個用于隔離潛在惡意文件的重要安全機制。

一定要注意跨域是瀏覽器的限制,其實你用抓包工具抓取接口數(shù)據(jù),是可以看到接口已經(jīng)把數(shù)據(jù)返回回來了,只是瀏覽器的限制,你獲取不到數(shù)據(jù)。用postman請求接口能夠請求到數(shù)據(jù)。這些再次印證了跨域是瀏覽器的限制。
如何解決跨域?

jsonp: 帶有src屬性的標簽都可以用來, 但是只能處理GET請求

document.domain + iframe跨域

location.hash + iframe

window.name + iframe

postMessage跨域

Nginx配置反向代理

CORS(跨域資源共享):支持所有類型的HTTP請求

相信大家對于以上的解決方法都很熟悉,這里不再對每一種方法展開講解,接下來主要講一下CORS;

簡單請求和非簡單請求

瀏覽器將CORS跨域請求分為簡單請求和非簡單請求;

如果你使用nginx反向代理解決的跨域問題,則不會有跨域請求這個說法了,因為nginx反向代理就使得前后端是同一個域了,就不存在跨域問題了。

只要同時滿足一下兩個條件,就屬于簡單請求
(1)使用下列方法之一:

head

get

post

(2)請求的Heder是

Accept

Accept-Language

Content-Language

Content-Type: 只限于三個值:

application/x-www-form-urlencoded

multipart/form-data

text/plain

不同時滿足上面的兩個條件,就屬于非簡單請求。
瀏覽器對這兩種的處理,是不一樣的。

簡單請求 例子

對于簡單請求,瀏覽器直接發(fā)出CORS請求。具體來說,就是頭信息之中,增加一個Origin字段。

上面這個例子,post請求,Content-Typeapplication/x-www-form-urlencoded,滿足簡單請求的條件;響應(yīng)頭部返回Access-Control-Allow-Origin: http://127.0.0.1:3000;
瀏覽器發(fā)現(xiàn)這次跨域請求是簡單請求,就自動在頭信息之中,添加一個Origin字段;Origin字段用來說明請求來自哪個源(協(xié)議+域名+端口號)。服務(wù)端根據(jù)這個值,決定是否同意本次請求。

CORS請求相關(guān)的字段,都以 Access-Control-開頭

Access-Control-Allow-Origin:必選

請求頭Origin字段的值

*:接受任何域名

Access-Control-Allow-Credentials:可選,

true: 表示允許發(fā)送cookie,此時Access-Control-Allow-Origin不能設(shè)置為*,必須指定明確的,與請求網(wǎng)頁一致的域名。

不設(shè)置該字段:不需要瀏覽器發(fā)送cookie

Access-Control-Expose-Headers:可選

響應(yīng)報頭指示哪些報頭可以公開為通過列出他們的名字的響應(yīng)的一部分。默認情況下,只顯示6個簡單的響應(yīng)標頭:

Cache-Control

Content-Language

Content-Type

Expires

Last-Modified

Pragma

如果想要讓客戶端可以訪問到其他的首部信息,可以將它們在 Access-Control-Expose-Headers 里面列出來。

withCredentials 屬性

CORS請求默認不發(fā)送Cookie和HTTP認證信息,如果要把Cookie發(fā)到服務(wù)器,一方面需要服務(wù)器同意,設(shè)置響應(yīng)頭Access-Control-Allow-Credentials: true,另一方面在客戶端發(fā)出請求的時候也要進行一些設(shè)置;

// XHR
var xhr = new XMLHttpRequest();
xhr.open("GET", "http://example.com/", true); 
xhr.withCredentials = true; 
xhr.send(null);

// Fetch
fetch(url, {
  credentials: "include"  
})
非簡單請求

非簡單請求就是那種對服務(wù)器有特殊要求的請求,比如請求方法為PUTDELETE,或者Content-Type字段為application/json;

1. 預(yù)檢請求和回應(yīng)

非簡單請求的CORS請求,會在正式通信之前,增加一次HTTP查詢請求,稱為“預(yù)檢”請求;
瀏覽器先詢問服務(wù)器,當(dāng)前網(wǎng)頁所在的域名是否在服務(wù)器的許可名單之中,以及可以使用哪些HTTP動詞和頭信息字段,只有得到肯定答復(fù),瀏覽器才會發(fā)出正式的接口請求,否則就會報錯;

HTTP請求的方法是POST,請求頭Content-Type字段為application/json。瀏覽器發(fā)現(xiàn),這是一個非簡單請求,就自動發(fā)出一個預(yù)檢請求,要求服務(wù)器確認可以這樣請求。

1.1預(yù)檢請求

預(yù)檢請求用的請求方法是OPTIONS,表示這個請求是用來詢問的。頭信息里面,關(guān)鍵字段是Origin,表示請求來自哪個域。
除了Origin,預(yù)檢請求的頭信息包括兩個特殊字段:

Access-Control-Request-Method: 必選,用來列出瀏覽器的CORS請求會用到哪些HTTP方法,上例是POST

Access-Control-Request-Headers:該字段是一個用逗號分割的字符串,執(zhí)行瀏覽器CORS請求會額外發(fā)送的頭信息字段,上例是Content-Type;

1.2預(yù)檢回應(yīng)

服務(wù)器收到預(yù)檢請求以后,檢查了OriginAccess-Control-Request-MethodAccess-Control-Request-Headers字段以后,確認允許跨域請求,就可以做出回應(yīng)。
上面的HTTP回應(yīng)中,關(guān)鍵的是Access-Control-Allow-Origin字段,表示http://127.0.0.1:3000可以請求數(shù)據(jù)。該字段也可以設(shè)為星號,表示同意任意跨源請求。

如果瀏覽器否定了“預(yù)檢”請求,就會返回一個正常的HTTP回應(yīng),但是沒有任何CORS相關(guān)的頭信息字段,這時,瀏覽器就會認定,服務(wù)器不同意預(yù)檢請求,因此觸發(fā)一個錯誤,被XMLHttpRequest對象的onerror回調(diào)函數(shù)捕獲h。

服務(wù)器回應(yīng)的其他CORS字段

Access-Control-Allow-Methods:必需;它的值是逗號分隔的一個字符串,表明服務(wù)器支持的所有跨域請求的方法。注意,返回的是所有支持的方法,而不單是瀏覽器請求的方法。這是為了避免多次預(yù)檢請求。

Access-Control-Allow-Headers:如果瀏覽器請求頭里包括Access-Control-Request-Headers字段,則Access-Control-Allow-Headers字段是必需的。它也是一個逗號分隔的字符串,表明服務(wù)器支持的所有頭信息字段,不限于瀏覽器在預(yù)檢中請求的字段。

Access-Control-Allow-Credentials:與簡單請求時含義相同。

Access-Control-Allow-Max-Age: 可選,用來指定本次預(yù)檢請求的有效期。單位為秒。在有效期內(nèi),不用發(fā)出另一條預(yù)檢請求

2.正常請求和回應(yīng)

一旦服務(wù)器通過了預(yù)檢請求,以后每次瀏覽器正常的CORS請求,就都跟簡單請求一樣,會有一個Origin頭信息字段。服務(wù)器的回應(yīng),也都會有一個Access-Control-Allow-Origin頭信息字段;

服務(wù)端如何設(shè)置CORS 多帶帶接口多帶帶處理

比如一個簡單的登錄頁面,需要給接口接口傳入 username和password 兩個字段;前端的域名為 localhost:8900,后端的域名為 localhost:3200,構(gòu)成跨域。

1. 如果設(shè)置請求頭"Content-Type": "application/x-www-form-urlencoded",這種情況則為簡單請求;

會有跨域問題,直接設(shè)置 響應(yīng)頭 Access-Control-Allow-Origin*, 或者具體的域名;注意如果設(shè)置響應(yīng)頭Access-Control-Allow-Credentialstrue,表示要發(fā)送cookie,則此時Access-Control-Allow-Origin的值不能設(shè)置為星號,必須指定明確的,與請求網(wǎng)頁一致的域名。

const login = ctx => {
    const req = ctx.request.body;
    const userName = req.userName;
    ctx.set("Access-Control-Allow-Origin", "*");
    ctx.response.body = {
        data: {},
        msg: "登陸成功"
    };
}
2. 如果設(shè)置請求頭"Content-Type": "application/json",這種情況則為非簡單請求

處理OPTIONS請求,服務(wù)端可以多帶帶寫一個路由,來處理login的OPTIONS的請求

app.use(route.options("/login", ctx => {
    ctx.set("Access-Control-Allow-Origin", "*");
    ctx.set("Access-Control-Allow-Headers", "Content-Type");
    ctx.status = 204;
    
}));

大家都知道前端調(diào)用服務(wù)端的時候,會調(diào)用很多個接口,并且每個接口處理跨域請求的邏輯是完全一樣的,我們可以把這部分抽離出來,作為一個中間件;

寫一個中間件進行處理 首先了解一下koa中間件的“洋蔥圈”模型

將洋蔥的一圈看做是一個中間件,直線型就是從第一個中間件走到最后一個,但是洋蔥圈就很特殊了,最早use的中間件在洋蔥最外層,開始的時候會按照順序走到所有中間件,然后按照倒序再走一遍所有的中間件,相當(dāng)于每個中間件都會進入兩次,這就給了我們更多的操作空間。

const Koa = require("koa");
const app = new Koa();
app.use((ctx, next) => {
    console.log("a - 1");
    next();
    console.log("a - 2");
})
app.use((ctx, next) => {
    console.log("b - 1");
    next();
    console.log("b - 2");
})
app.use((ctx, next) => {
    console.log("c - 1");
    next();
    console.log("c - 2");
})

app.listen(3200, () => {
    console.log("啟動成功");
});

輸出

a - 1
b - 1
c - 1
c - 2
b - 2
a - 2

Koa官方文檔上把外層的中間件稱為“上游”,內(nèi)層的中間件為“下游”。
一般的中間件都會執(zhí)行兩次,調(diào)用next之前為一次,調(diào)用next時把控制按順序傳遞給下游的中間件。當(dāng)下游不再有中間件或者中間件沒有執(zhí)行 next 函數(shù)時,就將依次恢復(fù)上游中間件的行為,讓上游中間件執(zhí)行 next之后的代碼;

處理跨域的中間件簡單示例
const Koa = require("koa");
const app = new Koa();
const route = require("koa-route");
var bodyParser = require("koa-bodyparser");

app.use(bodyParser()); // 處理post請求的參數(shù)

const login = ctx => {
    const req = ctx.request.body;
    const userName = req.userName;
    const expires = Date.now() + 3600000; // 設(shè)置超時時間為一小時后
    
    var payload = { 
        iss: userName,
        exp: expires
    };
    const Token = jwt.encode(payload, secret);
    ctx.response.body = {
        data: Token,
        msg: "登陸成功"
    };
}

// 將公共邏輯方法放到中間件中處理
app.use((ctx, next)=> {
    const headers = ctx.request.headers;
    if(ctx.method === "OPTIONS") {
        ctx.set("Access-Control-Allow-Origin", "*");
        ctx.set("Access-Control-Allow-Headers", "Authorization");
        ctx.status = 204;
    } else {
        next();
    }
})
app.use(route.post("/login", login));

app.listen(3200, () => {
    console.log("啟動成功");
});

上述示例代碼地址

@koa/cors是怎么實現(xiàn)的
"use strict";

const vary = require("vary");

/**
 * CORS middleware
 *
 * @param {Object} [options]
 *  - {String|Function(ctx)} origin `Access-Control-Allow-Origin`, default is request Origin header
 *  - {String|Array} allowMethods `Access-Control-Allow-Methods`, default is "GET,HEAD,PUT,POST,DELETE,PATCH"
 *  - {String|Array} exposeHeaders `Access-Control-Expose-Headers`
 *  - {String|Array} allowHeaders `Access-Control-Allow-Headers`
 *  - {String|Number} maxAge `Access-Control-Max-Age` in seconds
 *  - {Boolean} credentials `Access-Control-Allow-Credentials`
 *  - {Boolean} keepHeadersOnError Add set headers to `err.header` if an error is thrown
 * @return {Function} cors middleware
 * @api public
 */
module.exports = function (options) {
    const defaults = {
        allowMethods: "GET,HEAD,PUT,POST,DELETE,PATCH",
    };
    // 默認的配置項和使用時設(shè)置的options進行一個融合
    options = Object.assign({}, defaults, options);

    // 因為函數(shù)的一些參數(shù),exposeHeaders,allowMethods,allowHeaders的形式既可以是String,也可以是Array類型,
    // 如果是Array類型,也轉(zhuǎn)換為用逗號分隔的字符串。
    if (Array.isArray(options.exposeHeaders)) {
        options.exposeHeaders = options.exposeHeaders.join(",");
    }

    if (Array.isArray(options.allowMethods)) {
        options.allowMethods = options.allowMethods.join(",");
    }

    if (Array.isArray(options.allowHeaders)) {
        options.allowHeaders = options.allowHeaders.join(",");
    }

    if (options.maxAge) {
        options.maxAge = String(options.maxAge);
    }

    options.credentials = !!options.credentials;
    options.keepHeadersOnError = options.keepHeadersOnError === undefined || !!options.keepHeadersOnError;

    return async function cors(ctx, next) {
        // If the Origin header is not present terminate this set of steps.
        // The request is outside the scope of this specification.
        const requestOrigin = ctx.get("Origin");

        // Always set Vary header
        // https://github.com/rs/cors/issues/10
        ctx.vary("Origin");
        // 如果請求頭不存在 origin,則直接跳出該中間件,執(zhí)行下一個中間件
        if (!requestOrigin) return await next();

        // 對origin參數(shù)的不同類型做一個處理
        let origin;
        if (typeof options.origin === "function") {
            origin = options.origin(ctx);
            if (origin instanceof Promise) origin = await origin;
            if (!origin) return await next();
        } else {
            origin = options.origin || requestOrigin;
        }

        const headersSet = {};

        function set(key, value) {
            ctx.set(key, value);
            headersSet[key] = value;
        }
        /**
        * 非OPTIONS請求的處理
        * 
        */
       
        if (ctx.method !== "OPTIONS") {
            // Simple Cross-Origin Request, Actual Request, and Redirects
            set("Access-Control-Allow-Origin", origin);

            if (options.credentials === true) {
                set("Access-Control-Allow-Credentials", "true");
            }

            if (options.exposeHeaders) {
                set("Access-Control-Expose-Headers", options.exposeHeaders);
            }

            if (!options.keepHeadersOnError) {
                return await next();
            }
            try {
                return await next();
            } catch (err) {
                const errHeadersSet = err.headers || {};
                const varyWithOrigin = vary.append(errHeadersSet.vary || errHeadersSet.Vary || "", "Origin");
                delete errHeadersSet.Vary;

                err.headers = Object.assign({}, errHeadersSet, headersSet, {
                    vary: varyWithOrigin
                });

                throw err;
            }
        } else {
            // Preflight Request

            // If there is no Access-Control-Request-Method header or if parsing failed,
            // do not set any additional headers and terminate this set of steps.
            // The request is outside the scope of this specification.
            if (!ctx.get("Access-Control-Request-Method")) {
                // this not preflight request, ignore it
                return await next();
            }

            ctx.set("Access-Control-Allow-Origin", origin);

            if (options.credentials === true) {
                ctx.set("Access-Control-Allow-Credentials", "true");
            }

            if (options.maxAge) {
                ctx.set("Access-Control-Max-Age", options.maxAge);
            }

            if (options.allowMethods) {
                ctx.set("Access-Control-Allow-Methods", options.allowMethods);
            }

            let allowHeaders = options.allowHeaders;
            if (!allowHeaders) {
                allowHeaders = ctx.get("Access-Control-Request-Headers");
            }
            if (allowHeaders) {
                ctx.set("Access-Control-Allow-Headers", allowHeaders);
            }

            ctx.status = 204;
        }
    };
};

以上是 @koa/cors V3.0.0的源碼實現(xiàn),如果你真正理解的CORS,看源碼的邏輯就會非常輕松。

主要是分兩個邏輯來處理,有預(yù)檢請求的和沒有預(yù)檢請求的。

對于非OPTIONS請求的處理,要根據(jù)情況加上 Access-Control-Allow-Origin,Access-Control-Allow-Credentials,Access-Control-Expose-Headers這三個響應(yīng)頭部;

對于OPTIONS請求(預(yù)檢請求)的處理,要根據(jù)情況加上 Access-Control-Allow-OriginAccess-Control-Allow-Credentials,Access-Control-Max-AgeAccess-Control-Allow-Methods,Access-Control-Allow-Headers這幾個響應(yīng)頭部;

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/104954.html

相關(guān)文章

  • koa搭建node服務(wù)

    摘要:介紹使用搭建服務(wù)并連接返回前端數(shù)據(jù)項目初始化首先保證你的環(huán)境已經(jīng)就緒創(chuàng)建項目文件夾創(chuàng)建文件夾,在文件夾中右鍵在此處運行命令行運行安裝依賴創(chuàng)建服務(wù)在項目文件中創(chuàng)建一個文件在項目跟目錄運行瀏覽器地址輸入返回表示一次對話的上下文包括請求 介紹 使用koa搭建node服務(wù) 并連接mongodb返回前端數(shù)據(jù)git https://gitee.com/wjj0720/koa... 項目初始化 首...

    biaoxiaoduan 評論0 收藏0
  • vue+koa2+token登陸驗證

    摘要:用搭建前端項目用搭建后臺,給前端提供數(shù)據(jù)訪問接口項目結(jié)構(gòu)用搭建的項目,紅色框中是新建的文件夾用于存放剩下的文件在寫項目中慢慢增加,最初就是這樣的之后將項目跑起來,看一下有沒有問題這里就當(dāng)作沒有問題前端這里選用和搭配這里采用的是的完整 koa2+vue 用vue-cli搭建前端項目 用koa2搭建后臺,給前端提供數(shù)據(jù)訪問接口 項目結(jié)構(gòu) showImg(https://segmentf...

    econi 評論0 收藏0
  • 基于koajs的一個簡易Excel生成服務(wù)

    摘要:起因運營人員需要將后臺的表格導(dǎo)出成,由于后端的同學(xué)忙于其他事情,想著是不是可以自己做一個服務(wù)來生成。另外再搭配就可以提供一個允許跨域請求的服務(wù)。這樣一個簡單的接口就寫完了,只要調(diào)用傳入和就可以生成文檔。 起因 運營人員需要將后臺的表格導(dǎo)出成Excel,由于后端的同學(xué)忙于其他事情,想著是不是可以自己做一個服務(wù)來生成。了解到有node-xlsx這樣的工具以后就開工了。 框架 后臺選用了ko...

    qylost 評論0 收藏0
  • 通過koa2和Promise.race()構(gòu)造一個超時取消的ajax。

    摘要:上說你可以使用構(gòu)造函數(shù)創(chuàng)建一個新的對象。使用對象完成與請求的通信。服務(wù)端使用重要的點在于不能直接使用這樣返回給前端會直接報錯。前端的代碼要注意的第三個參數(shù)設(shè)置成將請求設(shè)置為異步,然后由于超時會取消請求,所以這里根本不需要來顯式的取消請求 MDN上說: 你可以使用AbortController.AbortController()構(gòu)造函數(shù)創(chuàng)建一個新的AbortController對象。 使...

    e10101 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<