Spaces:
Paused
Paused
| ; | |
| var __create = Object.create; | |
| var __defProp = Object.defineProperty; | |
| var __getOwnPropDesc = Object.getOwnPropertyDescriptor; | |
| var __getOwnPropNames = Object.getOwnPropertyNames; | |
| var __getProtoOf = Object.getPrototypeOf; | |
| var __hasOwnProp = Object.prototype.hasOwnProperty; | |
| var __export = (target, all) => { | |
| for (var name in all) | |
| __defProp(target, name, { get: all[name], enumerable: true }); | |
| }; | |
| var __copyProps = (to, from, except, desc) => { | |
| if (from && typeof from === "object" || typeof from === "function") { | |
| for (let key of __getOwnPropNames(from)) | |
| if (!__hasOwnProp.call(to, key) && key !== except) | |
| __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); | |
| } | |
| return to; | |
| }; | |
| var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps( | |
| // If the importer is in node compatibility mode or this is not an ESM | |
| // file that has been converted to a CommonJS file using a Babel- | |
| // compatible transform (i.e. "__esModule" has not been set), then set | |
| // "default" to the CommonJS "module.exports" for node compatibility. | |
| isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, | |
| mod | |
| )); | |
| var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); | |
| var converter_exports = {}; | |
| __export(converter_exports, { | |
| ModlogConverter: () => ModlogConverter, | |
| ModlogConverterSQLite: () => ModlogConverterSQLite, | |
| ModlogConverterTest: () => ModlogConverterTest, | |
| ModlogConverterTxt: () => ModlogConverterTxt, | |
| modernizeLog: () => modernizeLog, | |
| parseBrackets: () => parseBrackets, | |
| parseModlog: () => parseModlog, | |
| rawifyLog: () => rawifyLog | |
| }); | |
| module.exports = __toCommonJS(converter_exports); | |
| var import_lib = require("../../lib"); | |
| var import_ip_tools = require("../../server/ip-tools"); | |
| if (!global.Config) { | |
| let hasSQLite = true; | |
| try { | |
| require.resolve("better-sqlite3"); | |
| } catch { | |
| console.warn(`Warning: the modlog conversion script is running without a SQLite library.`); | |
| hasSQLite = false; | |
| } | |
| global.Config = { | |
| nofswriting: false, | |
| usesqlitemodlog: hasSQLite, | |
| usesqlite: hasSQLite | |
| }; | |
| } | |
| const Database = Config.usesqlite ? require("better-sqlite3") : null; | |
| const { Modlog } = require("../../server/modlog"); | |
| const ENTRIES_TO_BUFFER = 7500; | |
| const ALTS_REGEX = /\(.*?'s (lock|mut|bann|blacklist)ed alts: (.*)\)/; | |
| const AUTOCONFIRMED_REGEX = /\(.*?'s ac account: (.*)\)/; | |
| const IP_ONLY_ACTIONS = /* @__PURE__ */ new Set([ | |
| "SHAREDIP", | |
| "UNSHAREDIP", | |
| "UNLOCKIP", | |
| "UNLOCKRANGE", | |
| "RANGEBAN", | |
| "RANGELOCK" | |
| ]); | |
| function parseBrackets(line, openingBracket, greedy) { | |
| const brackets = { | |
| "(": ")", | |
| "[": "]" | |
| }; | |
| const bracketOpenIndex = line.indexOf(openingBracket); | |
| const bracketCloseIndex = greedy ? line.lastIndexOf(brackets[openingBracket]) : line.indexOf(brackets[openingBracket]); | |
| if (bracketCloseIndex < 0 || bracketOpenIndex < 0) | |
| return ""; | |
| return line.slice(bracketOpenIndex + 1, bracketCloseIndex); | |
| } | |
| function toID(text) { | |
| return (text && typeof text === "string" ? text : "").toLowerCase().replace(/[^a-z0-9]+/g, ""); | |
| } | |
| function modernizeLog(line, nextLine) { | |
| const prefix = line.match(/\[.+?\] \(.+?\) /i)?.[0]; | |
| if (!prefix) | |
| return; | |
| if (ALTS_REGEX.test(line) || AUTOCONFIRMED_REGEX.test(line)) | |
| return; | |
| line = line.replace(prefix, ""); | |
| if (line.startsWith("(")) | |
| line = line.replace(/\([a-z0-9-]*\) /, ""); | |
| if (line.startsWith("(") && line.endsWith(")")) { | |
| line = line.slice(1, -1); | |
| } | |
| const getAlts = () => { | |
| let alts = ""; | |
| nextLine?.replace(ALTS_REGEX, (_a, _b, rawAlts) => { | |
| if (rawAlts) | |
| alts = `alts: [${rawAlts.split(",").map(toID).join("], [")}] `; | |
| return ""; | |
| }); | |
| return alts; | |
| }; | |
| const getAutoconfirmed = () => { | |
| let autoconfirmed = ""; | |
| nextLine?.replace(AUTOCONFIRMED_REGEX, (_a, rawAutoconfirmed) => { | |
| if (rawAutoconfirmed) | |
| autoconfirmed = `ac: [${toID(rawAutoconfirmed)}] `; | |
| return ""; | |
| }); | |
| return autoconfirmed; | |
| }; | |
| if (line.startsWith("SCAV ")) { | |
| line = line.replace(/: (\[room: .*?\]) by (.*)/, (match, roominfo, rest) => `: by ${rest} ${roominfo}`); | |
| } | |
| line = line.replace( | |
| /(GIVEAWAY WIN|GTS FINISHED): ([A-Za-z0-9].*?)(won|has finished)/, | |
| (match, action, user) => `${action}: [${toID(user)}]:` | |
| ); | |
| if (line.includes(":")) { | |
| const possibleModernAction = line.slice(0, line.indexOf(":")).trim(); | |
| if (possibleModernAction === possibleModernAction.toUpperCase()) { | |
| if (possibleModernAction.includes("[")) { | |
| const [drop, ...keep] = line.split("["); | |
| process.stderr.write(`Ignoring malformed line: ${drop} | |
| `); | |
| return modernizeLog(keep.join("")); | |
| } | |
| if (/\(.+\) by [a-z0-9]{1,19}$/.test(line) && !["OLD MODLOG", "NOTE"].includes(possibleModernAction)) { | |
| const reason = parseBrackets(line, "(", true); | |
| return `${prefix}${line.replace(` (${reason})`, "")}: ${reason}`; | |
| } | |
| return `${prefix}${line}`; | |
| } | |
| } | |
| if (/\[(the|a)poll\] was (started|ended) by/.test(line)) { | |
| const actionTaker = toID(line.slice(line.indexOf(" by ") + " by ".length)); | |
| const isEnding = line.includes("was ended by"); | |
| return `${prefix}POLL${isEnding ? " END" : ""}: by ${actionTaker}`; | |
| } | |
| if (/User (.*?) won the game of (.*?) mode trivia/.test(line)) { | |
| return `${prefix}TRIVIAGAME: by unknown: ${line}`; | |
| } | |
| const modernizerTransformations = { | |
| "notes: ": (log) => { | |
| const [actionTaker, ...rest] = line.split(" notes: "); | |
| return `NOTE: by ${toID(actionTaker)}: ${rest.join("")}`; | |
| }, | |
| " declared": (log) => { | |
| let newAction = "DECLARE"; | |
| let oldAction = " declared"; | |
| if (log.includes(" globally declared")) { | |
| oldAction = " globally declared"; | |
| newAction = "GLOBALDECLARE"; | |
| } | |
| if (log.includes("(chat level)")) { | |
| oldAction += " (chat level)"; | |
| newAction = `CHATDECLARE`; | |
| } | |
| const actionTakerName = toID(log.slice(0, log.lastIndexOf(oldAction))); | |
| log = log.slice(actionTakerName.length); | |
| log = log.slice(oldAction.length); | |
| log = log.replace(/^\s?:/, "").trim(); | |
| return `${newAction}: by ${actionTakerName}: ${log}`; | |
| }, | |
| "changed the roomdesc to: ": (log) => { | |
| const actionTaker = parseBrackets(log, "["); | |
| log = log.slice(actionTaker.length + 3); | |
| log = log.slice("changed the roomdesc to: ".length + 1, -2); | |
| return `ROOMDESC: by ${actionTaker}: to "${log}"`; | |
| }, | |
| 'roomevent titled "': (log) => { | |
| let action; | |
| if (log.includes(' added a roomevent titled "')) { | |
| action = "added a"; | |
| } else if (log.includes(' removed a roomevent titled "')) { | |
| action = "removed a"; | |
| } else { | |
| action = "edited the"; | |
| } | |
| const actionTakerName = log.slice(0, log.lastIndexOf(` ${action} roomevent titled "`)); | |
| log = log.slice(actionTakerName.length + 1); | |
| const eventName = log.slice(` ${action} roomevent titled `.length, -2); | |
| return `ROOMEVENT: by ${toID(actionTakerName)}: ${action.split(" ")[0]} "${eventName}"`; | |
| }, | |
| "set modchat to ": (log) => { | |
| const actionTaker = parseBrackets(log, "["); | |
| log = log.slice(actionTaker.length + 3); | |
| log = log.slice("set modchat to ".length); | |
| return `MODCHAT: by ${actionTaker}: to ${log}`; | |
| }, | |
| "set modjoin to ": (log) => { | |
| const actionTakerName = log.slice(0, log.lastIndexOf(" set")); | |
| log = log.slice(actionTakerName.length + 1); | |
| log = log.slice("set modjoin to ".length); | |
| const rank = log.startsWith("sync") ? "sync" : log.replace(".", ""); | |
| return `MODJOIN${rank === "sync" ? " SYNC" : ""}: by ${toID(actionTakerName)}${rank !== "sync" ? `: ${rank}` : ``}`; | |
| }, | |
| "turned off modjoin": (log) => { | |
| const actionTakerName = log.slice(0, log.lastIndexOf(" turned off modjoin")); | |
| return `MODJOIN: by ${toID(actionTakerName)}: OFF`; | |
| }, | |
| "changed the roomintro": (log) => { | |
| const isDeletion = /deleted the (staff|room)intro/.test(log); | |
| const isRoomintro = log.includes("roomintro"); | |
| const actionTaker = toID(log.slice(0, log.indexOf(isDeletion ? "deleted" : "changed"))); | |
| return `${isDeletion ? "DELETE" : ""}${isRoomintro ? "ROOM" : "STAFF"}INTRO: by ${actionTaker}`; | |
| }, | |
| "deleted the roomintro": (log) => modernizerTransformations["changed the roomintro"](log), | |
| "changed the staffintro": (log) => modernizerTransformations["changed the roomintro"](log), | |
| "deleted the staffintro": (log) => modernizerTransformations["changed the roomintro"](log), | |
| "created a tournament in": (log) => { | |
| const actionTaker = parseBrackets(log, "["); | |
| log = log.slice(actionTaker.length + 3); | |
| log = log.slice(24, -8); | |
| return `TOUR CREATE: by ${actionTaker}: ${log}`; | |
| }, | |
| "was disqualified from the tournament by": (log) => { | |
| const disqualified = parseBrackets(log, "["); | |
| log = log.slice(disqualified.length + 3); | |
| log = log.slice("was disqualified from the tournament by".length); | |
| return `TOUR DQ: [${toID(disqualified)}] by ${toID(log)}`; | |
| }, | |
| "The tournament auto disqualify timeout was set to": (log) => { | |
| const byIndex = log.indexOf(" by "); | |
| const actionTaker = log.slice(byIndex + " by ".length); | |
| const length = log.slice("The tournament auto disqualify timeout was set to".length, byIndex); | |
| return `TOUR AUTODQ: by ${toID(actionTaker)}: ${length.trim()}`; | |
| }, | |
| " was blacklisted from ": (log) => { | |
| const isName = log.includes(" was nameblacklisted from "); | |
| const banned = toID(log.slice(0, log.indexOf(` was ${isName ? "name" : ""}blacklisted from `))); | |
| log = log.slice(log.indexOf(" by ") + " by ".length); | |
| let reason, ip; | |
| if (/\(.*\)/.test(log)) { | |
| reason = parseBrackets(log, "("); | |
| if (/\[.*\]/.test(log)) | |
| ip = parseBrackets(log, "["); | |
| log = log.slice(0, log.indexOf("(")); | |
| } | |
| const actionTaker = toID(log); | |
| return `${isName ? "NAME" : ""}BLACKLIST: [${banned}] ${getAutoconfirmed()}${getAlts()}${ip ? `[${ip}] ` : ``}by ${actionTaker}${reason ? `: ${reason}` : ``}`; | |
| }, | |
| " was nameblacklisted from ": (log) => modernizerTransformations[" was blacklisted from "](log), | |
| " was banned from room ": (log) => { | |
| const banned = toID(log.slice(0, log.indexOf(" was banned from room "))); | |
| log = log.slice(log.indexOf(" by ") + " by ".length); | |
| let reason, ip; | |
| if (/\(.*\)/.test(log)) { | |
| reason = parseBrackets(log, "("); | |
| if (/\[.*\]/.test(log)) | |
| ip = parseBrackets(log, "["); | |
| log = log.slice(0, log.indexOf("(")); | |
| } | |
| const actionTaker = toID(log); | |
| return `ROOMBAN: [${banned}] ${getAutoconfirmed()}${getAlts()}${ip ? `[${ip}] ` : ``}by ${actionTaker}${reason ? `: ${reason}` : ``}`; | |
| }, | |
| " was muted by ": (log) => { | |
| let muted = ""; | |
| let isHour = false; | |
| [muted, log] = log.split(" was muted by "); | |
| muted = toID(muted); | |
| let reason, ip; | |
| if (/\(.*\)/.test(log)) { | |
| reason = parseBrackets(log, "("); | |
| if (/\[.*\]/.test(log)) | |
| ip = parseBrackets(log, "["); | |
| log = log.slice(0, log.indexOf("(")); | |
| } | |
| let actionTaker = toID(log); | |
| if (actionTaker.endsWith("for1hour")) { | |
| isHour = true; | |
| actionTaker = actionTaker.replace(/^(.*)(for1hour)$/, (match, staff) => staff); | |
| } | |
| return `${isHour ? "HOUR" : ""}MUTE: [${muted}] ${getAutoconfirmed()}${getAlts()}${ip ? `[${ip}] ` : ``}by ${actionTaker}${reason ? `: ${reason}` : ``}`; | |
| }, | |
| " was locked from talking ": (log) => { | |
| const isWeek = log.includes(" was locked from talking for a week "); | |
| const locked = toID(log.slice(0, log.indexOf(" was locked from talking "))); | |
| log = log.slice(log.indexOf(" by ") + " by ".length); | |
| let reason, ip; | |
| if (/\(.*\)/.test(log)) { | |
| reason = parseBrackets(log, "("); | |
| if (/\[.*\]/.test(log)) | |
| ip = parseBrackets(log, "["); | |
| log = log.slice(0, log.indexOf("(")); | |
| } | |
| const actionTaker = toID(log); | |
| return `${isWeek ? "WEEK" : ""}LOCK: [${locked}] ${getAutoconfirmed()}${getAlts()}${ip ? `[${ip}] ` : ``}by ${actionTaker}${reason ? `: ${reason}` : ``}`; | |
| }, | |
| " was banned ": (log) => { | |
| if (log.includes(" was banned from room ")) | |
| return modernizerTransformations[" was banned from room "](log); | |
| const banned = toID(log.slice(0, log.indexOf(" was banned "))); | |
| log = log.slice(log.indexOf(" by ") + " by ".length); | |
| let reason, ip; | |
| if (/\(.*\)/.test(log)) { | |
| reason = parseBrackets(log, "("); | |
| if (/\[.*\]/.test(log)) | |
| ip = parseBrackets(log, "["); | |
| log = log.slice(0, log.indexOf("(")); | |
| } | |
| const actionTaker = toID(log); | |
| return `BAN: [${banned}] ${getAutoconfirmed()}${getAlts()}${ip ? `[${ip}] ` : ``}by ${actionTaker}${reason ? `: ${reason}` : ``}`; | |
| }, | |
| "was promoted to ": (log) => { | |
| const isDemotion = log.includes("was demoted to "); | |
| const userid = toID(log.split(" was ")[0]); | |
| if (!userid) { | |
| throw new Error(`Ignoring malformed line: ${prefix}${log}`); | |
| } | |
| log = log.slice(userid.length + 3); | |
| log = log.slice(`was ${isDemotion ? "demoted" : "promoted"} to `.length); | |
| let rank = log.slice(0, log.indexOf(" by")).replace(/ /, "").toUpperCase(); | |
| log = log.slice(`${rank} by `.length); | |
| if (!rank.startsWith("ROOM")) | |
| rank = `GLOBAL ${rank}`; | |
| const actionTaker = parseBrackets(log, "["); | |
| return `${rank}: [${userid}] by ${actionTaker}${isDemotion ? ": (demote)" : ""}`; | |
| }, | |
| "was demoted to ": (log) => modernizerTransformations["was promoted to "](log), | |
| "was appointed Room Owner by ": (log) => { | |
| const userid = parseBrackets(log, "["); | |
| log = log.slice(userid.length + 3); | |
| log = log.slice("was appointed Room Owner by ".length); | |
| const actionTaker = parseBrackets(log, "["); | |
| return `ROOMOWNER: [${userid}] by ${actionTaker}`; | |
| }, | |
| " claimed this ticket": (log) => { | |
| const actions = { | |
| " claimed this ticket": "TICKETCLAIM", | |
| " closed this ticket": "TICKETCLOSE", | |
| " deleted this ticket": "TICKETDELETE" | |
| }; | |
| for (const oldAction in actions) { | |
| if (log.includes(oldAction)) { | |
| const actionTaker = toID(log.slice(0, log.indexOf(oldAction))); | |
| return `${actions[oldAction]}: by ${actionTaker}`; | |
| } | |
| } | |
| return log; | |
| }, | |
| "This ticket is now claimed by ": (log) => { | |
| const claimer = toID(log.slice(log.indexOf(" by ") + " by ".length)); | |
| return `TICKETCLAIM: by ${claimer}`; | |
| }, | |
| " is no longer interested in this ticket": (log) => { | |
| const abandoner = toID(log.slice(0, log.indexOf(" is no longer interested in this ticket"))); | |
| return `TICKETABANDON: by ${abandoner}`; | |
| }, | |
| " opened a new ticket": (log) => { | |
| const opener = toID(log.slice(0, log.indexOf(" opened a new ticket"))); | |
| const problem = log.slice(log.indexOf(" Issue: ") + " Issue: ".length).trim(); | |
| return `TICKETOPEN: by ${opener}: ${problem}`; | |
| }, | |
| " closed this ticket": (log) => modernizerTransformations[" claimed this ticket"](log), | |
| " deleted this ticket": (log) => modernizerTransformations[" claimed this ticket"](log), | |
| "This ticket is no longer claimed": () => "TICKETUNCLAIM", | |
| " has been caught attempting a hunt with ": (log) => { | |
| const index = log.indexOf(" has been caught attempting a hunt with "); | |
| const user = toID(log.slice(0, index)); | |
| log = log.slice(index + " has been caught attempting a hunt with ".length); | |
| log = log.replace(". The user has also", "; has also").replace(".", ""); | |
| return `SCAV CHEATER: [${user}]: caught attempting a hunt with ${log}`; | |
| }, | |
| "made this room hidden": (log) => { | |
| const user = toID(log.slice(0, log.indexOf(" made this room hidden"))); | |
| return `HIDDENROOM: by ${user}`; | |
| }, | |
| "The tournament auto start timer was set to ": (log) => { | |
| log = log.slice("The tournament auto start timer was set to".length); | |
| const [length, setter] = log.split(" by ").map(toID); | |
| return `TOUR AUTOSTART: by ${setter}: ${length}`; | |
| }, | |
| "The tournament auto disqualify timer was set to ": (log) => { | |
| log = log.slice("The tournament auto disqualify timer was set to".length); | |
| const [length, setter] = log.split(" by ").map(toID); | |
| return `TOUR AUTODQ: by ${setter}: ${length}`; | |
| }, | |
| " set the tournament's banlist to ": (log) => { | |
| const [setter, banlist] = log.split(` set the tournament's banlist to `); | |
| return `TOUR BANLIST: by ${toID(setter)}: ${banlist.slice(0, -1)}`; | |
| }, | |
| " set the tournament's custom rules to": (log) => { | |
| const [setter, rules] = log.split(` set the tournament's custom rules to `); | |
| return `TOUR RULES: by ${toID(setter)}: ${rules.slice(0, -1)}`; | |
| }, | |
| "[agameofhangman] was started by ": (log) => `HANGMAN: by ${toID(log.slice("[agameofhangman] was started by ".length))}`, | |
| "[agameofunowas] created by ": (log) => `UNO CREATE: by ${toID(log.slice("[agameofunowas] created by ".length))}`, | |
| "[thetournament] was set to autostart": (log) => { | |
| const [, user] = log.split(" by "); | |
| return `TOUR AUTOSTART: by ${toID(user)}: when playercap is reached`; | |
| }, | |
| "[thetournament] was set to allow scouting": (log) => { | |
| const [, user] = log.split(" by "); | |
| return `TOUR SCOUT: by ${toID(user)}: allow`; | |
| }, | |
| "[thetournament] was set to disallow scouting": (log) => { | |
| const [, user] = log.split(" by "); | |
| return `TOUR SCOUT: by ${toID(user)}: disallow`; | |
| } | |
| }; | |
| for (const oldAction in modernizerTransformations) { | |
| if (line.includes(oldAction)) { | |
| try { | |
| return prefix + modernizerTransformations[oldAction](line); | |
| } catch (err) { | |
| if (Config.nofswriting) | |
| throw err; | |
| process.stderr.write(`${err.message} | |
| `); | |
| } | |
| } | |
| } | |
| return `${prefix}${line}`; | |
| } | |
| function parseModlog(raw, nextLine, isGlobal = false) { | |
| let line = modernizeLog(raw); | |
| if (!line) | |
| return; | |
| const timestamp = parseBrackets(line, "["); | |
| line = line.slice(timestamp.length + 3); | |
| const [roomID, ...bonus] = parseBrackets(line, "(").split(" "); | |
| const log = { | |
| action: "NULL", | |
| roomID, | |
| visualRoomID: "", | |
| userid: null, | |
| autoconfirmedID: null, | |
| alts: [], | |
| ip: null, | |
| isGlobal, | |
| loggedBy: null, | |
| note: "", | |
| time: Math.floor(new Date(timestamp).getTime()) || 0 | |
| }; | |
| if (bonus.length) | |
| log.visualRoomID = `${log.roomID} ${bonus.join(" ")}`; | |
| line = line.slice((log.visualRoomID || log.roomID).length + 3); | |
| const actionColonIndex = line.indexOf(":"); | |
| const action = line.slice(0, actionColonIndex); | |
| if (action !== action.toUpperCase()) { | |
| log.action = "OLD MODLOG"; | |
| log.loggedBy = "unknown"; | |
| log.note = line.trim(); | |
| return log; | |
| } else { | |
| log.action = action; | |
| if (log.action === "OLD MODLOG") { | |
| log.loggedBy = "unknown"; | |
| log.note = line.slice(line.indexOf("by unknown: ") + "by unknown :".length).trim(); | |
| return log; | |
| } | |
| line = line.slice(actionColonIndex + 2); | |
| } | |
| if (line[0] === "[") { | |
| if (!IP_ONLY_ACTIONS.has(log.action)) { | |
| const userid = toID(parseBrackets(line, "[")); | |
| log.userid = userid; | |
| line = line.slice(userid.length + 3).trim(); | |
| if (line.startsWith("ac:")) { | |
| line = line.slice(3).trim(); | |
| const ac = parseBrackets(line, "["); | |
| log.autoconfirmedID = toID(ac); | |
| line = line.slice(ac.length + 3).trim(); | |
| } | |
| if (line.startsWith("alts:")) { | |
| line = line.slice(5).trim(); | |
| const alts = /* @__PURE__ */ new Set(); | |
| let alt = parseBrackets(line, "["); | |
| do { | |
| if (alt.includes(", ")) { | |
| for (const trueAlt of alt.split(", ")) { | |
| alts.add(toID(trueAlt)); | |
| } | |
| line = line.slice(line.indexOf(`[${alt}],`) + `[${alt}],`.length).trim(); | |
| if (!line.startsWith("[")) | |
| line = `[${line}`; | |
| } else { | |
| if (import_ip_tools.IPTools.ipRegex.test(alt)) | |
| break; | |
| alts.add(toID(alt)); | |
| line = line.slice(line.indexOf(`[${alt}],`) + `[${alt}],`.length).trim(); | |
| if (alt.includes("[") && !line.startsWith("[")) | |
| line = `[${line}`; | |
| } | |
| alt = parseBrackets(line, "["); | |
| } while (alt); | |
| log.alts = [...alts]; | |
| } | |
| } | |
| if (line[0] === "[") { | |
| log.ip = parseBrackets(line, "["); | |
| line = line.slice(log.ip.length + 3).trim(); | |
| } | |
| } | |
| let regex = /\bby .*:/; | |
| let actionTakerIndex = regex.exec(line)?.index; | |
| if (actionTakerIndex === void 0) { | |
| actionTakerIndex = line.indexOf("by "); | |
| regex = /\bby .*/; | |
| } | |
| if (actionTakerIndex !== -1) { | |
| const colonIndex = line.indexOf(": "); | |
| const actionTaker = line.slice(actionTakerIndex + 3, colonIndex > actionTakerIndex ? colonIndex : void 0); | |
| if (toID(actionTaker).length < 19) { | |
| log.loggedBy = toID(actionTaker) || null; | |
| if (colonIndex > actionTakerIndex) | |
| line = line.slice(colonIndex); | |
| line = line.replace(regex, " "); | |
| } | |
| } | |
| if (line) | |
| log.note = line.replace(/^\s?:\s?/, "").trim(); | |
| return log; | |
| } | |
| function rawifyLog(log) { | |
| let result = `[${new Date(log.time || Date.now()).toJSON()}] (${(log.visualRoomID || log.roomID || "global").replace(/^global-/, "")}) ${log.action}`; | |
| if (log.userid) | |
| result += `: [${log.userid}]`; | |
| if (log.autoconfirmedID) | |
| result += ` ac: [${log.autoconfirmedID}]`; | |
| if (log.alts.length) | |
| result += ` alts: [${log.alts.join("], [")}]`; | |
| if (log.ip) { | |
| if (!log.userid) | |
| result += `:`; | |
| result += ` [${log.ip}]`; | |
| } | |
| if (log.loggedBy) | |
| result += `${result.endsWith("]") ? "" : ":"} by ${log.loggedBy}`; | |
| if (log.note) | |
| result += `: ${log.note}`; | |
| return result + ` | |
| `; | |
| } | |
| class ModlogConverterSQLite { | |
| constructor(databaseFile, textLogDir, isTesting, newestAllowedTimestamp) { | |
| this.isTesting = null; | |
| this.databaseFile = databaseFile; | |
| this.textLogDir = textLogDir; | |
| if (isTesting || Config.nofswriting) { | |
| this.isTesting = { files: /* @__PURE__ */ new Map(), db: isTesting || new Database(":memory:") }; | |
| } | |
| this.newestAllowedTimestamp = newestAllowedTimestamp; | |
| } | |
| async toTxt() { | |
| const database = this.isTesting?.db || new Database(this.databaseFile, { fileMustExist: true }); | |
| const roomids = database.prepare("SELECT DISTINCT roomid FROM modlog").all(); | |
| const globalEntries = []; | |
| for (const { roomid } of roomids) { | |
| if (!Config.nofswriting) | |
| console.log(`Reading ${roomid}...`); | |
| const results = database.prepare( | |
| `SELECT *, (SELECT group_concat(userid, ',') FROM alts WHERE alts.modlog_id = modlog.modlog_id) as alts FROM modlog WHERE roomid = ? ORDER BY timestamp ASC` | |
| ).all(roomid); | |
| const trueRoomID = roomid.replace(/^global-/, ""); | |
| let entriesLogged = 0; | |
| let entries = []; | |
| const insertEntries = async () => { | |
| if (roomid === "global") | |
| return; | |
| entriesLogged += entries.length; | |
| if (!Config.nofswriting && (entriesLogged % ENTRIES_TO_BUFFER === 0 || entriesLogged < ENTRIES_TO_BUFFER)) { | |
| process.stdout.clearLine(0); | |
| process.stdout.cursorTo(0); | |
| process.stdout.write(`Wrote ${entriesLogged} entries from '${trueRoomID}'`); | |
| } | |
| await this.writeFile(`${this.textLogDir}/modlog_${trueRoomID}.txt`, entries.join("")); | |
| entries = []; | |
| }; | |
| for (const result of results) { | |
| if (this.newestAllowedTimestamp && result.timestamp > this.newestAllowedTimestamp) | |
| break; | |
| const entry = { | |
| action: result.action, | |
| roomID: result.roomid?.replace(/^global-/, ""), | |
| visualRoomID: result.visual_roomid, | |
| userid: result.userid, | |
| autoconfirmedID: result.autoconfirmed_userid, | |
| alts: result.alts?.split(","), | |
| ip: result.ip, | |
| isGlobal: result.roomid?.startsWith("global-") || result.roomid === "global" || result.is_global, | |
| loggedBy: result.action_taker_userid, | |
| note: result.note, | |
| time: result.timestamp | |
| }; | |
| const rawLog = rawifyLog(entry); | |
| entries.push(rawLog); | |
| if (entry.isGlobal) { | |
| globalEntries.push(rawLog); | |
| } | |
| if (entries.length === ENTRIES_TO_BUFFER) | |
| await insertEntries(); | |
| } | |
| await insertEntries(); | |
| if (entriesLogged) | |
| process.stdout.write("\n"); | |
| } | |
| if (!Config.nofswriting) | |
| console.log(`Writing the global modlog...`); | |
| await this.writeFile(`${this.textLogDir}/modlog_global.txt`, globalEntries.join("")); | |
| } | |
| async writeFile(path, text) { | |
| if (this.isTesting) { | |
| const old = this.isTesting.files.get(path); | |
| return this.isTesting.files.set(path, `${old || ""}${text}`); | |
| } | |
| return (0, import_lib.FS)(path).append(text); | |
| } | |
| } | |
| class ModlogConverterTxt { | |
| constructor(databaseFile, textLogDir, isTesting, newestAllowedTimestamp) { | |
| this.isTesting = null; | |
| this.databaseFile = databaseFile; | |
| this.textLogDir = textLogDir; | |
| if (isTesting || Config.nofswriting) { | |
| this.isTesting = { | |
| files: isTesting || /* @__PURE__ */ new Map() | |
| }; | |
| } | |
| this.modlog = new Modlog( | |
| this.isTesting ? ":memory:" : this.databaseFile, | |
| // wait 15 seconds for DB to no longer be busy - this is important since I'm trying to do | |
| // a no-downtime transfer of text -> SQLite | |
| { sqliteOptions: { timeout: 15e3 } } | |
| ); | |
| this.newestAllowedTimestamp = newestAllowedTimestamp; | |
| } | |
| async toSQLite() { | |
| await this.modlog.readyPromise; | |
| const files = this.isTesting ? [...this.isTesting.files.keys()] : await (0, import_lib.FS)(this.textLogDir).readdir(); | |
| if (files.includes("modlog_global.txt")) { | |
| files.splice(files.indexOf("modlog_global.txt"), 1); | |
| files.unshift("modlog_global.txt"); | |
| } | |
| const globalEntries = {}; | |
| for (const file of files) { | |
| if (file === "README.md") | |
| continue; | |
| const roomid = file.slice(7, -4); | |
| const lines = this.isTesting ? this.isTesting.files.get(file)?.split("\n") || [] : (0, import_lib.FS)(`${this.textLogDir}/${file}`).createReadStream().byLine(); | |
| let entriesLogged = 0; | |
| let lastLine = void 0; | |
| let entries = []; | |
| const insertEntries = async () => { | |
| await this.modlog.writeSQL(entries); | |
| entriesLogged += entries.length; | |
| if (!Config.nofswriting) { | |
| process.stdout.clearLine(0); | |
| process.stdout.cursorTo(0); | |
| process.stdout.write(`Inserted ${entriesLogged} entries from '${roomid}'`); | |
| } | |
| entries = []; | |
| }; | |
| for await (const line of lines) { | |
| const entry = parseModlog(line, lastLine, roomid === "global"); | |
| lastLine = line; | |
| if (!entry) | |
| continue; | |
| if (this.newestAllowedTimestamp && entry.time > this.newestAllowedTimestamp) | |
| break; | |
| if (roomid !== "global" && globalEntries[entry.roomID]?.includes(line)) { | |
| continue; | |
| } | |
| if (entry.isGlobal) { | |
| if (!globalEntries[entry.roomID]) | |
| globalEntries[entry.roomID] = []; | |
| globalEntries[entry.roomID].push(line); | |
| } | |
| entries.push(entry); | |
| if (entries.length === ENTRIES_TO_BUFFER) | |
| await insertEntries(); | |
| } | |
| delete globalEntries[roomid]; | |
| await insertEntries(); | |
| if (entriesLogged) | |
| process.stdout.write("\n"); | |
| } | |
| return this.modlog.database; | |
| } | |
| } | |
| class ModlogConverterTest { | |
| constructor(inputDir, outputDir) { | |
| this.inputDir = inputDir; | |
| this.outputDir = outputDir; | |
| } | |
| async toTxt() { | |
| const files = await (0, import_lib.FS)(this.inputDir).readdir(); | |
| if (files.includes("modlog_global.txt")) { | |
| files.splice(files.indexOf("modlog_global.txt"), 1); | |
| files.push("modlog_global.txt"); | |
| } | |
| const globalEntries = []; | |
| for (const file of files) { | |
| if (file === "README.md") | |
| continue; | |
| const roomid = file.slice(7, -4); | |
| let entriesLogged = 0; | |
| let lastLine = void 0; | |
| let entries = []; | |
| const insertEntries = async () => { | |
| if (roomid === "global") | |
| return; | |
| entriesLogged += entries.length; | |
| if (!Config.nofswriting && (entriesLogged % ENTRIES_TO_BUFFER === 0 || entriesLogged < ENTRIES_TO_BUFFER)) { | |
| process.stdout.clearLine(0); | |
| process.stdout.cursorTo(0); | |
| process.stdout.write(`Wrote ${entriesLogged} entries from '${roomid}'`); | |
| } | |
| await (0, import_lib.FS)(`${this.outputDir}/modlog_${roomid}.txt`).append(entries.join("")); | |
| entries = []; | |
| }; | |
| const readStream = (0, import_lib.FS)(`${this.inputDir}/${file}`).createReadStream(); | |
| for await (const line of readStream.byLine()) { | |
| const entry = parseModlog(line, lastLine, roomid === "global"); | |
| lastLine = line; | |
| if (!entry) | |
| continue; | |
| const rawLog = rawifyLog(entry); | |
| if (roomid !== "global") | |
| entries.push(rawLog); | |
| if (entry.isGlobal) { | |
| globalEntries.push(rawLog); | |
| } | |
| if (entries.length === ENTRIES_TO_BUFFER) | |
| await insertEntries(); | |
| } | |
| await insertEntries(); | |
| if (entriesLogged) | |
| process.stdout.write("\n"); | |
| } | |
| if (!Config.nofswriting) | |
| console.log(`Writing the global modlog...`); | |
| await (0, import_lib.FS)(`${this.outputDir}/modlog_global.txt`).append(globalEntries.join("")); | |
| } | |
| } | |
| const ModlogConverter = { | |
| async convert(from, to, databasePath, textLogDirectoryPath, outputLogPath, newestAllowedTimestamp) { | |
| if (from === "txt" && to === "txt" && outputLogPath) { | |
| const converter = new ModlogConverterTest(textLogDirectoryPath, outputLogPath); | |
| await converter.toTxt(); | |
| console.log("\nDone!"); | |
| process.exit(); | |
| } else if (from === "sqlite" && to === "txt") { | |
| const converter = new ModlogConverterSQLite(databasePath, textLogDirectoryPath, void 0, newestAllowedTimestamp); | |
| await converter.toTxt(); | |
| console.log("\nDone!"); | |
| process.exit(); | |
| } else if (from === "txt" && to === "sqlite") { | |
| const converter = new ModlogConverterTxt(databasePath, textLogDirectoryPath, void 0, newestAllowedTimestamp); | |
| await converter.toSQLite(); | |
| console.log("\nDone!"); | |
| process.exit(); | |
| } | |
| } | |
| }; | |
| //# sourceMappingURL=converter.js.map | |