chore: reformat all code with prettier
This commit is contained in:
@@ -1,12 +1,11 @@
|
||||
import { Logger } from '@cvmts/shared';
|
||||
import { Rank, User } from './User.js';
|
||||
|
||||
|
||||
export default class AuthManager {
|
||||
apiEndpoint: string;
|
||||
secretKey: string;
|
||||
|
||||
private logger = new Logger("CVMTS.AuthMan");
|
||||
private logger = new Logger('CVMTS.AuthMan');
|
||||
|
||||
constructor(apiEndpoint: string, secretKey: string) {
|
||||
this.apiEndpoint = apiEndpoint;
|
||||
@@ -14,7 +13,7 @@ export default class AuthManager {
|
||||
}
|
||||
|
||||
async Authenticate(token: string, user: User): Promise<JoinResponse> {
|
||||
var response = await fetch(this.apiEndpoint + '/api/v1/join', {
|
||||
let response = await fetch(this.apiEndpoint + '/api/v1/join', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json'
|
||||
@@ -26,7 +25,7 @@ export default class AuthManager {
|
||||
})
|
||||
});
|
||||
|
||||
var json = (await response.json()) as JoinResponse;
|
||||
let json = (await response.json()) as JoinResponse;
|
||||
|
||||
if (!json.success) {
|
||||
this.logger.Error(`Failed to query auth server: ${json.error}`);
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import { Logger } from "@cvmts/shared";
|
||||
import { Logger } from '@cvmts/shared';
|
||||
|
||||
export class IPData {
|
||||
tempMuteExpireTimeout?: NodeJS.Timeout;
|
||||
@@ -15,19 +15,17 @@ export class IPData {
|
||||
|
||||
// Call when a connection is closed to "release" the ip data
|
||||
Unref() {
|
||||
if(this.refCount - 1 < 0)
|
||||
this.refCount = 0;
|
||||
if (this.refCount - 1 < 0) this.refCount = 0;
|
||||
this.refCount--;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
export class IPDataManager {
|
||||
static ipDatas = new Map<string, IPData>();
|
||||
static logger = new Logger("CVMTS.IPDataManager");
|
||||
static logger = new Logger('CVMTS.IPDataManager');
|
||||
|
||||
static GetIPData(address: string) {
|
||||
if(IPDataManager.ipDatas.has(address)) {
|
||||
if (IPDataManager.ipDatas.has(address)) {
|
||||
// Note: We already check for if it exists, so we use ! here
|
||||
// because TypeScript can't exactly tell that in this case,
|
||||
// only in explicit null or undefined checks
|
||||
@@ -43,8 +41,7 @@ export class IPDataManager {
|
||||
}
|
||||
|
||||
static ForEachIPData(callback: (d: IPData) => void) {
|
||||
for(let tuple of IPDataManager.ipDatas)
|
||||
callback(tuple[1]);
|
||||
for (let tuple of IPDataManager.ipDatas) callback(tuple[1]);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -52,11 +49,10 @@ export class IPDataManager {
|
||||
// Strictly speaking this will just allow the v8 GC to finally
|
||||
// delete the objects, but same difference.
|
||||
setInterval(() => {
|
||||
for(let tuple of IPDataManager.ipDatas) {
|
||||
if(tuple[1].refCount == 0) {
|
||||
IPDataManager.logger.Info("Deleted ipdata for IP {0}", tuple[0]);
|
||||
for (let tuple of IPDataManager.ipDatas) {
|
||||
if (tuple[1].refCount == 0) {
|
||||
IPDataManager.logger.Info('Deleted ipdata for IP {0}', tuple[0]);
|
||||
IPDataManager.ipDatas.delete(tuple[0]);
|
||||
}
|
||||
}
|
||||
}, 15000);
|
||||
|
||||
|
||||
@@ -1,36 +1,36 @@
|
||||
import { EventEmitter } from "events";
|
||||
import { EventEmitter } from 'events';
|
||||
|
||||
// Class to ratelimit a resource (chatting, logging in, etc)
|
||||
export default class RateLimiter extends EventEmitter {
|
||||
private limit : number;
|
||||
private interval : number;
|
||||
private requestCount : number;
|
||||
private limiter? : NodeJS.Timeout;
|
||||
private limiterSet : boolean;
|
||||
constructor(limit : number, interval : number) {
|
||||
super();
|
||||
this.limit = limit;
|
||||
this.interval = interval;
|
||||
this.requestCount = 0;
|
||||
this.limiterSet = false;
|
||||
}
|
||||
// Return value is whether or not the action should be continued
|
||||
request() : boolean {
|
||||
this.requestCount++;
|
||||
if (this.requestCount === this.limit) {
|
||||
this.emit('limit');
|
||||
clearTimeout(this.limiter);
|
||||
this.limiterSet = false;
|
||||
this.requestCount = 0;
|
||||
return false;
|
||||
}
|
||||
if (!this.limiterSet) {
|
||||
this.limiter = setTimeout(() => {
|
||||
this.limiterSet = false;
|
||||
this.requestCount = 0;
|
||||
}, this.interval * 1000);
|
||||
this.limiterSet = true;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
private limit: number;
|
||||
private interval: number;
|
||||
private requestCount: number;
|
||||
private limiter?: NodeJS.Timeout;
|
||||
private limiterSet: boolean;
|
||||
constructor(limit: number, interval: number) {
|
||||
super();
|
||||
this.limit = limit;
|
||||
this.interval = interval;
|
||||
this.requestCount = 0;
|
||||
this.limiterSet = false;
|
||||
}
|
||||
// Return value is whether or not the action should be continued
|
||||
request(): boolean {
|
||||
this.requestCount++;
|
||||
if (this.requestCount === this.limit) {
|
||||
this.emit('limit');
|
||||
clearTimeout(this.limiter);
|
||||
this.limiterSet = false;
|
||||
this.requestCount = 0;
|
||||
return false;
|
||||
}
|
||||
if (!this.limiterSet) {
|
||||
this.limiter = setTimeout(() => {
|
||||
this.limiterSet = false;
|
||||
this.requestCount = 0;
|
||||
}, this.interval * 1000);
|
||||
this.limiterSet = true;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -26,7 +26,7 @@ export class User {
|
||||
TurnRateLimit: RateLimiter;
|
||||
VoteRateLimit: RateLimiter;
|
||||
|
||||
private logger = new Logger("CVMTS.User");
|
||||
private logger = new Logger('CVMTS.User');
|
||||
|
||||
constructor(ws: WebSocket, ip: IPData, config: IConfig, username?: string, node?: string) {
|
||||
this.IP = ip;
|
||||
@@ -59,6 +59,7 @@ export class User {
|
||||
this.VoteRateLimit = new RateLimiter(3, 3);
|
||||
this.VoteRateLimit.on('limit', () => this.closeConnection());
|
||||
}
|
||||
|
||||
assignGuestName(existingUsers: string[]): string {
|
||||
var username;
|
||||
do {
|
||||
@@ -67,25 +68,30 @@ export class User {
|
||||
this.username = username;
|
||||
return username;
|
||||
}
|
||||
|
||||
sendNop() {
|
||||
this.socket.send('3.nop;');
|
||||
}
|
||||
|
||||
sendMsg(msg: string | Buffer) {
|
||||
if (this.socket.readyState !== this.socket.OPEN) return;
|
||||
clearInterval(this.nopSendInterval);
|
||||
this.nopSendInterval = setInterval(() => this.sendNop(), 5000);
|
||||
this.socket.send(msg);
|
||||
}
|
||||
|
||||
private onNoMsg() {
|
||||
this.sendNop();
|
||||
this.nopRecieveTimeout = setTimeout(() => {
|
||||
this.closeConnection();
|
||||
}, 3000);
|
||||
}
|
||||
|
||||
closeConnection() {
|
||||
this.socket.send(guacutils.encode('disconnect'));
|
||||
this.socket.close();
|
||||
}
|
||||
|
||||
onMsgSent() {
|
||||
if (!this.Config.collabvm.automute.enabled) return;
|
||||
// rate limit guest and unregistered chat messages, but not staff ones
|
||||
@@ -99,6 +105,7 @@ export class User {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
mute(permanent: boolean) {
|
||||
this.IP.muted = true;
|
||||
this.sendMsg(guacutils.encode('chat', '', `You have been muted${permanent ? '' : ` for ${this.Config.collabvm.tempMuteTime} seconds`}.`));
|
||||
|
||||
@@ -1,54 +1,54 @@
|
||||
import { Permissions } from "./IConfig";
|
||||
import { Permissions } from './IConfig';
|
||||
|
||||
export function Randint(min : number, max : number) {
|
||||
return Math.floor((Math.random() * (max - min)) + min);
|
||||
}
|
||||
export function HTMLSanitize(input : string) : string {
|
||||
var output = "";
|
||||
for (var i = 0; i < input.length; i++) {
|
||||
switch (input[i]) {
|
||||
case "<":
|
||||
output += "<"
|
||||
break;
|
||||
case ">":
|
||||
output += ">"
|
||||
break;
|
||||
case "&":
|
||||
output += "&"
|
||||
break;
|
||||
case "\"":
|
||||
output += """
|
||||
break;
|
||||
case "'":
|
||||
output += "'";
|
||||
break;
|
||||
case "/":
|
||||
output += "/";
|
||||
break;
|
||||
case "\n":
|
||||
output += " ";
|
||||
break;
|
||||
default:
|
||||
var charcode : number = input.charCodeAt(i);
|
||||
if (charcode >= 32 && charcode <= 126)
|
||||
output += input[i];
|
||||
break;
|
||||
}
|
||||
}
|
||||
return output;
|
||||
export function Randint(min: number, max: number) {
|
||||
return Math.floor(Math.random() * (max - min) + min);
|
||||
}
|
||||
|
||||
export function MakeModPerms(modperms : Permissions) : number {
|
||||
var perms = 0;
|
||||
if (modperms.restore) perms |= 1;
|
||||
if (modperms.reboot) perms |= 2;
|
||||
if (modperms.ban) perms |= 4;
|
||||
if (modperms.forcevote) perms |= 8;
|
||||
if (modperms.mute) perms |= 16;
|
||||
if (modperms.kick) perms |= 32;
|
||||
if (modperms.bypassturn) perms |= 64;
|
||||
if (modperms.rename) perms |= 128;
|
||||
if (modperms.grabip) perms |= 256;
|
||||
if (modperms.xss) perms |= 512;
|
||||
return perms;
|
||||
export function HTMLSanitize(input: string): string {
|
||||
var output = '';
|
||||
for (var i = 0; i < input.length; i++) {
|
||||
switch (input[i]) {
|
||||
case '<':
|
||||
output += '<';
|
||||
break;
|
||||
case '>':
|
||||
output += '>';
|
||||
break;
|
||||
case '&':
|
||||
output += '&';
|
||||
break;
|
||||
case '"':
|
||||
output += '"';
|
||||
break;
|
||||
case "'":
|
||||
output += ''';
|
||||
break;
|
||||
case '/':
|
||||
output += '/';
|
||||
break;
|
||||
case '\n':
|
||||
output += ' ';
|
||||
break;
|
||||
default:
|
||||
var charcode: number = input.charCodeAt(i);
|
||||
if (charcode >= 32 && charcode <= 126) output += input[i];
|
||||
break;
|
||||
}
|
||||
}
|
||||
return output;
|
||||
}
|
||||
|
||||
export function MakeModPerms(modperms: Permissions): number {
|
||||
var perms = 0;
|
||||
if (modperms.restore) perms |= 1;
|
||||
if (modperms.reboot) perms |= 2;
|
||||
if (modperms.ban) perms |= 4;
|
||||
if (modperms.forcevote) perms |= 8;
|
||||
if (modperms.mute) perms |= 16;
|
||||
if (modperms.kick) perms |= 32;
|
||||
if (modperms.bypassturn) perms |= 64;
|
||||
if (modperms.rename) perms |= 128;
|
||||
if (modperms.grabip) perms |= 256;
|
||||
if (modperms.xss) perms |= 512;
|
||||
return perms;
|
||||
}
|
||||
File diff suppressed because it is too large
Load Diff
@@ -1,43 +1,37 @@
|
||||
export function decode(string : string) : string[] {
|
||||
let pos = -1;
|
||||
let sections = [];
|
||||
export function decode(string: string): string[] {
|
||||
let pos = -1;
|
||||
let sections = [];
|
||||
|
||||
for(;;) {
|
||||
let len = string.indexOf('.', pos + 1);
|
||||
for (;;) {
|
||||
let len = string.indexOf('.', pos + 1);
|
||||
|
||||
if(len === -1)
|
||||
break;
|
||||
if (len === -1) break;
|
||||
|
||||
pos = parseInt(string.slice(pos + 1, len)) + len + 1;
|
||||
pos = parseInt(string.slice(pos + 1, len)) + len + 1;
|
||||
|
||||
// don't allow funky protocol length
|
||||
if(pos > string.length)
|
||||
return [];
|
||||
// don't allow funky protocol length
|
||||
if (pos > string.length) return [];
|
||||
|
||||
sections.push(string.slice(len + 1, pos));
|
||||
sections.push(string.slice(len + 1, pos));
|
||||
|
||||
const sep = string.slice(pos, pos + 1);
|
||||
|
||||
const sep = string.slice(pos, pos + 1);
|
||||
if (sep === ',') continue;
|
||||
else if (sep === ';') break;
|
||||
// Invalid data.
|
||||
else return [];
|
||||
}
|
||||
|
||||
if(sep === ',')
|
||||
continue;
|
||||
else if(sep === ';')
|
||||
break;
|
||||
else
|
||||
// Invalid data.
|
||||
return [];
|
||||
}
|
||||
|
||||
return sections;
|
||||
return sections;
|
||||
}
|
||||
|
||||
export function encode(...string : string[]) : string {
|
||||
let command = '';
|
||||
export function encode(...string: string[]): string {
|
||||
let command = '';
|
||||
|
||||
for(var i = 0; i < string.length; i++) {
|
||||
let current = string[i];
|
||||
command += current.toString().length + '.' + current;
|
||||
command += ( i < string.length - 1 ? ',' : ';');
|
||||
}
|
||||
return command;
|
||||
for (var i = 0; i < string.length; i++) {
|
||||
let current = string[i];
|
||||
command += current.toString().length + '.' + current;
|
||||
command += i < string.length - 1 ? ',' : ';';
|
||||
}
|
||||
return command;
|
||||
}
|
||||
@@ -1,6 +1,6 @@
|
||||
import * as toml from 'toml';
|
||||
import IConfig from './IConfig.js';
|
||||
import * as fs from "fs";
|
||||
import * as fs from 'fs';
|
||||
import WSServer from './WSServer.js';
|
||||
|
||||
import { QemuVM, QemuVmDefinition } from '@cvmts/qemu';
|
||||
@@ -8,51 +8,50 @@ import { QemuVM, QemuVmDefinition } from '@cvmts/qemu';
|
||||
import * as Shared from '@cvmts/shared';
|
||||
import AuthManager from './AuthManager.js';
|
||||
|
||||
let logger = new Shared.Logger("CVMTS.Init");
|
||||
let logger = new Shared.Logger('CVMTS.Init');
|
||||
|
||||
logger.Info("CollabVM Server starting up");
|
||||
logger.Info('CollabVM Server starting up');
|
||||
|
||||
// Parse the config file
|
||||
|
||||
var Config : IConfig;
|
||||
let Config: IConfig;
|
||||
|
||||
if (!fs.existsSync("config.toml")) {
|
||||
logger.Error("Fatal error: Config.toml not found. Please copy config.example.toml and fill out fields")
|
||||
process.exit(1);
|
||||
if (!fs.existsSync('config.toml')) {
|
||||
logger.Error('Fatal error: Config.toml not found. Please copy config.example.toml and fill out fields');
|
||||
process.exit(1);
|
||||
}
|
||||
try {
|
||||
var configRaw = fs.readFileSync("config.toml").toString();
|
||||
Config = toml.parse(configRaw);
|
||||
var configRaw = fs.readFileSync('config.toml').toString();
|
||||
Config = toml.parse(configRaw);
|
||||
} catch (e) {
|
||||
logger.Error("Fatal error: Failed to read or parse the config file: {0}", (e as Error).message);
|
||||
process.exit(1);
|
||||
logger.Error('Fatal error: Failed to read or parse the config file: {0}', (e as Error).message);
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
|
||||
async function start() {
|
||||
// Print a warning if qmpSockDir is set
|
||||
// and the host OS is Windows, as this
|
||||
// configuration will very likely not work.
|
||||
if(process.platform === "win32" && Config.vm.qmpSockDir) {
|
||||
logger.Warning("You appear to have the option 'qmpSockDir' enabled in the config.")
|
||||
logger.Warning("This is not supported on Windows, and you will likely run into issues.");
|
||||
logger.Warning("To remove this warning, use the qmpHost and qmpPort options instead.");
|
||||
}
|
||||
// Print a warning if qmpSockDir is set
|
||||
// and the host OS is Windows, as this
|
||||
// configuration will very likely not work.
|
||||
if (process.platform === 'win32' && Config.vm.qmpSockDir) {
|
||||
logger.Warning("You appear to have the option 'qmpSockDir' enabled in the config.");
|
||||
logger.Warning('This is not supported on Windows, and you will likely run into issues.');
|
||||
logger.Warning('To remove this warning, use the qmpHost and qmpPort options instead.');
|
||||
}
|
||||
|
||||
// Init the auth manager if enabled
|
||||
let auth = Config.auth.enabled ? new AuthManager(Config.auth.apiEndpoint, Config.auth.secretKey) : null;
|
||||
// Init the auth manager if enabled
|
||||
let auth = Config.auth.enabled ? new AuthManager(Config.auth.apiEndpoint, Config.auth.secretKey) : null;
|
||||
|
||||
// Fire up the VM
|
||||
let def: QemuVmDefinition = {
|
||||
id: Config.collabvm.node,
|
||||
command: Config.vm.qemuArgs
|
||||
}
|
||||
// Fire up the VM
|
||||
let def: QemuVmDefinition = {
|
||||
id: Config.collabvm.node,
|
||||
command: Config.vm.qemuArgs
|
||||
};
|
||||
|
||||
var VM = new QemuVM(def);
|
||||
await VM.Start();
|
||||
var VM = new QemuVM(def);
|
||||
await VM.Start();
|
||||
|
||||
// Start up the websocket server
|
||||
var WS = new WSServer(Config, VM, auth);
|
||||
WS.listen();
|
||||
// Start up the websocket server
|
||||
var WS = new WSServer(Config, VM, auth);
|
||||
WS.listen();
|
||||
}
|
||||
start();
|
||||
@@ -1,4 +1,4 @@
|
||||
import { Size, Rect } from "@cvmts/shared";
|
||||
import { Size, Rect } from '@cvmts/shared';
|
||||
|
||||
export function BatchRects(size: Size, rects: Array<Rect>): Rect {
|
||||
var mergedX = size.width;
|
||||
|
||||
@@ -30,8 +30,6 @@ const kMaxFailCount = 5;
|
||||
// TODO: This should be added to QemuVmDefinition and the below export removed
|
||||
let gVMShouldSnapshot = true;
|
||||
|
||||
|
||||
|
||||
export function setSnapshot(val: boolean) {
|
||||
gVMShouldSnapshot = val;
|
||||
}
|
||||
@@ -68,10 +66,9 @@ export class QemuVM extends EventEmitter {
|
||||
|
||||
// build additional command line statements to enable qmp/vnc over unix sockets
|
||||
// FIXME: Still use TCP if on Windows.
|
||||
if(!this.addedAdditionalArguments) {
|
||||
if (!this.addedAdditionalArguments) {
|
||||
cmd += ' -no-shutdown';
|
||||
if(gVMShouldSnapshot)
|
||||
cmd += ' -snapshot';
|
||||
if (gVMShouldSnapshot) cmd += ' -snapshot';
|
||||
cmd += ` -qmp unix:${this.GetQmpPath()},server,wait -vnc unix:${this.GetVncPath()}`;
|
||||
this.definition.command = cmd;
|
||||
this.addedAdditionalArguments = true;
|
||||
@@ -275,16 +272,13 @@ export class QemuVM extends EventEmitter {
|
||||
|
||||
private async DisconnectQmp() {
|
||||
if (this.qmpConnected) return;
|
||||
if(this.qmpInstance == null)
|
||||
return;
|
||||
if (this.qmpInstance == null) return;
|
||||
|
||||
this.qmpConnected = false;
|
||||
this.qmpInstance.end();
|
||||
this.qmpInstance = null;
|
||||
try {
|
||||
await unlink(this.GetQmpPath());
|
||||
} catch(err) {
|
||||
|
||||
}
|
||||
} catch (err) {}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -76,7 +76,7 @@ export default class QmpClient extends Socket {
|
||||
// just rethrow lol
|
||||
//throw err;
|
||||
|
||||
console.log("you have pants: rules,", err);
|
||||
console.log('you have pants: rules,', err);
|
||||
});
|
||||
|
||||
this.once('data', (data) => {
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
|
||||
// TODO: `Object` has a toString(), but we should probably gate that off
|
||||
/// Interface for things that can be turned into strings
|
||||
export interface ToStringable {
|
||||
|
||||
@@ -17,8 +17,8 @@ export type Size = {
|
||||
};
|
||||
|
||||
export type Rect = {
|
||||
x: number,
|
||||
y: number,
|
||||
width: number,
|
||||
height: number
|
||||
x: number;
|
||||
y: number;
|
||||
width: number;
|
||||
height: number;
|
||||
};
|
||||
Reference in New Issue
Block a user