Skip to content

Instantly share code, notes, and snippets.

@Yanrishatum
Last active January 31, 2023 20:44
Show Gist options
  • Save Yanrishatum/8125378fb548e8a882a368c7ff5ee9ed to your computer and use it in GitHub Desktop.
Save Yanrishatum/8125378fb548e8a882a368c7ff5ee9ed to your computer and use it in GitHub Desktop.
Unstrusted LOGgy: UX tweaks for reading dem circus
// ==UserScript==
// @name Untrusted LOGgy
// @namespace https://yanrishatum.ru
// @version 1.1-99
// @description Improve log reading experience for Untrusted
// @author Yanrishatum
// @match *://*.playuntrusted.com/opsec/*
// @match *://www.playuntrusted.com/manual/skills/
// @grant GM_xmlhttpRequest
// @grant GM.xmlHttpRequest
// @connect playuntrusted.com
// ==/UserScript==
(function() {
var untrusted = {};
let skills = {};
let skillsLoading = false;
if (localStorage["untrusted_skills"])
skills = JSON.parse(localStorage["untrusted_skills"]);
else
fetchSkills();
async function fetchSkills() {
skillsLoading = true;
if (typeof GM !== "undefined")
GM_xmlhttpRequest = GM.xmlHttpRequest;
// DOES NOT WORK: Servers don't have CORS properly configured
// let xhr = new XMLHttpRequest();
// xhr.responseType = "document";
// xhr.open("GET", "https://www.playuntrusted.com/manual/skills/");
// await new Promise((r) => { xhr.onload = r; xhr.send(); });
// cacheSkills(xhr.responseXML!);
try {
const docStr = await new Promise((r) => {
GM_xmlhttpRequest({ method: "GET", url: "https://www.playuntrusted.com/manual/skills/", onload: (res) => r(res.responseText) });
});
const doc = new DOMParser().parseFromString(docStr, "text/html");
cacheSkills(doc);
}
catch (e) {
console.log(e);
// Not in userscript
}
skillsLoading = false;
// TODO: Force update
}
function cacheSkills(doc) {
skillsLoading = true;
console.log(doc);
for (const hr of doc.querySelectorAll("#skillstable>hr")) {
const left = hr.nextElementSibling;
const right = left.nextElementSibling;
const idEl = left.querySelector("a");
const nameEl = left.querySelector("h1");
idEl.remove();
nameEl.remove();
const skill = {
id: idEl.getAttribute("name"),
name: nameEl.textContent.trim(),
description: left.innerHTML,
img: right.querySelector("img").src,
stats: right.querySelector(":scope>p").innerHTML
};
left.prepend(nameEl);
left.prepend(idEl);
skills[skill.id] = skill;
}
localStorage["untrusted_skills"] = JSON.stringify(skills);
console.log("Finished skill caching");
// TODO: Force update
}
if (location.pathname == '/manual/skills/') {
cacheSkills(document);
throw "Early exit: Manual page, only need to fetch skills";
}
const roles = {};
function makeRole(name, abbr, sabbr, fact) {
var r = { name: name, abbr: abbr, short_abbr: sabbr, faction: fact };
roles[name] = r;
}
// Netsec
makeRole('Operation Leader', 'OL', ' OL ', 'NETSEC');
makeRole('Original Operation Leader', 'OOL', 'OOL ', 'NETSEC');
makeRole('Sociopath Operation Leader', 'SOL', 'SOL ', 'NETSEC');
makeRole('CCTV Specialist', 'CCTV', 'CCTV', 'NETSEC');
makeRole('Enforcer', 'ENFO', 'ENFO', 'NETSEC'); // NO OFFICIAL ABBR
makeRole('Inside Man', 'IM', ' IM ', 'NETSEC');
makeRole('Analyst', 'ANAL', 'ANAL', 'NETSEC'); // NO OFFICIAL ABBR
makeRole('Network Specialist', 'NS', ' NS ', 'NETSEC');
makeRole('Social Engineer', 'SE', ' SE ', 'NETSEC');
makeRole('Blackhat', 'BH', ' BH ', 'NETSEC');
makeRole('Improvised Hacker', 'IH', ' IH ', 'NETSEC');
makeRole('Spearphisher', 'SPH', ' SP ', 'NETSEC');
// Neutral
makeRole('Bounty Hunter', 'BOHU', 'BOHU', "Neutral");
makeRole('Corupt Detective', 'CD', ' CD ', "Neutral"); // NO OFFICIAL ABBR
makeRole('Double-crosser', 'DC', ' DC ', "Neutral");
makeRole('Journalist', 'JOURNO', 'JOUR', "Neutral");
makeRole('Loose Cannon', 'LC', ' LC ', "Neutral"); // Technically AGENT/NETSEC based on RNG
makeRole('Loose Cannon (AGENT)', 'LC', ' LC ', "AGENT");
makeRole('Loose Cannon (NETSEC)', 'LC', ' LC ', "NETSEC");
makeRole('Script Kiddie', 'SK', ' SK ', "Neutral");
makeRole('Panicked Blabbermouth', 'PB', ' PB ', "Neutral");
makeRole('Resentful Criminal', 'RC', ' RC ', "Neutral");
makeRole('Sociopath', 'SOCIO', 'SOCI', "Neutral");
makeRole('Rival Hacker', 'RH', ' RH ', "Neutral");
// Agent
makeRole('Agent Leader', 'AL', ' AL ', "AGENT");
makeRole('Field Agent', 'FA', ' FA ', "AGENT");
makeRole('Forensics Specialist', 'FS', ' FS ', "AGENT");
makeRole('Mole (Converted Field Ops)', 'MOLE(F)', 'MILF', "AGENT"); // NO OFFICIAL ABBR
makeRole('Mole (Converted Inv.)', 'MOLE(I)', 'MOLI', "AGENT"); // NO OFFICIAL ABBR
makeRole('Mole (Converted Offensive)', 'MOLE(O)', 'MOLO', "AGENT"); // NO OFFICIAL ABBR
makeRole('Mole (Unknown Class)', 'MOLE(D)', 'MOLD', "AGENT"); // In case I didn't cover some mole conversion
makeRole('Runaway Snitch', 'RS', ' RS ', "AGENT");
makeRole('unknown', '?', ' ?? ', "Neutral");
{ // CSS
const css = `
/* #include-css(../css/untrusted_log.css) */
.col-agent {
color: #fffeb2;
}
.col-netsec {
color: #ffd5de;
}
.col-neutral {
color: #cae4f6;
}
.ip-ttip {
text-decoration: underline;
}
.skill {
display: grid;
grid-template: 'a a c'
'b b d';
gap: 5px;
white-space: normal;
max-width: 600px;
}
.skill > h3 {
grid-area: a;
justify-self: center;
align-self: center;
}
.skill > div:nth-child(2) {
grid-area: b;
}
.skill > img {
grid-area: c;
justify-self: end;
width: 96px;
}
.skill > div:last-child {
grid-area: d;
}
.options label {
white-space: nowrap;
}
.options select {
background: black;
color: white;
}
.options input[type='checkbox'] {
width: 1em;
height: 1em;
background: black;
border: 1px solid white;
appearance: none;
vertical-align: text-bottom;
transition: background-color 0.1s ease;
cursor: pointer;
}
.options input[type='checkbox']:hover {
background-color: #272727;
}
.options input[type='checkbox']:checked {
background-color: white;
}
.options input[type='checkbox']:checked:hover {
background-color: #c4c4c4;
}
.tooltip {
position: absolute;
display: none;
color: #fff;
padding: 8px;
margin: 0;
}
.event-target-name {
margin-right: 1ch;
display: inline-flex;
flex-direction: row;
}
.event-target-name .name {
order: 1;
}
.event-target-name .abbr {
order: 2;
}
.abbr {
white-space: nowrap;
}
.events-align-left .event-align {
text-align: left !important;
}
.show-abbr-short.events-align-left .event-target-name .abbr {
order: 0;
}
.abbr-short, .abbr-full {
display: none;
}
.show-abbr-short .abbr-short {
display: initial;
}
.show-abbr-full .abbr-full {
display: initial;
}
.show-abbr-none .abbr {
display: none;
}
.color-abbrs-faction .abbr.agent {
color: #fffeb2;
}
.color-abbrs-faction .abbr.netsec {
color: #ffd5de;
}
.color-abbrs-faction .abbr.neutral {
color: #cae4f6;
}
.compact-chat .broadcast pre, .compact-chat .chat-message pre {
margin: 0;
padding: 4px 10px;
white-space: normal;
}
.compact-chat .broadcast pre {
padding-left: 24px;
}
.compact-chat .chat-message > td:first-child > table {
margin-bottom: 0;
display: flex;
align-items: center;
}
.compact-chat .chat-message > td:first-child > table td {
margin: 0 !important;
padding: 0;
}
.compact-chat .chat-message > td:first-child > table td:nth-child(2) {
flex-grow: 1;
text-align: center;
}
.compact-chat .chat-message > td:first-child > table tbody, .compact-chat .chat-message > td:first-child > table tr {
display: contents;
}
.compact-chat .chat-message img {
width: 24px !important;
height: 24px !important;
}
`;
const style = document.querySelector("#untrusted-log-css") ?? document.createElement("style");
style.id = "#untrusted-log-css";
style.textContent = css;
document.head.appendChild(style);
}
const OPTS_KEY = "untrusted_opts";
// Options menu
const options = {
abbrs: "short",
colorCode: "faction",
leftAlignEvents: true,
compactChat: true,
};
if (localStorage[OPTS_KEY])
Object.assign(options, JSON.parse(localStorage[OPTS_KEY]));
function saveOpts() { localStorage[OPTS_KEY] = JSON.stringify(options); applyOpts(); }
{
const optsCont = document.querySelector("h1").nextElementSibling.nextElementSibling;
const panel = document.createElement("div");
panel.classList.add("options");
panel.innerHTML = `
<hr/>
<div class="warn">Note: You need to realod the page for some changes to apply!</div>
<label>Display abbreviations: <select id="abbr_mode">
<option value="none">Hidden</option>
<option value="short">Short (4 chars)</option>
<option value="full">Full (variable length)</option>
</select></label>
<label>Abbreviation colors: <select id="color_code">
<option value="none">White</option>
<option value="faction">By faction</option>
<option value="name">By name (DOES NOT WORK)</option>
</select></label>
<label><input type="checkbox" id="events_left_align"/>Align events on the left (readability)</label>
<label><input type="checkbox" id="compact_chat"/>Compact chat</label>
`;
function bindSelect(key, query) {
const sel = panel.querySelector(query);
sel.value = options[key];
sel.onchange = () => { options[key] = sel.value; saveOpts(); };
}
function bindCheck(key, query) {
const check = panel.querySelector(query);
check.checked = options[key];
check.onchange = () => { options[key] = check.checked; saveOpts(); };
}
bindSelect("abbrs", "#abbr_mode");
bindSelect("colorCode", "#color_code");
bindCheck("leftAlignEvents", "#events_left_align");
bindCheck("compactChat", "#compact_chat");
optsCont.appendChild(panel);
}
function applyOpts() {
document.body.classList.toggle('show-abbr-short', options.abbrs == "short");
document.body.classList.toggle('show-abbr-full', options.abbrs == "full");
document.body.classList.toggle('show-abbr-none', options.abbrs == "none");
document.body.classList.toggle('color-abbrs-none', options.colorCode == "none");
document.body.classList.toggle('color-abbrs-faction', options.colorCode != "none");
document.body.classList.toggle('color-abbrs-name', options.colorCode == "name");
document.body.classList.toggle("events-align-left", options.leftAlignEvents);
document.body.classList.toggle('compact-chat', options.compactChat);
}
applyOpts();
const players = {};
let hackDuration = 7;
let ol = null;
let root = null;
class Player {
player; // Player name
level; // Player level
handle; // Player in-game handle (colors)
item; // Held item
img; // Avatar
role;
disguisedAs;
skills;
constructor(el) {
if (el) {
this.player = el.querySelector("div>a").textContent;
this.level = el.querySelector(":scope>div:last-child").firstChild.textContent;
this.handle = el.querySelector("div>div>div").textContent.replace(/^as "(.+)"$/, "$1");
this.item = el.querySelector("div>div>div:last-child").textContent;
this.img = el.querySelector("img").src;
}
else {
this.player = 'Unknown';
this.level = 'Level -';
this.handle = 'Unknown';
this.item = 'All the items';
this.img = undefined;
}
this.role = roles['unknown'];
this.disguisedAs = null;
this.skills = [];
players[this.handle] = this;
}
abbr(pre = '', post = '') {
return `<span class='abbr ${this.role.faction.toLowerCase()} name-${this.handle.substr(this.handle.indexOf(".") + 1).toLowerCase()}'>${pre}<span class='abbr-short'>[${this.role.short_abbr}]</span><span class='abbr-full'>[${this.role.abbr}]</span>${post}</span>`;
}
image(size) {
if (this.img)
return `<img style='width:${size}px;height:${size}px;' src='${this.img}'>`;
return '';
}
toString() {
let base = `${this.image(48)} ${this.player} as <b>${this.role.name}</b>`.trimStart();
if (this.disguisedAs)
base += ` <i>(${this.disguisedAs.name})</i>`;
return base;
}
}
// Make players
new Player(null);
Array.from(document.querySelectorAll("h1")).find(e => e.textContent == "OPSEC OPERATIVES LIST").nextElementSibling.nextElementSibling.querySelectorAll("td").forEach(el => new Player(el));
;
;
const TOPO_STEP_X = 98;
const TOPO_STEP_Y = 78;
const TOPO_SIZE = 48;
const TOPO_CENTER = 24;
const SERVER_SIZE_HACKED = 3550;
const SERVER_SIZE_UNHACKED = 3886;
const COMPUTER_SIZE_HACKED = 2906;
const COMPUTER_SIZE_UNHACKED = 2990;
const ipRegex = /\d+\.\d+\.\d+\.\d+/;
function topoImg(src) {
var i = new Image(TOPO_SIZE, TOPO_SIZE);
i.src = src;
i.loading = "eager";
return i;
}
// Had to embed to avoid topo breaking
const topoImages = {
"computer": topoImg(""),
"computer_hacked": topoImg(""),
"server": topoImg(""),
"server_hacked": topoImg("")
};
class Topology {
// lines: HTMLCanvasElement;
canvas;
ctx;
nodes;
nodesByPos;
conns;
constructor() {
let lines = document.querySelector("#topology");
this.canvas = document.createElement("canvas");
this.canvas.width = 4 * TOPO_STEP_X - TOPO_CENTER;
this.canvas.height = 3 * TOPO_STEP_Y - TOPO_CENTER;
this.ctx = this.canvas.getContext("2d");
this.nodes = [];
this.nodesByPos = [];
this.conns = [];
let w = 0, h = 0;
for (const img of lines.nextElementSibling.nextElementSibling.querySelectorAll("img")) {
const node = {
ip: "000.000.000.000",
id: img.id,
connections: [],
x: parseFloat(img.style.marginLeft) / TOPO_STEP_X,
y: parseFloat(img.style.marginTop) / TOPO_STEP_Y,
kind: img.src.length == SERVER_SIZE_HACKED || img.src.length == SERVER_SIZE_UNHACKED ? "server" : "computer",
hacked: false
};
if (w < node.x)
w = node.x;
if (h < node.y)
h = node.y;
// const imgName = node.kind + (img.src.length == SERVER_SIZE_HACKED || img.src.length == COMPUTER_SIZE_HACKED ? "_hacked" : "") as ImageNames;
// if (!this.images[imgName]) this.images[imgName] = img;
this.nodes.push(node);
if (!this.nodesByPos[node.x])
this.nodesByPos[node.x] = [];
this.nodesByPos[node.x][node.y] = node;
}
this.canvas.width = (w + 1) * TOPO_STEP_X - TOPO_CENTER;
this.canvas.height = (h + 1) * TOPO_STEP_Y - TOPO_CENTER;
// Analyze topology connections
let from = null, x, y;
for (const line of lines.nextElementSibling.textContent.split("\n")) {
let tmp = /ctx.moveTo\((\d+),(\d+)\);/.exec(line);
if (tmp) {
x = (parseFloat(tmp[1]) - 24) / TOPO_STEP_X;
y = (parseFloat(tmp[2]) - 24) / TOPO_STEP_Y;
from = this.nodesByPos[x][y];
}
else if ((tmp = /ctx.lineTo\((\d+),(\d+)\);/.exec(line))) {
x = (parseFloat(tmp[1]) - 24) / TOPO_STEP_X;
y = (parseFloat(tmp[2]) - 24) / TOPO_STEP_Y;
const node = this.nodesByPos[x][y];
const conn = { a: from, b: node };
from.connections.push(conn);
node.connections.push(conn);
this.conns.push(conn);
}
}
// Analyze topology IPs
const iter = lines.nextElementSibling.nextElementSibling.querySelector("script").textContent.split("\n").map(s => s.trim()).filter(s => s.includes("logs for:") || s.includes(".mouseover"))[Symbol.iterator]();
let ival;
do {
ival = iter.next();
if (ival.done)
break;
const node = this.nodes.find(n => n.id == "node" + parseInt(/"#node(\d+)"/.exec(ival.value)[1]));
ival = iter.next();
if (node)
node.ip = ipRegex.exec(ival.value)[0];
} while (!ival.done);
}
hack(ip) {
const node = this.nodes.find(n => n.ip == ip);
if (node)
node.hacked = true;
}
rollback(ip) {
const node = this.nodes.find(n => n.ip == ip);
if (node)
node.hacked = false;
}
paint(ip) {
const hackState = new Map();
for (const n of this.nodes)
hackState.set(n.ip, n.hacked);
const ctx = this.ctx;
const canvas = this.canvas;
const conns = this.conns;
const nodes = this.nodes;
return {
toString() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
// Draw connections
for (const conn of conns) {
ctx.beginPath();
ctx.strokeStyle = hackState.get(conn.a.ip) && hackState.get(conn.b.ip) ? "#00ff01" : "#fff";
ctx.lineWidth = 2;
ctx.moveTo(conn.a.x * TOPO_STEP_X + TOPO_CENTER, conn.a.y * TOPO_STEP_Y + TOPO_CENTER);
ctx.lineTo(conn.b.x * TOPO_STEP_X + TOPO_CENTER, conn.b.y * TOPO_STEP_Y + TOPO_CENTER);
ctx.stroke();
ctx.closePath();
}
// Draw nodes
for (const node of nodes) {
const img = topoImages[(node.kind + (hackState.get(node.ip) ? "_hacked" : ""))];
if (node.ip == ip) {
ctx.beginPath();
ctx.strokeStyle = "#00ff01";
ctx.roundRect(node.x * TOPO_STEP_X, node.y * TOPO_STEP_Y, TOPO_SIZE, TOPO_SIZE, 4);
ctx.stroke();
ctx.closePath();
}
if (!img)
console.log("WTF");
ctx.drawImage(img, node.x * TOPO_STEP_X, node.y * TOPO_STEP_Y, TOPO_SIZE, TOPO_SIZE);
}
return canvas.toDataURL();
}
};
}
}
const topo = new Topology();
// Process log
const logStart = Array.from(document.querySelectorAll("h1")).find(e => e.textContent == "OPSEC LOG");
const logTable = logStart.nextElementSibling.nextElementSibling.querySelector("table tbody");
function* logIter() {
let el = logTable.firstElementChild;
while (el.nextElementSibling) {
if (el.textContent == '' && el.nextElementSibling) {
el = el.nextElementSibling; // Skip empty ones
continue;
}
yield el;
el = el.nextElementSibling;
}
return el;
}
let stage = 'roles';
function parseEvent(ev) {
const privateExec = /^\(([^)]+)\)\s(.+)$/.exec(ev);
if (privateExec) {
return {
isPrivate: true,
target: players[privateExec[1]],
message: privateExec[2]
};
}
else {
// TODO
return {
isPrivate: false,
target: players['Everyone'],
message: ev
};
}
}
const ttip = document.createElement("pre");
ttip.classList.add("tooltip");
document.body.appendChild(ttip);
let ttipCounter = 0;
/** @param {HTMLElement} el */
function addTooltip(el, text) {
el.addEventListener("mouseenter", (e) => {
ttipCounter++;
const bbox = el.getBoundingClientRect();
ttip.style.left = (bbox.left + window.scrollX) + "px";
ttip.style.top = (window.scrollY + bbox.bottom + 2) + "px";
ttip.style.display = 'block';
ttip.innerHTML = text.toString();
});
el.addEventListener("mouseleave", () => {
ttipCounter--;
if (ttipCounter <= 0) {
ttipCounter = 0;
ttip.style.display = 'none';
}
});
}
function addSkillTooltip(el, id) {
const disp = {
toString() {
if (skillsLoading)
return 'Skill list is still being loaded!';
const skill = skills[id];
if (!skill)
return "Unknown skill!";
return `
<div class="skill">
<h3>${skill.name}</h3>
<div>${skill.description}</div>
<img src="${skill.img}">
<div>${skill.stats}</div>
</div>`.trim();
}
};
addTooltip(el, disp);
}
function processIP(text) {
let ip = ipRegex.exec(text);
if (ip) {
const left = text.substring(0, ip.index);
const right = text.substring(ip.index + ip[0].length);
const disp = document.createElement("span");
disp.textContent = ip[0];
disp.classList.add("ip-ttip");
addTooltip(disp, `<img src="${topo.paint(ip[0])}">`);
let result = [left, disp].concat(processIP(right));
return result;
}
else
return [text];
}
let line, ev;
console.time("Log parser");
for (const logLine of logIter()) {
switch (stage) {
case 'roles':
line = logLine.textContent.trim();
if (line == 'Preparation Night') {
stage = 'chat';
break;
}
let durExec = /The hack must be completed within (\d+) days?./.exec(line)?.[1];
if (durExec) {
hackDuration = parseInt(durExec);
break;
}
ev = parseEvent(line);
// console.log(ev, line);
const player = ev.target;
const bEl = logLine.querySelector("b");
let reg;
if (bEl) {
player.role = roles[bEl.textContent.trim()] ?? roles['unknown'];
if (player.role.name == 'Operation Leader')
ol = player; // Assign OL for anonymous messages
}
else if ((reg = /^You have a cover as '([^']+)'/.exec(ev.message))) {
player.disguisedAs = roles[reg[1]];
}
else if ((reg = /'([^']+)' skill.$/.exec(ev.message))) {
player.skills.push(reg[1]);
}
if (player.role.name == 'Loose Cannon') {
if (ev.message.includes('make sure AGENT wins'))
player.role = roles['Loose Cannon (AGENT)'];
else if (ev.message.includes('make sure NETSEC wins'))
player.role = roles['Loose Cannon (NETSEC)'];
}
break;
case 'chat':
if (logLine.querySelector(":scope>td[colspan='2']>pre")) {
// Broadcast
logLine.classList.add("broadcast");
const sender = logLine.querySelector("b");
if (sender)
addTooltip(sender, ol.toString());
}
else if (logLine.querySelector(":scope>td[colspan='2']")?.textContent?.trim()?.startsWith("Events")) {
// End of chat
stage = 'events';
break;
}
else {
if (!logLine.querySelector(":scope>td[colspan='2']"))
logLine.classList.add("chat-message");
// Add tooltips for all users
for (let userImg of logLine.querySelectorAll("img")) {
let name = userImg.parentElement.textContent.trim();
if (name.endsWith(":"))
name = name.substring(0, name.length - 1);
const player = players[name];
if (player) {
userImg.insertAdjacentHTML("afterend", player.abbr('', ' '));
addTooltip(userImg.parentElement, player.toString());
}
}
const chatText = logLine.querySelector("td:nth-child(2)>pre, td:nth-child(2)>i") ?? logLine.querySelector("td:nth-child(2)");
if (chatText?.lastChild?.nodeType == Node.TEXT_NODE) {
const proc = processIP(chatText.lastChild.textContent);
if (proc.length > 1) {
chatText.lastChild.remove();
chatText.append(...proc);
}
else {
//TODO: on Dr.X
}
}
}
break;
case 'events':
line = logLine.textContent.trim();
if (line.startsWith("Night") || line.startsWith("Day")) {
// End of events
stage = 'chat';
break;
}
const ctd = logLine.querySelector("td[style*='text-align:center']");
if (ctd)
ctd.classList.add("event-align");
ev = parseEvent(line);
if (ev.isPrivate) {
const player = ev.target;
// Reclass
if (ev.message.includes('and stolen their precious, precious Operation Leader identity')) {
player.role = roles['Sociopath Operation Leader'];
ol.role = roles['Original Operation Leader'];
ol = player;
}
else if (ev.message.includes('you are now the new Operation Leader')) {
player.role = roles['Operation Leader'];
ol.role = roles['Original Operation Leader']; // Defo gonna conflict if socio is in play
ol = player;
}
else if (ev.message.includes('and became an Improvised Hacker')) {
player.role = roles['Improvised Hacker'];
}
else if (ev.message.includes('You are panicking - NETSEC is out to get you')) {
player.role = roles['Runaway Snitch'];
}
else if (ev.message.includes('All of the death around you made you question your life choices')) {
player.role = roles['Panicked Blabbermouth'];
}
else if (ev.message.includes('you are now the Field Agent') || ev.message.includes('You have been promoted to Field Agent')) {
player.role = roles['Field Agent'];
}
else if (ev.message.includes('Do not reveal that you\'ve become a mole')) {
switch (player.role.name) {
case 'CCTV Specialist':
case 'Enforcer':
case 'Inside Man':
case 'Loose Cannon':
case 'Loose Cannon (AGENT)':
case 'Loose Cannon (NETSEC)':
player.role = roles['Mole (Converted Field Ops)'];
break;
case 'Analyst':
case 'Network Specialist':
case 'Social Engineer':
case 'Double-crosser':
player.role = roles['Mole (Converted Inv.)'];
break;
case 'Improvised Hacker':
case 'Spearphisher':
case 'Panicked Blabbermouth':
case 'Resentful Criminal':
case 'Rival Hacker':
player.role = roles['Mole (Converted Offensive)'];
break;
default:
player.role = roles['Mole (Unknown Class)'];
break;
}
}
else if (ev.message.includes('and you have been granted root access')) {
root = player;
}
const td = logLine.querySelector("td");
const nameSub = document.createElement("span");
let disp = `<span class="name">(${player.handle})</span>${player.abbr('', '')}`;
nameSub.innerHTML = disp;
nameSub.classList.add("event-target-name");
addTooltip(nameSub, player.toString());
td.textContent = "";
td.append(nameSub, ...processIP(ev.message));
}
else {
if (ev.message.includes("NETSEC now has root privileges on")) {
let ip = ipRegex.exec(ev.message);
if (ip)
topo.hack(ip[0]);
}
else if (ev.message.includes("Someone rolled back")) {
let ip = ipRegex.exec(ev.message);
if (ip)
topo.rollback(ip[0]);
}
else if (ev.message.includes("The suspect was charged as Operation Leader") || ev.message.includes("Rumor is that the victim was a(n) Operation Leader")) {
if (root) {
ol = root;
root = null;
}
}
let data = processIP(ev.message);
if (data.length > 1) {
const td = logLine.querySelector("td");
td.innerHTML = "";
td.append(...data);
}
}
}
}
console.timeEnd("Log parser");
console.time("Skill tooltips");
for (const skillLink of document.querySelectorAll("a[href*='manual/skills/#']")) {
addSkillTooltip(skillLink, new URL(skillLink.href).hash.substring(1));
}
console.timeEnd("Skill tooltips");
})()

1.1-88/1.1-99

  • Fixed final topology with no unhacked server/laptop breaking stuff.

1.1-87

  • Add compact chat option
  • Make topology dimensions dynamic (4x4 grid for large lobby)

1.0-80

  • First public release
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment