mirror of
https://github.com/EvernodeXRPL/hpcore.git
synced 2026-04-29 15:37:59 +00:00
178 lines
5.1 KiB
JavaScript
178 lines
5.1 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 });
|
|
let dataParts = [];
|
|
let msgCount = -1;
|
|
let msgLen = -1;
|
|
let pos = 0;
|
|
socket.on("data", (buf) => {
|
|
pos = 0;
|
|
if (msgCount == -1) {
|
|
const msgCountBuf = readBytes(buf, 0, 4)
|
|
msgCount = msgCountBuf.readUInt32BE();
|
|
pos += 4;
|
|
}
|
|
while (pos < buf.byteLength) {
|
|
if (msgLen == -1) {
|
|
const msgLenBuf = readBytes(buf, pos, 4);
|
|
pos += 4;
|
|
msgLen = msgLenBuf.readUInt32BE();
|
|
}
|
|
let possible_read_len;
|
|
if (((buf.byteLength - pos) - msgLen) >= 0) {
|
|
// Can finish reading a full message.
|
|
possible_read_len = msgLen;
|
|
msgLen = -1;
|
|
} else {
|
|
// Only partial message is recieved.
|
|
possible_read_len = buf.byteLength - pos
|
|
msgLen -= possible_read_len;
|
|
}
|
|
const msgBuf = readBytes(buf, pos, possible_read_len);
|
|
pos += possible_read_len;
|
|
dataParts.push(msgBuf)
|
|
|
|
if (msgLen == -1) {
|
|
events.emit("user_message", userPubKey, Buffer.concat(dataParts));
|
|
dataParts = [];
|
|
msgCount--
|
|
}
|
|
if (msgCount == 0) {
|
|
msgCount = -1
|
|
events.emit("user_finished", userPubKey);
|
|
}
|
|
}
|
|
});
|
|
|
|
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) {
|
|
const outputStringBuf = Buffer.from(output);
|
|
let headerBuf = Buffer.alloc(4);
|
|
// Writing message length in big endian format.
|
|
headerBuf.writeUInt32BE(outputStringBuf.byteLength)
|
|
fs.writeSync(fd, headerBuf);
|
|
fs.writeSync(fd, outputStringBuf);
|
|
}
|
|
|
|
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
|
|
} |