195 lines
3.7 KiB
JavaScript
195 lines
3.7 KiB
JavaScript
/*!
|
|
* master.js - master process for bcoin
|
|
* Copyright (c) 2014-2015, Fedor Indutny (MIT License)
|
|
* Copyright (c) 2014-2017, Christopher Jeffrey (MIT License).
|
|
* https://github.com/bcoin-org/bcoin
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
const assert = require('assert');
|
|
const EventEmitter = require('events');
|
|
const util = require('../utils/util');
|
|
const Network = require('../protocol/network');
|
|
const jobs = require('./jobs');
|
|
const Parser = require('./parser');
|
|
const Framer = require('./framer');
|
|
const packets = require('./packets');
|
|
const Parent = require('./parent');
|
|
|
|
/**
|
|
* Represents the master process.
|
|
* @alias module:workers.Master
|
|
* @constructor
|
|
*/
|
|
|
|
function Master() {
|
|
if (!(this instanceof Master))
|
|
return new Master();
|
|
|
|
EventEmitter.call(this);
|
|
|
|
this.parent = new Parent();
|
|
this.framer = new Framer();
|
|
this.parser = new Parser();
|
|
this.listening = false;
|
|
this.color = false;
|
|
|
|
this.init();
|
|
}
|
|
|
|
util.inherits(Master, EventEmitter);
|
|
|
|
/**
|
|
* Initialize master. Bind events.
|
|
* @private
|
|
*/
|
|
|
|
Master.prototype.init = function init() {
|
|
this.parent.on('data', (data) => {
|
|
this.parser.feed(data);
|
|
});
|
|
|
|
this.parent.on('error', (err) => {
|
|
this.emit('error', err);
|
|
});
|
|
|
|
this.parent.on('exception', (err) => {
|
|
this.send(new packets.ErrorPacket(err));
|
|
setTimeout(() => this.destroy(), 1000);
|
|
});
|
|
|
|
this.parser.on('error', (err) => {
|
|
this.emit('error', err);
|
|
});
|
|
|
|
this.parser.on('packet', (packet) => {
|
|
this.emit('packet', packet);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Set environment.
|
|
* @param {Object} env
|
|
*/
|
|
|
|
Master.prototype.setEnv = function setEnv(env) {
|
|
this.color = env.BCOIN_WORKER_ISTTY === '1';
|
|
this.set(env.BCOIN_WORKER_NETWORK);
|
|
};
|
|
|
|
/**
|
|
* 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) {
|
|
return this.parent.write(data);
|
|
};
|
|
|
|
/**
|
|
* Frame and send a packet.
|
|
* @param {Packet} packet
|
|
* @returns {Boolean}
|
|
*/
|
|
|
|
Master.prototype.send = function send(packet) {
|
|
return this.write(this.framer.packet(packet));
|
|
};
|
|
|
|
/**
|
|
* Emit an event on the worker side.
|
|
* @param {String} event
|
|
* @param {...Object} arg
|
|
* @returns {Boolean}
|
|
*/
|
|
|
|
Master.prototype.sendEvent = function sendEvent(...items) {
|
|
return this.send(new packets.EventPacket(items));
|
|
};
|
|
|
|
/**
|
|
* Destroy the worker.
|
|
*/
|
|
|
|
Master.prototype.destroy = function destroy() {
|
|
return this.parent.destroy();
|
|
};
|
|
|
|
/**
|
|
* Write a message to stdout in the master process.
|
|
* @param {Object|String} obj
|
|
* @param {...String} args
|
|
*/
|
|
|
|
Master.prototype.log = function log(...items) {
|
|
const text = util.format(items, this.color);
|
|
this.send(new packets.LogPacket(text));
|
|
};
|
|
|
|
/**
|
|
* Listen for messages from master process (only if worker).
|
|
*/
|
|
|
|
Master.prototype.listen = function listen() {
|
|
assert(!this.listening, 'Already listening.');
|
|
|
|
this.listening = true;
|
|
|
|
this.on('error', (err) => {
|
|
this.send(new packets.ErrorPacket(err));
|
|
});
|
|
|
|
this.on('packet', (packet) => {
|
|
try {
|
|
this.handlePacket(packet);
|
|
} catch (e) {
|
|
this.emit('error', e);
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Handle packet.
|
|
* @private
|
|
* @param {Packet}
|
|
*/
|
|
|
|
Master.prototype.handlePacket = function handlePacket(packet) {
|
|
let result;
|
|
|
|
switch (packet.cmd) {
|
|
case packets.types.ENV:
|
|
this.setEnv(packet.env);
|
|
break;
|
|
case packets.types.EVENT:
|
|
this.emit('event', packet.items);
|
|
this.emit(...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;
|
|
}
|
|
};
|
|
|
|
/*
|
|
* Expose
|
|
*/
|
|
|
|
module.exports = Master;
|