itns-sidechain/lib/workers/master.js

275 lines
5.3 KiB
JavaScript
Raw Normal View History

2016-11-19 02:27:26 -08:00
/*!
* workers.js - worker processes for bcoin
* Copyright (c) 2014-2015, Fedor Indutny (MIT License)
2017-02-03 22:47:26 -08:00
* Copyright (c) 2014-2017, Christopher Jeffrey (MIT License).
2016-11-19 02:27:26 -08:00
* https://github.com/bcoin-org/bcoin
*/
'use strict';
2016-12-02 18:14:17 -08:00
var assert = require('assert');
2016-11-19 02:27:26 -08:00
var EventEmitter = require('events').EventEmitter;
2016-11-19 10:45:31 -08:00
var util = require('../utils/util');
2016-11-19 02:27:26 -08:00
var Network = require('../protocol/network');
var jobs = require('./jobs');
var Parser = require('./parser-client');
var Framer = require('./framer');
2016-12-02 11:06:01 -08:00
var packets = require('./packets');
2016-11-19 10:45:31 -08:00
var global = util.global;
2016-11-19 02:27:26 -08:00
var server;
/**
* Represents the master process.
2017-02-03 22:47:26 -08:00
* @alias module:workers.Master
2016-11-19 02:27:26 -08:00
* @constructor
*/
function Master() {
if (!(this instanceof Master))
return new Master();
EventEmitter.call(this);
this.framer = new Framer();
this.parser = new Parser();
2016-12-02 18:14:17 -08:00
this.env = {};
this.listening = false;
2016-12-02 18:34:29 -08:00
this.color = false;
2016-11-19 02:27:26 -08:00
this._init();
}
2016-11-19 10:45:31 -08:00
util.inherits(Master, EventEmitter);
2016-11-19 02:27:26 -08:00
/**
* Initialize master. Bind events.
* @private
*/
Master.prototype._init = function _init() {
var self = this;
this.on('data', function(data) {
self.parser.feed(data);
});
2016-12-02 18:14:17 -08:00
this.parser.on('error', function(err) {
self.emit('error', err);
2016-11-19 02:27:26 -08:00
});
this.parser.on('packet', function(packet) {
self.emit('packet', packet);
});
2016-12-02 18:14:17 -08:00
if (util.isBrowser) {
// Web workers
this._initWebWorkers();
} else {
// Child process + pipes
this._initChildProcess();
}
};
/**
* Initialize master (web workers).
* @private
*/
Master.prototype._initWebWorkers = function _initWebWorkers() {
var self = this;
global.onerror = function onerror(err) {
self.emit('error', err);
};
global.onmessage = function onmessage(event) {
var data;
if (typeof event.data !== 'string') {
data = event.data.buf;
data.__proto__ = Buffer.prototype;
} else {
data = new Buffer(event.data, 'hex');
}
self.emit('data', data);
};
};
/**
* Initialize master (node.js).
* @private
*/
Master.prototype._initChildProcess = function _initChildProcess() {
var self = this;
process.stdin.on('data', function(data) {
self.emit('data', data);
});
// Nowhere to send these errors:
process.stdin.on('error', util.nop);
process.stdout.on('error', util.nop);
process.stderr.on('error', util.nop);
process.on('uncaughtException', function(err) {
self.send(new packets.ErrorPacket(err));
util.nextTick(function() {
process.exit(1);
});
});
2016-11-19 02:27:26 -08:00
};
/**
* Set primary network.
* @param {NetworkType|Network} network
*/
Master.prototype.set = function set(network) {
return Network.set(network);
};
/**
* Send data to worker.
* @param {Buffer} data
* @returns {Boolean}
*/
Master.prototype.write = function write(data) {
2016-11-19 10:45:31 -08:00
if (util.isBrowser) {
2016-11-19 02:27:26 -08:00
if (global.postMessage.length === 2) {
data.__proto__ = Uint8Array.prototype;
global.postMessage({ buf: data }, [data]);
} else {
global.postMessage(data.toString('hex'));
}
return true;
}
return process.stdout.write(data);
};
/**
* Frame and send a packet.
2016-12-02 11:06:01 -08:00
* @param {Packet} packet
2016-11-19 02:27:26 -08:00
* @returns {Boolean}
*/
2016-12-02 11:06:01 -08:00
Master.prototype.send = function send(packet) {
return this.write(this.framer.packet(packet));
2016-11-19 02:27:26 -08:00
};
/**
* Emit an event on the worker side.
* @param {String} event
* @param {...Object} arg
* @returns {Boolean}
*/
Master.prototype.sendEvent = function sendEvent() {
var items = new Array(arguments.length);
var i;
for (i = 0; i < items.length; i++)
items[i] = arguments[i];
2016-12-02 11:06:01 -08:00
return this.send(new packets.EventPacket(items));
2016-11-19 02:27:26 -08:00
};
/**
* Destroy the worker.
*/
Master.prototype.destroy = function destroy() {
2016-11-19 10:45:31 -08:00
if (util.isBrowser)
2016-11-19 02:27:26 -08:00
return global.close();
return process.exit(0);
};
/**
* Write a message to stdout in the master process.
* @param {Object|String} obj
* @param {...String} args
*/
Master.prototype.log = function log() {
var items = new Array(arguments.length);
2016-12-02 18:34:29 -08:00
var i, text;
2016-11-19 02:27:26 -08:00
for (i = 0; i < items.length; i++)
items[i] = arguments[i];
2016-12-02 18:34:29 -08:00
text = util.format(items, this.color);
this.send(new packets.LogPacket(text));
2016-11-19 02:27:26 -08:00
};
/**
* Listen for messages from master process (only if worker).
* @param {Object} env
* @returns {Master}
*/
Master.prototype.listen = function listen(env) {
var self = this;
2016-12-02 18:14:17 -08:00
assert(!this.listening, 'Already listening.');
this.env = env;
this.listening = true;
2016-12-02 18:34:29 -08:00
this.color = +env.BCOIN_WORKER_ISTTY === 1;
2016-12-02 18:14:17 -08:00
2016-12-04 18:59:49 -08:00
this.set(env.BCOIN_WORKER_NETWORK);
2016-11-19 02:27:26 -08:00
2016-11-19 10:45:31 -08:00
util.log = this.log.bind(this);
util.error = util.log;
2016-11-19 02:27:26 -08:00
this.on('error', function(err) {
2016-12-02 11:06:01 -08:00
self.send(new packets.ErrorPacket(err));
2016-11-19 02:27:26 -08:00
});
this.on('packet', function(packet) {
2016-12-02 18:14:17 -08:00
try {
self.handlePacket(packet);
} catch (e) {
self.emit('error', e);
2016-11-19 02:27:26 -08:00
}
});
2016-12-02 18:14:17 -08:00
return this;
};
/**
* Handle packet.
* @private
* @param {Packet}
*/
Master.prototype.handlePacket = function handlePacket(packet) {
var result;
switch (packet.cmd) {
case packets.types.EVENT:
this.emit('event', packet.items);
this.emit.apply(this, packet.items);
break;
case packets.types.ERROR:
this.emit('error', packet.error);
break;
default:
result = jobs.execute(packet);
result.id = packet.id;
this.send(result);
break;
}
2016-11-19 02:27:26 -08:00
};
/*
* Expose
*/
server = new Master();
2016-11-19 10:45:31 -08:00
if (util.isBrowser)
2016-11-19 02:27:26 -08:00
global.master = server;
module.exports = server;