const Koa = require('koa'); //引入koa
const app = new Koa();// 实例化一个对象
// 中间件一
app.use(ctx => {
ctx.body = 'Hello Koa'; // 设置响应体
});
app.listen(3000); //在3000端口上启动koa服务
两个核心方法:
koa的源码清晰明了,有四个见名知义的文件:
listen(...args) {
debug('listen');
const server = http.createServer(this.callback());
return server.listen(...args);
}
http.createServe()创建了一个server对象,每当有请求时都会执行callback()这个回调方法
callback() {
const fn = compose(this.middleware);
if (!this.listenerCount('error')) this.on('error', this.onerror);
// 这就是原生Node起服务时On('request',(req,res)=>{.....})
const handleRequest = (req, res) => {
const ctx = this.createContext(req, res);
return this.handleRequest(ctx, fn);
};
return handleRequest;
}
callback 返回了一个闭包 handleRequest
handleRequest(ctx, fnMiddleware) {
const res = ctx.res;
res.statusCode = 404;
const onerror = err => ctx.onerror(err);
const handleResponse = () => respond(ctx);
onFinished(res, onerror);
return fnMiddleware(ctx).then(handleResponse).catch(onerror);
}
接收上下文ctx和合成的中间件fn,完成了最后的响应。此处有两个问题:
回过头来慢慢体会compose,渐入佳境
(最高潮),compose()方法是koa、koa-router的核心方法
function compose (middleware) {
if (!Array.isArray(middleware)) throw new TypeError('Middleware stack must be an array!')
for (const fn of middleware) {
if (typeof fn !== 'function') throw new TypeError('Middleware must be composed of functions!')
}
/**
* @param {Object} context
* @return {Promise}
* @api public
*/
return function (context, next) {
// last called middleware #
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, function next () {
return dispatch(i + 1)
}))
} catch (err) {
return Promise.reject(err)
}
}
}
}
去繁从简,去掉参数校验,容错处理等看主流程
function compose (middleware) {
return function (context, next) {
return dispatch(0) //直接执行第一个中间件
function dispatch (i) {
let fn = middleware[i]
if (!fn) return Promise.resolve() //走完最后一个中间件
//执行一个中间件
fn(context, function next () {
return dispatch(i + 1) //递归调用 闭包
})
}
}
}
const chalk = require('chalk')
const Koa = require('./src/tiny-koa')
const util = require('util')
const port = 9001
const app = new Koa()
app.use(async (ctx, next) => {
console.log('1.1')
await next()
console.log('1.2')
})
app.use(async (ctx, next) => {
console.log('2.1')
await next()
console.log('2.2')
})
app.use(async (ctx, next) => {
console.log('3.1')
await next()
console.log('3.2')
})
app.listen(port, _ => {
console.log(chalk.red(`koa server started at ${port}`))
})
app.on('error',error=>{
console.log( util.inspect(error))
})
// process.on('uncaughtException', function (err) {
// console.error('Caught exception: ' +chalk.red(err));
// });
const http = require('http')
const EventEmiter = require('events')
class TinyKoa extends EventEmiter {
constructor() {
super();
this.middlewares = []
this.ctx = {}
}
use(fn) {
if (typeof fn !== 'function') {
throw new Error('the middleware must be a function')
}
this.middlewares.push(fn)
}
listen(...argvs) {
try {
const server = http.createServer(this.onRequest.bind(this))
server.listen(...argvs)
} catch (error) {
this.emit('error', error)
}
}
onRequest(req, res) {
let middlewares = this.middlewares
let ctx = this.ctx
ctx.request = req
ctx.response = res
dispatch(0)
function dispatch(idx) {
try {
let fn = middlewares[idx]
fn && fn(ctx, () => dispatch(idx + 1))
} catch (error) {
this.emit('error', error)
}
}
}
}
module.exports = TinyKoa
onRequest(req, res) {
let middlewares = this.middlewares
let ctx = this.ctx
ctx.request = req
ctx.response = res
dispatch(0)
function dispatch(idx) {
try {
let fn = middlewares[idx]
fn && fn(ctx, () => dispatch(idx + 1))
} catch (error) {
this.emit('error', error)
}
}
}
/**
* Response delegation.
*/
delegate(proto, 'response')
.method('redirect')
.method('remove')
.method('vary')
.method('set')
.access('body')
.access('lastModified')
.access('etag')
/**
* Request delegation.
*/
delegate(proto, 'request')
.method('acceptsEncodings')
.method('accepts')
.method('get')
.access('querystring')
.access('search')
.access('method')
.access('query')
.access('path')
context.js 就是将res,req上的常用属性、方法通过delegate代理到ctx上
Delegator原型上定义有method、access方法,都返回了this,实现了链式调用,jquery也是这样
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 = [];
}
/**
* Delegate method `name`.
*
* @param {String} name
* @return {Delegator} self
* @api public
*/
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;
};
/**
* Delegator accessor `name`.
*
* @param {String} name
* @return {Delegator} self
* @api public
*/
Delegator.prototype.access = function(name){
return this.getter(name).setter(name);
};
ci