Koa 源碼淺析

本文圍繞koa服務從啓動,處處理請求再到回覆響應這個過程對源碼進行簡單的解析node

在koa中ctx是貫穿整個請求過程的,它是此次請求原信息的承載體,能夠從ctx上獲取到request、response、cookie等,方便咱們進行後續的計算處理。 ctx在實現上本來就是一個空對象,在koa服務起來時,往上掛載了不少對象和方法。固然開發者也能夠自定義掛載的方法。 在context.js文件中對ctx初始化了一些內置的對象和屬性,包括錯誤處理,設置cookie。git

get cookies() {
    if (!this[COOKIES]) {
      this[COOKIES] = new Cookies(this.req, this.res, {
        keys: this.app.keys,
        secure: this.request.secure
      });
    }
    return this[COOKIES];
  },

  set cookies(_cookies) {
    this[COOKIES] = _cookies;
  }
複製代碼

相對於這種寫法,還有另一種較爲優雅的掛載方法。github

// ./context.js

delegate(proto, 'response')
  .method('attachment')
  .method('redirect')
  .method('remove')
  .method('vary')
  .method('set')
  .method('append')
  .method('flushHeaders')
  .access('status')
  .access('message')
  .access('body')
  .access('length')
  .access('type')
  .access('lastModified')
  .access('etag')
  .getter('headerSent')
  .getter('writable');

複製代碼

delegate方法的做用是將其餘對象的方法或者屬性掛載在指定對象上,在這裏就是proto,也就是最初的ctx對象,屬性提供方就是第二個參數"response"。 method是代理方法,getter代理get,access代理set和get. 看delegate如何實現:json

function Delegator(proto, target) {
  if (!(this instanceof Delegator)) return new Delegator(proto, target);
  this.proto = proto;
  this.target = target;
  this.methods = [];
  this.getters = [];
  this.setters = [];
  this.fluents = [];
}

Delegator.prototype.method = function(name){
  var proto = this.proto;
  var target = this.target;
  this.methods.push(name);

  proto[name] = function(){
    return this[target][name].apply(this[target], arguments);
  };

  return this;
};

複製代碼

delegate中的method的實現是在調用原屬性上指定方法時,轉而調用提供方的方法。這裏能夠發現提供方也被收在了this上,這裏的不直接傳入一個對象而是將該對象賦值在原對象上的緣由,我想應該是存放一個副本在原對象上,這樣能夠經過原對象直接訪問到提供屬性的對象。數組

./context.js中使用delegates爲ctx賦值的過程並不完整,由於這裏的屬性提供方雖然是request和response, 可是是從./application.js createContext方法中傳入,這樣delegates纔算完成了工做promise

到這裏咱們就能夠看下平時用koa時常走的流程。服務器

const Koa = require('koa');
const app = new Koa();

// response
app.use(ctx => {
  ctx.body = 'Hello Koa';
});

app.listen(3000);
複製代碼

基本上就是分爲三步,實例化Koa,註冊中間件再監聽端口, 這裏正常能讓koa服務或者說一個http服務起的來的操做實際上是在app.listen(...args)裏,是否是和想象中的有點差距, 看下源碼實現。cookie

// ./application.js

  ...
  listen(...args) {
    debug('listen');
    const server = http.createServer(this.callback());
    return server.listen(...args);
  }
  ...
複製代碼

在listen方法裏使用了http模塊的createServer方法來啓動http服務,這裏至關因而聲明瞭一個http.Server實例,該實例也繼承於EventEmitter,是一個事件類型的服務器,並監聽了該實例的request事件,意爲當客戶端有請求發過來的時候,這個事件將會觸發,等價於以下代碼app

var http = require("http");
var server = new http.Server();

server.on("request", function(req, res){
    // handle request
});

server.listen(3000);
複製代碼

這個事件有兩個參數req和res,也就是此次事件的請求和響應信息。有點扯遠了,回到koa源碼, 處理req和res參數的任務就交給了this.callback()的返回值來作,繼續看callback裏作了什麼koa

// 去除了一些不影響主流程的處理代碼

  callback() {
    const fn = compose(this.middleware);
    const handleRequest = (req, res) => {
      const ctx = this.createContext(req, res);
      return this.handleRequest(ctx, fn);
    };
    return handleRequest;
  }

  handleRequest(ctx, fnMiddleware) {
    const handleResponse = () => respond(ctx);
    return fnMiddleware(ctx).then(handleResponse).catch(onerror);
  }
複製代碼

callback返回一個函數由他來處理req和res,這個函數內部作了兩件事, 這兩件事分別在koa服務的初始化和響應時期完成,上述代碼中compose中間件就是在服務初始化完成, 而當request事件觸發時,該事件會由callback返回的handleRequest方法處理,這個方法保持了對fn,也就是初始化事後中間件的應用, handleRequest先會初始化貫穿整個事件的ctx對象,這個時候就能夠將ctx以此走入到各個中間件中處理了。

能夠說koa到這裏主流程已經走一大半了,讓咱們理一理通過簡單分析過的源碼能夠作到哪一個地步(忽略錯誤處理)

  • 響應http請求 √
  • 生成ctx對象 √
  • 運用中間件 √
  • 返回請求 ×

如上咱們已經能夠作到將響應進入readly狀態,但尚未返回響應的能力,後續會說道。在前三個過程當中有兩個點須要注意,ctx和middleware,下面咱們依次深刻學習下這兩個關鍵點。

ctx是貫穿整個request事件的對象,它上面掛載瞭如req和res這種描述該次事件信息的屬性,開發者也能夠根據本身喜愛,經過前置中間件掛載一些屬性上去。 ctx在koa實例createContext方法上建立並被完善,再由callback返回的handleRequest也就是響應request的處理函數消費。看下createContext源碼

createContext(req, res) {
    const context = Object.create(this.context);
    const request = context.request = Object.create(this.request);
    const response = context.response = Object.create(this.response);
    context.app = request.app = response.app = this;
    
    context.req = request.req = response.req = req;
    context.res = request.res = response.res = res;

    request.ctx = response.ctx = context;
    
    request.response = response;
    response.request = request;
    
    context.originalUrl = request.originalUrl = req.url;
    context.state = {};
    return context;
  }

複製代碼

前三行依次聲明瞭context、request和response,分別繼承於koa實例的三個靜態屬性,這三個靜態屬性由koa本身定義,在上面有一些快捷操做方法,好比在Request靜態類上能夠獲取經過query獲取查詢參數,經過URL解析url等,能夠理解爲request的工具庫,Response和Context同理。res和rep是node的原生對象,還記得嗎,這兩個參數是由http.Server()實例觸發request事件帶來的入參。 res是http.incomingMessage的實例而rep繼承於http.ServerResponse, 貼一張圖。

al

箭頭指向說明了從屬關係,有五個箭頭指向ctx表面ctx上有五個這樣的的屬性,能夠很清楚看到ctx上各個屬性之間的關係。

接下來咱們再來看看koa中的中間件,在koa中使用use方法能夠註冊中間件.

use(fn) {
    if (typeof fn !== 'function') throw new TypeError('middleware must be a function!');
    if (isGeneratorFunction(fn)) {
      deprecate('Support for generators will be removed in v3. ' +
                'See the documentation for examples of how to convert old middleware ' +
                'https://github.com/koajs/koa/blob/master/docs/migration.md');
      fn = convert(fn);
    }
    debug('use %s', fn._name || fn.name || '-');
    this.middleware.push(fn);
    return this;
  }
複製代碼

兩件事,統一中間件格式,再將中間件推入中間件數組中。 在koa2.0之後middleware都是使用async/await語法,使用generator function也是能夠的,2.0之後版本內置了koa-convert,它能夠根據 fn.constructor.name == 'GeneratorFunction' check here.來判斷是legacyMiddleware仍是modernMiddleware,並根據結果來作相應的轉換。 koa-convert的核心使用是co這個庫,它提供了一個自動的執行器,而且返回的是promise,generator function有了這兩個特性也就能夠直接和async函數一塊兒使用了。

回到koa源碼來,callback中是這樣處理中間件數組的

const fn = compose(this.middleware);
複製代碼

這裏的compose也就是koa-compose模塊,它負責將全部的中間件串聯起來,並保證執行順序。經典的洋蔥圈圖:

koa mw

koa-compose模塊的介紹只有簡單的一句話

Compose the given middleware and return middleware.

言簡意賅,就是組合中間件。貼上源碼

function compose (middleware) {
  return function (context, next) {
    let index = -1
    return dispatch(0)
    function dispatch (i) {
      if (i <= index) return Promise.reject(new Error('next() called multiple times'))
      index = i
      let fn = middleware[i]
      if (i === middleware.length) fn = next
      if (!fn) return Promise.resolve()
      try {
        return Promise.resolve(fn(context, dispatch.bind(null, i + 1)));
      } catch (err) {
        return Promise.reject(err)
      }
    }
  }
}
複製代碼

compose先存入中間件數組,從第一個開始執行依次resolve到最後一個,中間件函數簽名爲(ctx,next)=>{},在內部調用next就會間接喚起下一個中間件,也就是執行dispatch.bind(null, i + 1),中間件執行順序以下(網上扒下來的圖)。

mw

圖上是遇到yield,和執行next同理。 也不是全部的中間件都須要next,在最後一個middleware執行完畢後能夠不調用next,由於這個時候已經走完了全部中間件的前置邏輯。固然這裏調用next也是能夠的,是爲了在全部前置邏輯執行完後有一個回調。咱們單獨使用koa-compose:

const compose = require("koa-compose");
let _ctx = {
    name: "ctx"
};

const mw_a = async function (ctx, next) {
    console.log("this is step 1");
    ctx.body = "lewis";
    await next();
    console.log("this is step 4");
}

const mw_b = async function (ctx, next) {
    console.log("this is step 2");
    await next();
    console.log("this is step 3");
}

const fn = compose([mw_a, mw_b]);

fn(_ctx, async function (ctx) {
    console.log("Done", ctx)
});

// => 
// 
// this is 1
// this is 2
// Done {name: "ctx", body: "lewis"}
// this is 3
// this is 4
複製代碼

compose返回的函數接受的參數不光是ctx,還能夠接受一個函數做爲走完全部中間件前置邏輯後的回調。有特殊需求的開發者能夠關注一下。 固然整個中間件執行完後會返回一個resolve狀態的promise,在這個回調中koa用來告訴客戶端「響應已經處理完畢,請查收」,這個時候客戶端才結束等待狀態,這個過程的源碼:

// handleRequest 的返回值
// 當中間件已經處理完畢後,交由handleResponse也就是respond方法來最後處理ctx
const handleResponse = () => respond(ctx);
fnMiddleware(ctx).then(handleResponse).catch(onerror);

/** * 交付response */
function respond(ctx) {
  // allow bypassing koa
  if (false === ctx.respond) return;

  const res = ctx.res;
  if (!ctx.writable) return;

  let body = ctx.body;
  const code = ctx.status;

  // ignore body
  if (statuses.empty[code]) {
    // strip headers
    ctx.body = null;
    return res.end();
  }

  if ('HEAD' == ctx.method) {
    if (!res.headersSent && isJSON(body)) {
      ctx.length = Buffer.byteLength(JSON.stringify(body));
    }
    return res.end();
  }

  // status body
  if (null == body) {
    body = ctx.message || String(code);
    if (!res.headersSent) {
      ctx.type = 'text';
      ctx.length = Buffer.byteLength(body);
    }
    return res.end(body);
  }

  // responses
  if (Buffer.isBuffer(body)) return res.end(body);
  if ('string' == typeof body) return res.end(body);
  if (body instanceof Stream) return body.pipe(res);

  // body: json
  body = JSON.stringify(body);
  if (!res.headersSent) {
    ctx.length = Buffer.byteLength(body);
  }
  res.end(body);
}

複製代碼

以上代碼對各類款式的status和ctx.body作了相應的處理,最關鍵的仍是這一句res.end(body),它調用了node原生response的end方法,來告訴服務器本次的請求以回傳body來結束,也就是告訴服務器此響應的全部報文頭及報文體已經發出,服務器在此調用後認爲這條信息已經發送完畢,而且這個方法必須對每一個響應調用一次。

總結

至此,koa整個流程已經走通,能夠看到koa的關鍵點集中在ctx對象和中間件的運用上。 經過delegate將原生res和req的方法屬性代理至ctx上,再掛載koa內置的Request和Reponse,提供koa風格操做底層res和req的實現途徑和獲取請求信息的工具方法。 中間件則是使用koa-compose庫將中間件串聯起來執行,並具備能夠逆回執行的能力。

相關文章
相關標籤/搜索