Files
hpcore/examples/nodejs_contract/hp-contract-lib.js
2020-11-02 16:50:21 +05:30

153 lines
4.2 KiB
JavaScript

const fs = require('fs');
const events = require('events');
const MAX_SEQ_PACKET_SIZE = 128 * 1024;
function HotPocketContract() {
const hpargs = JSON.parse(fs.readFileSync(0, 'utf8'));
this.readonly = hpargs.readonly;
this.timestamp = hpargs.ts;
if (!this.readonly) {
const lclParts = hpargs.lcl.split("-");
this.lcl = {
seqNo: parseInt(lclParts[0]),
hash: lclParts[1]
};
this.npl = new HotPocketNplChannel(hpargs.nplfd);
}
this.control = new HotPocketControlChannel(hpargs.hpfd);
this.events = new events.EventEmitter();
this.users = {};
Object.keys(hpargs.usrfd).forEach((userPubKey) => {
this.users[userPubKey] = new HotPocketChannel(hpargs.usrfd[userPubKey], userPubKey, this.events);
});
}
function HotPocketChannel(fd, userPubKey, events) {
let socket = null;
if (fd > 0) {
socket = fs.createReadStream(null, { fd: fd });
const dataParts = [];
let msgLen = -1;
let bytesRead = 0;
socket.on("data", (buf) => {
if (msgLen == -1) {
// First two bytes indicate the message len.
const msgLenBuf = readBytes(buf, 0, 4);
if (msgLenBuf) {
msgLen = msgLenBuf.readUInt32BE();
const msgBuf = readBytes(buf, 4, buf.byteLength - 4);
dataParts.push(msgBuf)
bytesRead = msgBuf.byteLength;
}
} else {
dataParts.push(buf);
bytesRead += buf.length;
}
if (bytesRead == msgLen) {
msgLen == -1;
events.emit("user_message", userPubKey, Buffer.concat(dataParts));
}
});
socket.on("error", (e) => {
events.emit("user_error", userPubKey, e);
})
}
// Read bytes from the given buffer.
const readBytes = function (buf, pos, count) {
if (pos + count > buf.byteLength)
return null;
return buf.slice(pos, pos + count);
}
this.sendOutput = function (output) {
fs.writeSync(fd, output);
}
this.closeChannel = function () {
if (fd > 0) {
socket.destroy();
}
}
}
function HotPocketNplChannel(fd) {
this.events = new events.EventEmitter();
let socket = null;
let isPubKeyReceived = false;
let pubKey;
if (fd > 0) {
// From the hotpocket when sending the npl messages first it sends the pubkey of the particular node
// and then the message, First data buffer is taken as pubkey and the second one as message,
// then npl message object is constructed and the event is emmited.
socket = fs.createReadStream(null, { fd: fd, highWaterMark: MAX_SEQ_PACKET_SIZE });
socket.on("data", d => {
if (!isPubKeyReceived) {
pubKey = d.toString('hex');
isPubKeyReceived = true;
}
else {
this.events.emit("message", {
pubkey: pubKey,
input: d
});
pubKey = null;
isPubKeyReceived = false;
}
});
socket.on("error", (e) => {
this.events.emit("error", e);
});
}
this.sendOutput = (output) => {
if (fd > 0) {
fs.writeSync(fd, output);
}
}
this.closeNplChannel = () => {
if (fd > 0) {
socket.destroy();
}
}
}
function HotPocketControlChannel(fd) {
this.events = new events.EventEmitter();
let socket = null;
if (fd > 0) {
socket = fs.createReadStream(null, { fd: fd, highWaterMark: MAX_SEQ_PACKET_SIZE });
socket.on("data", d => {
this.events.emit("message", d);
});
socket.on("error", (e) => {
this.events.emit("error", e);
});
}
this.sendOutput = (output) => {
if (fd > 0) {
fs.writeSync(fd, output);
}
}
this.closeControlChannel = () => {
if (fd > 0) {
socket.destroy();
}
}
}
module.exports = {
HotPocketContract
}