Fortify message handler

This commit is contained in:
LJNeon 2022-10-02 16:28:58 -07:00
parent 9fbe101203
commit 86c6a3ad35
No known key found for this signature in database
GPG key ID: 837C6B23D22030BE
6 changed files with 77 additions and 22 deletions

View file

@ -2,6 +2,7 @@ import { setupWatch } from "./fileWatch";
import { config, loadConfig } from "./config";
import { setupSocket } from "./networking/webSocket";
import signal from "signal-js";
import { RawData } from "ws";
import {
fileChangeEventToMsg,
fileRemovalEventToMsg,
@ -10,7 +11,7 @@ import {
} from "./networking/messageGenerators";
import { EventType } from "./eventTypes";
import { messageHandler } from "./networking/messageHandler";
import { FileEvent, Message } from "./interfaces";
import { FileEvent } from "./interfaces";
export async function start() {
loadConfig();
@ -18,7 +19,7 @@ export async function start() {
const socket = setupSocket(signal);
// Add a handler for received messages.
signal.on(EventType.MessageReceived, (msg: Message) => messageHandler(signal, msg, watch.paths));
signal.on(EventType.MessageReceived, (msg: RawData) => messageHandler(signal, msg, watch.paths));
// Add a handler for when a connection to a game is made.
signal.on(EventType.ConnectionMade, () => {

View file

@ -1,10 +1,35 @@
import type { Stats } from "fs";
export interface Message {
id: string;
jsonrpc: "2.0";
method?: string;
jsonrpc: string;
params?: object;
result?: ResultType;
params?: FileMetadata;
error?: string;
id?: number;
}
type ResultType = string | number | string[] | FileContent[];
type FileMetadata = FileData | FileContent | FileLocation | FileServer;
export interface FileData {
filename: string;
content: string;
server: string;
}
export interface FileContent {
filename: string;
content: string;
}
export interface FileLocation {
filename: string;
server: string;
}
export interface FileServer {
server: string;
}
export interface FileEvent {

View file

@ -14,7 +14,7 @@ export function fileChangeEventToMsg({ path }: FileEvent): Message {
filename: addLeadingSlash(path),
content: readFileSync(join(config.get("scriptsFolder"), path)).toString(),
},
id: (messageCounter++).toString(),
id: messageCounter++,
};
}
@ -26,7 +26,7 @@ export function fileRemovalEventToMsg({ path }: FileEvent): Message {
server: "home",
filename: addLeadingSlash(path),
},
id: (messageCounter++).toString(),
id: messageCounter++,
};
}
@ -34,7 +34,7 @@ export function requestDefinitionFile(): Message {
return {
jsonrpc: "2.0",
method: "getDefinitionFile",
id: (messageCounter++).toString(),
id: messageCounter++,
};
}
@ -45,7 +45,7 @@ export function requestFilenames(): Message {
params: {
server: "home",
},
id: (messageCounter++).toString(),
id: messageCounter++,
};
}

View file

@ -1,31 +1,59 @@
import { messageTracker } from "./messageTracker";
import { Stats, writeFile } from "fs";
import { RawData } from "ws";
import { config } from "../config";
import { EventType } from "../eventTypes";
import { fileChangeEventToMsg } from "./messageGenerators";
import type { Signal } from "signal-js";
import { Message } from "../interfaces";
export function messageHandler(signaller: Signal, msg: Message, paths: Map<string, Stats>) {
let incoming;
function deserialize(data: RawData): Message | void {
let msg;
try {
incoming = JSON.parse(msg.toString());
msg = JSON.parse(data.toString());
} catch (err) {
return console.log(err);
}
console.log(incoming);
if (incoming.id == undefined) return;
if (incoming.result) {
const request = messageTracker.get(incoming.id);
if (request?.method && request.method == "getDefinitionFile" && incoming.result) {
if (typeof msg.jsonrpc !== "string" || msg.jsonrpc !== "2.0" || typeof msg.id !== "number") return;
const id: number = msg.id;
const request = messageTracker.get(id);
if (typeof request?.method !== "string") return;
else if (msg.error != null) return { jsonrpc: "2.0", error: msg.error, id };
else if (msg.result == null) return;
return { jsonrpc: "2.0", method: request.method, result: msg.result, id };
}
function isStringArray(s: Array<unknown>): s is string[] {
return s.every((s) => typeof s === "string");
}
export function messageHandler(signaller: Signal, data: RawData, paths: Map<string, Stats>) {
const incoming = deserialize(data);
if (incoming == null) {
return console.log("Malformed data received.");
} else if (incoming.error) {
return console.log(incoming.error);
}
switch (incoming.method) {
case "getDefinitionFile":
if (typeof incoming.result !== "string") return console.log("Malformed data received.");
writeFile(config.get("definitionFile").location, incoming.result, (err) => {
if (err) return console.log(err);
});
}
if (request?.method && request.method == "getFileNames" && incoming.result) {
break;
case "getFileNames": {
if (!Array.isArray(incoming.result) || !isStringArray(incoming.result))
return console.log("Malformed data received.");
const gameFiles = incoming.result.map((file: string) => removeLeadingSlash(file));
paths.forEach((stats, fileName) => {

View file

@ -1,10 +1,12 @@
import type { Message } from "../interfaces";
class MessageTracker {
data = new Map<string, Message>();
data = new Map<number, Message>();
#maxLength = 200;
push(msg: Message) {
if (typeof msg.id !== "number") return;
this.data.set(msg.id, msg);
if (this.data.size > this.#maxLength) {
@ -13,7 +15,7 @@ class MessageTracker {
}
}
get(index: string) {
get(index: number) {
return this.data.get(index);
}
}

View file

@ -3,7 +3,6 @@ import { WebSocketServer } from "ws";
import { config } from "../config";
import { EventType } from "../eventTypes";
import { Message } from "../interfaces";
import { requestDefinitionFile } from "./messageGenerators";
import { messageTracker } from "./messageTracker";
export function setupSocket(signaller: Signal) {