Jofthomas's picture
Jofthomas HF staff
Upload 4781 files
5c2ed06 verified
import { Utils } from '../../lib';
import { FS } from '../../lib/fs';
const SUSPECTS_FILE = 'config/suspects.json';
interface SuspectTest {
tier: string;
suspect: string;
date: string;
url: string;
}
interface SuspectsFile {
whitelist: string[];
suspects: { [format: string]: SuspectTest };
}
export let suspectTests: SuspectsFile = JSON.parse(FS(SUSPECTS_FILE).readIfExistsSync() || "{}");
function saveSuspectTests() {
FS(SUSPECTS_FILE).writeUpdate(() => JSON.stringify(suspectTests));
}
const defaults: SuspectsFile = {
whitelist: [],
suspects: {},
};
if (!suspectTests.whitelist && !suspectTests.suspects) {
const suspects = { ...suspectTests } as unknown as { [format: string]: SuspectTest };
suspectTests = { ...defaults, suspects };
saveSuspectTests();
}
function checkPermissions(context: Chat.CommandContext) {
const user = context.user;
if (suspectTests.whitelist?.includes(user.id)) return true;
context.checkCan('gdeclare');
}
export const commands: Chat.ChatCommands = {
suspect: 'suspects',
suspects: {
''(target, room, user) {
const suspects = suspectTests.suspects;
if (!Object.keys(suspects).length) {
throw new Chat.ErrorMessage("There are no suspect tests running.");
}
if (!this.runBroadcast()) return;
let buffer = '<strong>Suspect tests currently running:</strong>';
for (const i of Object.keys(suspects)) {
const test = suspects[i];
buffer += '<br />';
buffer += `${Utils.escapeHTML(test.tier)}: <a href="${test.url}">${Utils.escapeHTML(test.suspect)}</a> (${test.date})`;
}
return this.sendReplyBox(buffer);
},
edit: 'add',
async add(target, room, user) {
checkPermissions(this);
const [tier, suspect, date, url, ...reqs] = target.split(',').map(x => x.trim());
if (!(tier && suspect && date && url && reqs)) {
return this.parse('/help suspects');
}
const format = Dex.formats.get(tier);
if (format.effectType !== 'Format') throw new Chat.ErrorMessage(`"${tier}" is not a valid tier.`);
const suspectString = suspect.trim();
const [month, day] = date.trim().split(date.includes('-') ? '-' : '/');
const isValidDate = /[0-1]?[0-9]/.test(month) && /[0-3]?[0-9]/.test(day);
if (!isValidDate) throw new Chat.ErrorMessage("Dates must be in the format MM/DD.");
const dateActual = `${month}/${day}`;
const urlActual = url.trim();
if (!/^https:\/\/www\.smogon\.com\/forums\/(threads|posts)\//.test(urlActual)) {
throw new Chat.ErrorMessage("Suspect test URLs must be Smogon threads or posts.");
}
const reqData: Record<string, number> = {};
if (!reqs.length) {
return this.errorReply("At least one requirement for qualifying must be provided.");
}
for (const req of reqs) {
let [k, v] = req.split('=');
k = toID(k);
if (k === 'b') {
await this.parse(`/suspects setcoil ${format},${v}`);
continue;
}
if (!['elo', 'gxe', 'coil'].includes(k)) {
return this.errorReply(`Invalid requirement type: ${k}. Must be 'coil', 'gxe', or 'elo'.`);
}
if (k === 'coil' && !reqs.some(x => toID(x).startsWith('b'))) {
throw new Chat.ErrorMessage("COIL reqs are specified, but you have not provided a B value (with the argument `b=num`)");
}
const val = Number(v);
if (isNaN(val) || val < 0) {
return this.errorReply(`Invalid value: ${v}`);
}
if (reqData[k]) {
return this.errorReply(`Requirement type ${k} specified twice.`);
}
reqData[k] = val;
}
const [out, error] = await LoginServer.request(suspectTests.suspects[format.id] ? "suspects/edit" : "suspects/add", {
format: format.id,
reqs: JSON.stringify(reqData),
url: urlActual,
});
if (out?.actionerror || error) {
throw new Chat.ErrorMessage("Error adding suspect test: " + (out?.actionerror || error?.message));
}
this.privateGlobalModAction(`${user.name} ${suspectTests.suspects[format.id] ? "edited the" : "added a"} ${format.name} suspect test.`);
this.globalModlog('SUSPECTTEST', null, `${suspectTests.suspects[format.id] ? "edited" : "added"} ${format.name}`);
suspectTests.suspects[format.id] = {
tier: format.name,
suspect: suspectString,
date: dateActual,
url: urlActual,
};
saveSuspectTests();
this.sendReply(`Added a suspect test notice for ${suspectString} in ${format.name}.`);
if (reqData.coil) this.sendReply('Remember to add a B value for your test\'s COIL setting with /suspects setbvalue.');
},
end: 'remove',
delete: 'remove',
async remove(target, room, user) {
checkPermissions(this);
const format = toID(target);
const test = suspectTests.suspects[format];
if (!test) return this.errorReply(`There is no suspect test for '${target}'. Check spelling?`);
const [out, error] = await LoginServer.request('suspects/end', {
format,
});
if (out?.actionerror || error) {
throw new Chat.ErrorMessage(`Error ending suspect: ${out?.actionerror || error?.message}`);
}
this.privateGlobalModAction(`${user.name} removed the ${test.tier} suspect test.`);
this.globalModlog('SUSPECTTEST', null, `removed ${test.tier}`);
delete suspectTests.suspects[format];
saveSuspectTests();
this.sendReply(`Removed a suspect test notice for ${test.suspect} in ${test.tier}.`);
this.sendReply(`Remember to remove COIL settings with /suspects deletecoil if you had them enabled.`);
},
whitelist(target, room, user) {
this.checkCan('gdeclare');
const userid = toID(target);
if (!userid || userid.length > 18) {
if (suspectTests.whitelist.length) {
this.sendReplyBox(`Current users with /suspects access: <username>${suspectTests.whitelist.join('</username>, <username>')}</username>`);
}
return this.parse(`/help suspects`);
}
if (suspectTests.whitelist.includes(userid)) {
throw new Chat.ErrorMessage(`${userid} is already whitelisted to add suspect tests.`);
}
this.privateGlobalModAction(`${user.name} whitelisted ${userid} to add suspect tests.`);
this.globalModlog('SUSPECTTEST', null, `whitelisted ${userid}`);
suspectTests.whitelist.push(userid);
saveSuspectTests();
},
unwhitelist(target, room, user) {
this.checkCan('gdeclare');
const userid = toID(target);
if (!userid || userid.length > 18) {
return this.parse(`/help suspects`);
}
const index = suspectTests.whitelist.indexOf(userid);
if (index < 0) {
throw new Chat.ErrorMessage(`${userid} is not whitelisted to add suspect tests.`);
}
this.privateGlobalModAction(`${user.name} unwhitelisted ${userid} from adding suspect tests.`);
this.globalModlog('SUSPECTTEST', null, `unwhitelisted ${userid}`);
suspectTests.whitelist.splice(index, 1);
saveSuspectTests();
},
async verify(target, room, user) {
const formatid = toID(target);
if (!suspectTests.suspects[formatid]) {
throw new Chat.ErrorMessage("There is no suspect test running for the given format.");
}
const [out, error] = await LoginServer.request("suspects/verify", {
formatid,
userid: user.id,
});
if (error) {
throw new Chat.ErrorMessage("Error verifying for suspect: " + error.message);
}
if (out?.actionerror) {
throw new Chat.ErrorMessage(out.actionerror);
}
this.sendReply(
out.result ?
`You have successfully verified for the ${formatid} suspect test.` :
`You could not verify for the ${formatid} suspect test, as you do not meet the requirements.`
);
},
help() {
return this.parse('/help suspects');
},
deletebvalue: 'setbvalue',
deletecoil: 'setbvalue',
sbv: 'setbvalue',
dbv: 'setbvalue',
sc: 'setbvalue',
dc: 'setbvalue',
setcoil: 'setbvalue',
async setbvalue(target, room, user, connection, cmd) {
checkPermissions(this);
if (!toID(target)) {
return this.parse(`/help ${cmd}`);
}
const [formatStr, source] = this.splitOne(target);
const format = Dex.formats.get(formatStr);
let bVal: number | undefined = parseFloat(source);
if (cmd.startsWith('d')) {
bVal = undefined;
} else if (!source || isNaN(bVal) || bVal < 1) {
throw new Chat.ErrorMessage(`Specify a valid COIL B value.`);
}
if (!toID(formatStr) || !format.exists) {
throw new Chat.ErrorMessage(`Specify a valid format to set a COIL B value for. Check spelling?`);
}
this.sendReply(`Updating...`);
const [res, error] = await LoginServer.request('updatecoil', {
format: format.id,
coil_b: bVal,
});
if (error) {
throw new Chat.ErrorMessage(error.message);
}
if (!res || res.actionerror) {
throw new Chat.ErrorMessage(res?.actionerror || "The loginserver is currently disabled.");
}
this.globalModlog(`${source ? 'SET' : 'REMOVE'}BVALUE`, null, `${format.id}${bVal ? ` to ${bVal}` : ""}`);
this.addGlobalModAction(
`${user.name} ${bVal ? `set B value for ${format.name} to ${bVal}` : `removed B value for ${format.name}`}.`
);
if (source) {
return this.sendReply(`COIL B value for ${format.name} set to ${bVal}.`);
} else {
return this.sendReply(`Removed COIL B value for ${format.name}.`);
}
},
setbvaluehelp: [
`/suspects setbvalue OR /suspects sbv [formatid], [B value] - Activate COIL ranking for the given [formatid] with the given [B value].`,
`Requires: suspect whitelist ~`,
],
},
suspectshelp() {
this.sendReplyBox(
`Commands to manage suspect tests:<br />` +
`<code>/suspects</code>: displays currently running suspect tests.<br />` +
`<code>/suspects add [tier], [suspect], [date], [link], [...reqs]</code>: adds a suspect test. Date in the format MM/DD. ` +
`Reqs in the format [key]=[value], where valid keys are 'coil', 'elo', and 'gxe', delimited by commas. At least one is required. <br />` +
`(note that if you are using COIL, you must set a B value indepedently with <code>/suspects setcoil</code>). Requires: ~<br />` +
`<code>/suspects remove [tier]</code>: deletes a suspect test. Requires: ~<br />` +
`<code>/suspects whitelist [username]</code>: allows [username] to add suspect tests. Requires: ~<br />` +
`<code>/suspects unwhitelist [username]</code>: disallows [username] from adding suspect tests. Requires: ~<br />` +
`<code>/suspects setbvalue OR /suspects sbv [formatid], [B value]</code>: Activate COIL ranking for the given [formatid] with the given [B value].` +
`Requires: suspect whitelist ~`
);
},
};