-
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathindex.js
379 lines (354 loc) · 20.1 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
const Discord = require('discord.js');
const bot = new Discord.Client();
const config = require('./config.json');
const needle = require('needle');
const regexLink = /\[\[(.+?)(\|.*?)?\]\]/g;
const regexTemp = /\{\{(.+?)(\|.*?)?\}\}/g;
const regexRaw = /--(.+?)(\|.*?)?--/g;
const TYPE_NORMAL = 'normal';
const TYPE_TEMPLATE = 'template';
const TYPE_RAW = 'raw';
var wikis = require('./_wikis.json');
const db = require('better-sqlite3')('_prefs.db');
bot.once('ready', () => {
db.prepare('CREATE TABLE IF NOT EXISTS guilds (GuildID TEXT NOT NULL PRIMARY KEY, WikiKey TEXT NOT NULL)').run();
db.prepare('CREATE TABLE IF NOT EXISTS channels (ChannelID TEXT NOT NULL PRIMARY KEY, WikiKey TEXT NOT NULL)').run();
db.prepare('CREATE TABLE IF NOT EXISTS users (UserID TEXT NOT NULL PRIMARY KEY, DisabledLinks TEXT)').run();
bot.user.setActivity(`Nindies | ${config.prefix}help`, {type: 'PLAYING'});
console.log(`Ready at ${new Date().toUTCString()} - ${bot.guilds.cache.size} guilds, ${bot.channels.cache.size} channels, ${bot.users.cache.size} users`);
});
bot.on('message', async msg => {
try {
if (msg.author.bot) {
return;
}
if (msg.cleanContent.startsWith(config.prefix)) {
let [cmd, ...args] = msg.cleanContent.split(' ');
switch (cmd.replace(config.prefix, '')) {
case 'serverWiki': {
if (msg.channel.type !== 'dm' && !msg.member.hasPermission('ADMINISTRATOR') && msg.user.id !== config.adminId) {
return;
}
if (msg.channel.type === 'dm') {
await msg.channel.send('Please use `channelWiki` to set the preferred wiki for our private conversations!');
return;
}
let wikiKey = realWikiName(args.join(' '));
if (wikiKey) {
try {
db.prepare('INSERT INTO guilds (GuildID, WikiKey) VALUES(?, ?) ON CONFLICT(GuildID) DO UPDATE SET WikiKey=excluded.WikiKey').run(msg.guild.id, wikiKey);
await msg.channel.send(`The wiki for this server has been successfully set to **${getWikiObj(wikiKey).name}**!`);
} catch(e) {
await msg.channel.send('Sorry, something went wrong. Please try again. If the issue persists, please contact invalidCards#0380 with a description of your issue.');
console.error(e);
}
} else {
await msg.channel.send(`Sorry, I did not recognise the wiki **${args.join(' ')}**. Please make sure you typed it correctly and try again. For a full list, use *${config.prefix}list*.`);
}
break;
}
case 'channelWiki': {
if (msg.channel.type !== 'dm' && !msg.member.hasPermission('ADMINISTRATOR') && msg.user.id !== config.adminId) {
return;
}
if (msg.channel.type !== 'dm') {
let serverRow = db.prepare('SELECT * FROM guilds WHERE GuildID=?').get(msg.guild.id);
if (!serverRow) {
await msg.channel.send(`Please set the default wiki for the guild first with *${config.prefix}serverWiki*.`);
}
}
if (args.join(' ') === 'default') {
if (msg.channel.type === 'dm') {
await msg.channel.send(`Sorry, you can't remove the set wiki of a private conversation. You can still change it with this command - for a full list, use *${config.prefix}list*.`);
return;
}
try {
db.prepare('DELETE FROM channels WHERE ChannelID=?').run(msg.channel.id);
await msg.channel.send('The wiki for this channel has been reset to the default for the server.');
} catch(e) {
await msg.channel.send('Sorry, something went wrong. Please try again. If the issue persists, please contact invalidCards#0380 with a description of your issue.');
console.error(e);
}
} else {
let wikiKey = realWikiName(args.join(' '));
if (wikiKey) {
try {
db.prepare('INSERT INTO channels (ChannelID, WikiKey) VALUES (?, ?) ON CONFLICT(ChannelID) DO UPDATE SET WikiKey=excluded.WikiKey').run(msg.channel.id, wikiKey);
await msg.channel.send(`The wiki for this channel has been successfully set to **${getWikiObj(wikiKey).name}**!`);
} catch(e) {
await msg.channel.send('Sorry, something went wrong. Please try again. If the issue persists, please contact invalidCards#0380 with a description of your issue.');
console.error(e);
}
} else {
await msg.channel.send(`Sorry, I did not recognise the wiki **${args.join(' ')}**. Please make sure you typed it correctly and try again. For a full list, use *${config.prefix}list*.`);
}
}
break;
}
case 'reloadWikis': {
if (msg.author.id !== config.adminId) {
return;
}
delete require.cache[require.resolve('./_wikis.json')];
wikis = require('./_wikis.json');
await msg.channel.send('Wiki JSON reloaded from file!');
break;
}
case 'list': {
for (let embedCount = 0; embedCount < Math.ceil(wikis.length / 24); embedCount++) {
let embed = new Discord.MessageEmbed().setColor('#B22222').setTitle('Available wikis').setTimestamp();
if (embedCount === 0) {
embed.setDescription(`The following is a list of available wikis and their aliases. Both the full wiki name and all aliases can be used to set a wiki using \`${config.prefix}serverWiki\` and \`${config.prefix}clientWiki\`, as well as to make a one-time lookup to another wiki other than the default of the server or channel.`);
}
for (let i = 0; i < 24; i++) {
let wikiData = wikis[24*embedCount + i];
if (!wikiData) continue;
let aliases = Array.from(wikiData.aliases);
aliases.unshift(wikiData.key);
embed.addField(wikiData.name, aliases.join(', '), true);
}
if (embedCount === Math.ceil(wikis.length / 24) - 1) {
embed.addField('Unsupported wikis', `The following wikis are not supported by WOB:
• Hard Drop runs a very old version of MediaWiki, and its API is not compatible with the inner workings of this bot.`);
}
await msg.channel.send(embed);
}
break;
}
case 'help': {
let embed = new Discord.MessageEmbed().setColor('#B22222').setTitle('WOB Help').setTimestamp();
embed.addField('Commands', `
• \`${config.prefix}serverWiki <wiki>\` - sets the server's default wiki to the given wiki
• \`${config.prefix}channelWiki <wiki>\` - overrides the server's default wiki for the current channel
• \`${config.prefix}channelWiki default\` - removes a previously set override for the current channel
• \`${config.prefix}disable none|raw|all\` - prevent the bot from parsing specific types of (or all) wiki links from your messages
• \`${config.prefix}list\` - lists all available wikis and their aliases
• \`${config.prefix}help\` - display this help message`);
embed.addField('Linking syntax', `
• \`[[search term]]\` - uses the API of the default wiki of the channel or server to find an existing page with the same name
• \`[[bp:search term]]\` - uses the API of a wiki that is not the default channel or server wiki (in this case Bulbapedia) to find an existing page with the same name (see \`${config.prefix}list\` for a full list of usable aliases)
• \`{{search term}}\` - uses the API (same as above) to find an existing template with the same name
• \`--search term--\` - creates a direct link to the search term, regardless of whether or not the page exists`);
embed.addField('Feedback and suggestions', 'If you have any ideas, or features you are missing, please contact `invalidCards#0380` with your suggestion, and I will try to add it to the bot!');
embed.addField('Code', 'The bot is fully open-source - you can look at [its GitHub repo](https://github.com/invalidCards/WikiOperatingBuddy) to see the complete inner workings!');
await msg.channel.send(embed);
break;
}
case 'disable': {
if (!['all','raw','none'].includes(args[0])) {
await msg.channel.send('Please supply one of the following values: `all`, `raw`, `none`.');
return;
}
try {
db.prepare('INSERT INTO users (UserID, DisabledLinks) VALUES (?, ?) ON CONFLICT(UserID) DO UPDATE SET DisabledLinks=excluded.DisabledLinks').run(msg.author.id, args[0]);
let returnMessage = 'further messages you send will be parsed for all types of wiki links.';
if (args[0] === 'raw') returnMessage = 'further messages you send will not be parsed for raw links.';
else if (args[0] === 'all') returnMessage = 'none of your further messages will be parsed for wiki links.';
msg.reply(returnMessage);
} catch(e) {
await msg.channel.send('Sorry, something went wrong. Please try again. If the issue persists, please contact invalidCards#0380 with a description of your issue.');
console.error(e);
}
break;
}
}
} else {
let userpref = db.prepare('SELECT * FROM users WHERE UserID=?').get(msg.author.id);
if (userpref && userpref.DisabledLinks === 'all') {
return;
}
let content = msg.cleanContent;
content = content.replace(/```.*?```/gms, '');
content = content.replace(/`.*?`/gms, '');
content = content.replace(/https?[^ ]+?/gm, '');
content = content.replace(/^> .*?$/gm, '');
content = content.replace(/\n{2,}/g, '\n');
let links = [];
if (content.search(regexLink) > -1) {
let matches = Array.from(content.matchAll(regexLink), m => m[1]);
for (let match of matches) {
if (match.startsWith('#')) continue;
links.push({type: TYPE_NORMAL, query: match});
}
}
if (content.search(regexTemp) > -1) {
let matches = Array.from(content.matchAll(regexTemp), m => m[1]);
for (let match of matches) {
if (match.startsWith('#')) continue;
links.push({type: TYPE_TEMPLATE, query: match});
}
}
if (content.search(regexRaw) > -1) {
let matches = Array.from(content.matchAll(regexRaw), m => m[1]);
for (let match of matches) {
match = match.replace(/^[^a-zA-Z]+/g, '').trim();
if (match === '') continue;
links.push({type: TYPE_RAW, query: match});
}
}
if (userpref && userpref.DisabledLinks === 'raw') {
links = links.filter(l => l.type !== TYPE_RAW);
}
if (links.length) {
let wiki = db.prepare('SELECT WikiKey FROM channels WHERE ChannelID=?').get(msg.channel.id);
if (!wiki) {
if (msg.channel.type === 'dm') {
await msg.channel.send(`Our private conversation does not have a wiki set. Please use the *${config.prefix}channelWiki* command to set it up.`);
return;
}
wiki = db.prepare('SELECT WikiKey FROM guilds WHERE GuildID=?').get(msg.guild.id);
if (!wiki) {
await msg.channel.send(`This server doesn't have a default wiki set yet. If you are an admin, use *${config.prefix}serverWiki* to set one. If you're not, go yell at one.`);
return;
} else {
wiki = wiki.WikiKey;
}
} else {
wiki = wiki.WikiKey;
}
let messageContent = '**Wiki links detected:**';
for (let linkData of links) {
if (linkData.query.includes(':')) {
let [inputAltWiki, ...actualQuery] = linkData.query.split(':');
if (!actualQuery[0]) {
actualQuery = ['Main Page'];
}
realAltWiki = realWikiName(inputAltWiki.toLowerCase());
if (realAltWiki) {
if (wiki === realAltWiki) {
actualQuery = [`${inputAltWiki}:${actualQuery.join(':')}`];
}
let wikiLink = '';
switch (linkData.type) {
case TYPE_NORMAL:
wikiLink = await fetchLink(realAltWiki, actualQuery.join(':'));
break;
case TYPE_TEMPLATE:
wikiLink = await fetchLink(realAltWiki, `Template:${actualQuery.join(':')}`);
break;
case TYPE_RAW:
wikiLink = fetchRawLink(realAltWiki, actualQuery.join(':'));
break;
}
if (wikiLink) {
messageContent += `\n<${wikiLink}>`;
}
continue;
}
}
if (!linkData.query) {
linkData.query = 'Main Page';
}
let wikiLink = '';
switch (linkData.type) {
case TYPE_NORMAL:
wikiLink = await fetchLink(wiki, linkData.query);
break;
case TYPE_TEMPLATE:
wikiLink = await fetchLink(wiki, `Template:${linkData.query}`);
break;
case TYPE_RAW:
wikiLink = fetchRawLink(wiki, linkData.query);
break;
}
if (wikiLink) {
messageContent += `\n<${wikiLink}>`;
}
}
if (messageContent.split('\n').length > 1) {
await msg.channel.send(messageContent);
}
}
}
}
catch(e) {
if(e.name == 'DiscordAPIError') {
switch(e.message) {
case 'Missing Permissions':
await msg.channel.send('Sorry, I couldn\'t respond to your message. I need the `Embed links` permission to function properly.').catch(()=>{}); //Try to send a message without an embed, if it fails to send, don't care ¯\_(ツ)_/¯
break;
case 'Invalid Form Body': //Happens, when we make a mistake and try to send too much
await msg.channel.send('Due to an internal error the message failed to send\nIf this keeps happening, please report this!').catch(()=>{});
break;
default:
await msg.channel.send(`An unexpected error occurred while trying to respond (${e.message})\nTry again later. If this keeps happening, please report this!`).catch(()=>{});
break;
}
}
else {
await msg.channel.send(`An unexpected error occurred while trying to respond (${e.name}: ${e.message})\nTry again later. If this keeps happening, please report this!`).catch(()=>{});
}
}
});
const realWikiName = (abbreviation) => {
abbreviation = abbreviation.toLowerCase();
let wiki = wikis.filter(w => w.key.toLowerCase() === abbreviation);
if (wiki.length) return wiki[0].key;
wiki = wikis.filter(w => w.name.toLowerCase() === abbreviation);
if (wiki.length) return wiki[0].key;
wiki = wikis.filter(w => w.aliases.map(a => a.toLowerCase()).includes(abbreviation));
if (wiki.length) return wiki[0].key;
return false;
};
const getWikiObj = (wikiName) => {
let wiki = wikis.filter(w => w.key === realWikiName(wikiName));
if (wiki.length) return wiki[0];
};
const getWikiBaseUrl = (wikiName) => {
let wiki = wikis.filter(w => w.key === realWikiName(wikiName));
if (wiki.length) return wiki[0].url;
};
const getWikiArticleUrl = (wikiName) => {
let wiki = wikis.filter(w => w.key === realWikiName(wikiName));
if (wiki.length) return wiki[0].articleUrl;
};
const fetchLink = async (wikiName, article) => {
article = article.replace(/ /g, '_');
let url = `${getWikiBaseUrl(wikiName)}/api.php?action=opensearch&search=${eURIC(article)}&redirects=resolve&format=json`;
let response = await needle('get', url);
if (!response.body[1].length) return await fetchLinkBackup(wikiName, article);
return response.body[3][0];
};
const fetchLinkBackup = async (wikiName, article) => {
let url = `${getWikiBaseUrl(wikiName)}/api.php?action=query&list=search&srsearch=${eURIC(article)}&srnamespace=*&format=json`;
let response = await needle('get', url);
if (response.body.query.searchinfo.totalhits === 0) return false;
return `${getWikiArticleUrl(wikiName)}/${encodeURI(response.body.query.search[0].title.replace(/ /g, '_'))}`;
};
const fetchRawLink = (wikiName, article) => {
return `${getWikiArticleUrl(wikiName)}/${encodeURI(article.replace(/ /g, '_'))}`;
};
const eURIC = (component) => {
return encodeURIComponent(component).replace(/%3A/g, ':');
};
let keys = [];
for (let obj of wikis) {
keys.push(obj.key.toLowerCase());
keys.push(...obj.aliases.map(a => a.toLowerCase()));
}
let reserved = ['default', 'none', '-', 'mediawiki', 'help', 'template', 'talk', 'user', 'project', 'file', 'category', 'forum'];
keys.push(...reserved);
var duplicates = keys.reduce((acc, el, i, arr) => {
if (arr.indexOf(el) !== i && acc.indexOf(el) < 0) acc.push(el); return acc;
}, []);
if (duplicates.length > 0) {
let msg = 'Duplicate keys/aliases detected! Please resolve these before attempting to restart the bot.';
for (let dupe of duplicates) {
msg += `\n- Duplicate "${dupe}":`;
let foundKeys = wikis.filter(w => w.key === dupe);
for (let foundKey of foundKeys) {
msg += `\n - Key of ${foundKey.name}`;
}
let foundAliases = wikis.filter(w => w.aliases.includes(dupe));
for (let foundAlias of foundAliases) {
msg += `\n - Alias of ${foundAlias.name} (${foundAlias.aliases.filter(a => a === dupe).length} time(s))`;
}
if (reserved.includes(dupe)) {
msg += '\n - Built-in reserved keyword';
}
}
console.log(msg);
process.exit(-1);
} else {
bot.login(config.token);
}