mirror of
https://github.com/EvernodeXRPL/hpcore.git
synced 2026-04-29 15:37:59 +00:00
153 lines
4.2 KiB
JavaScript
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
|
|
} |