pomelo-rpc是pomelo項目底層的rpc框架,提供了一個多服務器進程間進行rpc調用的基礎設施。 pomelo-rpc分為客戶端和服務器端兩個部分。 客戶端部分提供了rpc代理生成,消息路由和網絡通訊等功能,并支持動態添加代理和遠程服務器配置。 服務器端提供了遠程服務暴露,請求派發,網絡通訊等功能。
本文主要分析pomelo-rpc中server部分的實現原理以及運行邏輯。
server初始化
Server.create(opts)
創建一個rpc server實例。根據配置信息加載遠程服務代碼,并生成底層acceptor。
首先看create部分源碼:
module.exports.create = function(opts) {
if(!opts || !opts.port || opts.port < 0 || !opts.paths) {
throw new Error('opts.port or opts.paths invalid.');
}
// 根據paths加載遠程服務
var services = loadRemoteServices(opts.paths, opts.context);
opts.services = services;
var gateway = Gateway.create(opts);
return gateway;
};
首先loadRemoteServices()
方法根據opts參數中的paths加載遠程服務。
其中pomelo paths的格式類似,pomelo根據約定封裝,詳見
https://github.com/NetEase/pomelo/blob/master/lib/components/remote.js
[
{
"namespace": "user",
"serverType": "test",
"path": "/data/pomelo/app/servers/test/remote/"
},
{
"namespace": "sys",
"serverType": "test",
"path": "/data/pomelo/pomelo/lib/common/remote/backend/"
}
]
var loadRemoteServices = function(paths, context) {
var res = {}, item, m;
for(var i=0, l=paths.length; i<l; i++) {
item = paths[i];
// Loader是pomelo-loader,用來加載pomelo handler和remote服務
// 關于Loader的細節可以參考https://github.com/NetEase/pomelo-loader/
m = Loader.load(item.path, context);
if(m) {
createNamespace(item.namespace, res);
for(var s in m) {
res[item.namespace][s] = m[s];
}
}
}
return res;
};
var createNamespace = function(namespace, proxies) {
proxies[namespace] = proxies[namespace] || {};
};
loadRemoteServices()
最終得到的services對象類似如下結構的數據:
{
"user": {
"testRemote": require("/path/to/testRemote.js"),
"test2Remote": require("/path/to/test2Remote.js"),
...
},
"sys": {
"msgRemote": require("/data/pomelo/pomelo/lib/common/remote/backend/msgRemote.js")
}
}
Gateway.create(opts)
創建gateway對象,并初始化dispatcher和acceptor。
gateway的構造方法:
var Gateway = function(opts) {
EventEmitter.call(this);
this.opts = opts || {};
this.port = opts.port || 3050;
this.started = false;
this.stoped = false;
this.services = opts.services;
if(!!this.opts.reloadRemotes) {
// 如果remote配置reloadRemotes=true的話,gateway會通過`fs.watch()`來檢測remote文件的變化,
// 然后通過pomelo-loader重新require對應的remote文件來實現remote的熱更新。
watchServices(this, dispatcher);
}
var self = this;
this.acceptors = {};
// __defineGetter__的用法可以參考
// https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Object/__defineGetter__
this.acceptors.__defineGetter__('tcp', utils.load.bind(null, '../rpc-server/acceptors/tcp-acceptor'));
this.acceptors.__defineGetter__('ws', utils.load.bind(null,'../rpc-server/acceptors/ws-acceptor'));
if(!!opts.acceptorName && opts.acceptorName === 'ws') {
this.acceptorFactory = this.acceptors.ws;
} else {
// 默認是使用tcp協議的acceptors
this.acceptorFactory = this.acceptors.tcp;
}
if(!!opts.acceptorFactory) {
this.acceptorFactory = opts.acceptorFactory;
}
// Dispatcher初始化,沒什么好看的
var dispatcher = new Dispatcher(this.services);
// acceptorFactory.create初始化acceptor,也沒什么好看的
this.acceptor = this.acceptorFactory.create(opts, function(tracer, msg, cb) {
dispatcher.route(tracer, msg, cb);
});
};
Dispatcher.route(tracer, msg, cb)
提供路由服務,路由消息到對應的service。
/**
* route the msg to appropriate service object
*
* @param msg msg package {service:serviceString, method:methodString, args:[]}
* @param services services object collection, such as {service1: serviceObj1, service2: serviceObj2}
* @param cb(...) callback function that should be invoked as soon as the rpc finished
*/
pro.route = function(tracer, msg, cb) {
tracer.info('server', __filename, 'route', 'route messsage to appropriate service object');
// this.services 就是loadRemoteServices()加載的本地remote service
var namespace = this.services[msg.namespace];
if(!namespace) {
tracer.error('server', __filename, 'route', 'no such namespace:' + msg.namespace);
utils.invokeCallback(cb, new Error('no such namespace:' + msg.namespace));
return;
}
var service = namespace[msg.service];
if(!service) {
tracer.error('server', __filename, 'route', 'no such service:' + msg.service);
utils.invokeCallback(cb, new Error('no such service:' + msg.service));
return;
}
var method = service[msg.method];
if(!method) {
tracer.error('server', __filename, 'route', 'no such method:' + msg.method);
utils.invokeCallback(cb, new Error('no such method:' + msg.method));
return;
}
var args = msg.args.slice(0);
args.push(cb);
// 調用remote service方法
method.apply(service, args);
};
至此,pomelo-rpc的server初始化工作就完成了,接下來就是啟動server。
server啟動
Gateway.start()
pro.start = function() {
if(this.started) {
throw new Error('gateway already start.');
}
this.started = true;
var self = this;
this.acceptor.on('error', self.emit.bind(self, 'error'));
this.acceptor.on('closed', self.emit.bind(self, 'closed'));
// 啟動acceptor并監聽port端口
this.acceptor.listen(this.port);
};
Acceptor.listen(port)
啟動acceptor并監聽port端口,這里以tcp acceptor為例。
pro.listen = function(port) {
//check status
if(!!this.inited) {
utils.invokeCallback(this.cb, new Error('already inited.'));
return;
}
this.inited = true;
var self = this;
this.server = net.createServer();
this.server.listen(port);
this.server.on('error', function(err) {
logger.error('rpc server is error: %j', err.stack);
self.emit('error', err, this);
});
// 處理鏈接請求
this.server.on('connection', function(socket) {
// 設置socket自增id
socket.id = self.socketId++;
// 保存socket句柄到本地sockets
self.sockets[socket.id] = socket;
// 設置socket的消息編解碼處理器,處理各種類型的消息(ping,pong,msg等)
// pkgSize默認-1,不限制消息長度
socket.composer = new Composer({maxLength: self.pkgSize});
// 心跳超時檢測timer
self.timer[socket.id] = null;
// 啟動心跳檢測,如果heartbeat timeout間隔內沒有心跳包過來,就斷開連接。
// 下面socket.composer.on('data' ...中的self.heartbeat才是真正連接后的心跳
// 相當于這里只是處理初始連接時的心跳
self.heartbeat(socket.id);
socket.on('data', function(data) {
// 調用composer解析數據流
// feed讀完數據后會emit data事件
socket.composer.feed(data);
});
// 接收feed emit的data事件
socket.composer.on('data', function(data) {
self.heartbeat(socket.id);
if(data[0] === PING) {
//incoming::ping,response with PONG
socket.write(socket.composer.compose(PONG));
} else {
try {
var pkg = JSON.parse(data.toString('utf-8', 1));
var id = null;
// 處理消息
if(pkg instanceof Array) {
processMsgs(socket, self, pkg, id);
} else {
processMsg(socket, self, pkg, id);
}
} catch(err) { //json parse exception
if(err) {
// 重置編解器狀態
socket.composer.reset();
logger.error(err);
}
}
}
});
socket.on('error', function(err) {
logger.error('[pomelo-rpc] tcp socket error: %j', err);
});
socket.on('close', function() {
logger.error('[pomelo-rpc] tcp socket close: %s', socket.id);
delete self.sockets[socket.id];
delete self.msgQueues[socket.id];
if(self.timer[socket.id]){
clearTimeout(self.timer[socket.id]);
}
delete self.timer[socket.id];
});
});
// 定時flush緩存的消息數據
if(this.bufferMsg) {
this._interval = setInterval(function() {
flush(self);
}, this.interval);
}
};
processMsg()
處理rpc消息。實際就是調用dispatcher.route()來根據namespace和service來調用對應的remote method。
var processMsg = function(socket, acceptor, pkg, id) {
var tracer = new Tracer(acceptor.rpcLogger, acceptor.rpcDebugLog, pkg.remote, pkg.source, pkg.msg, pkg.traceId, pkg.seqId);
tracer.info('server', __filename, 'processMsg', 'tcp-acceptor receive message and try to process message');
// 實際就是調用dispatcher.route()
acceptor.cb.call(null, tracer, pkg.msg, function() {
var args = Array.prototype.slice.call(arguments, 0);
for(var i=0, l=args.length; i<l; i++) {
if(args[i] instanceof Error) {
args[i] = cloneError(args[i]);
}
}
var resp;
if(tracer.isEnabled) {
resp = {traceId: tracer.id, seqId: tracer.seq, source: tracer.source, id: pkg.id, resp: Array.prototype.slice.call(args, 0)};
}
else {
resp = {id: pkg.id, resp: Array.prototype.slice.call(args, 0)};
}
if(acceptor.bufferMsg) {
// 如果開啟緩沖消息則將結果緩存到隊列,由_interval定時flush,用以減少網絡io次數
enqueue(socket, acceptor, resp);
} else {
// 發送封裝remote方法的返回結果
socket.write(socket.composer.compose(RES_TYPE, JSON.stringify(resp), id));
}
});
};
總結
通過對pomelo-rpc server部分代碼的分析,可以很清晰了解到server端主要作用就是暴露遠程服務(remote目錄下的.js文件)、根據消息的namespace和service信息派發到對應的remote服務處理、基于tcp/ws來提供底層的網絡通訊。
在日常開發中,新手很容易遇到rpc調用超時的情況,一般來看都是因為某些remote方法沒有正確回調或者根本漏寫了回調。