general.js 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. const {MessageEmbed} = require('discord.js');
  2. const parse_page = require('../../functions/parse_page.js');
  3. const logging = require('../../util/logging.js');
  4. const {parse_infobox, htmlToPlain, htmlToDiscord, partialURIdecode} = require('../../util/functions.js');
  5. const extract_desc = require('../../util/extract_desc.js');
  6. const {limit: {interwiki: interwikiLimit}, wikiProjects} = require('../../util/default.json');
  7. const Wiki = require('../../util/wiki.js');
  8. const fs = require('fs');
  9. var fn = {
  10. special_page: require('../../functions/special_page.js'),
  11. discussion: require('../../functions/discussion.js')
  12. };
  13. fs.readdir( './cmds/wiki', (error, files) => {
  14. if ( error ) return error;
  15. files.filter( file => ( file !== 'general.js' && file.endsWith('.js') ) ).forEach( file => {
  16. var command = require('./' + file);
  17. fn[command.name] = command.run;
  18. } );
  19. } );
  20. var minecraft = {};
  21. fs.readdir( './cmds/minecraft', (error, files) => {
  22. if ( error ) return error;
  23. files.filter( file => file.endsWith('.js') ).forEach( file => {
  24. var command = require('../minecraft/' + file);
  25. minecraft[command.name] = command.run;
  26. } );
  27. } );
  28. /**
  29. * Checks a Gamepedia wiki.
  30. * @param {import('../../util/i18n.js')} lang - The user language.
  31. * @param {import('discord.js').Message} msg - The Discord message.
  32. * @param {String} title - The page title.
  33. * @param {Wiki} wiki - The wiki for the page.
  34. * @param {String} cmd - The command at this point.
  35. * @param {import('discord.js').MessageReaction} reaction - The reaction on the message.
  36. * @param {String} [spoiler] - If the response is in a spoiler.
  37. * @param {URLSearchParams} [querystring] - The querystring for the link.
  38. * @param {String} [fragment] - The section for the link.
  39. * @param {String} [interwiki] - The fallback interwiki link.
  40. * @param {Number} [selfcall] - The amount of followed interwiki links.
  41. */
  42. function gamepedia_check_wiki(lang, msg, title, wiki, cmd, reaction, spoiler = '', querystring = new URLSearchParams(), fragment = '', interwiki = '', selfcall = 0) {
  43. var full_title = title;
  44. if ( title.includes( '#' ) ) {
  45. fragment = title.split('#').slice(1).join('#').replace( /(?:%[\dA-F]{2})+/g, partialURIdecode );
  46. title = title.split('#')[0];
  47. }
  48. if ( /\?\w+=/.test(title) ) {
  49. let querystart = title.search(/\?\w+=/);
  50. querystring = new URLSearchParams(querystring + '&' + title.substring(querystart + 1));
  51. title = title.substring(0, querystart);
  52. }
  53. title = title.replace( /(?:%[\dA-F]{2})+/g, partialURIdecode );
  54. if ( title.length > 250 ) {
  55. title = title.substring(0, 250);
  56. msg.reactEmoji('⚠️');
  57. }
  58. var invoke = title.split(' ')[0].toLowerCase();
  59. var aliasInvoke = ( lang.aliases[invoke] || invoke );
  60. var args = title.split(' ').slice(1);
  61. if ( !msg.notMinecraft && wiki.href === lang.get('minecraft.link') && ( minecraft.hasOwnProperty(aliasInvoke) || invoke.startsWith( '/' ) ) ) {
  62. logging(wiki, msg.guild?.id, 'minecraft', ( minecraft.hasOwnProperty(aliasInvoke) ? aliasInvoke : 'command' ));
  63. minecraft.WIKI = this;
  64. if ( minecraft.hasOwnProperty(aliasInvoke) ) minecraft[aliasInvoke](lang, msg, args, title, cmd, querystring, fragment, reaction, spoiler);
  65. else minecraft.SYNTAX(lang, msg, invoke.substring(1), args, title, cmd, querystring, fragment, reaction, spoiler);
  66. return;
  67. }
  68. if ( aliasInvoke === 'random' && !args.join('') && !querystring.toString() && !fragment ) {
  69. return fn.random(lang, msg, wiki, reaction, spoiler);
  70. }
  71. if ( aliasInvoke === 'overview' && !args.join('') && !querystring.toString() && !fragment ) {
  72. return fn.overview(lang, msg, wiki, reaction, spoiler);
  73. }
  74. if ( aliasInvoke === 'test' && !args.join('') && !querystring.toString() && !fragment ) {
  75. this.test(lang, msg, args, '', wiki);
  76. if ( reaction ) reaction.removeEmoji();
  77. return;
  78. }
  79. if ( aliasInvoke === 'page' ) {
  80. msg.sendChannel( spoiler + '<' + wiki.toLink(args.join('_'), querystring, fragment) + '>' + spoiler );
  81. if ( reaction ) reaction.removeEmoji();
  82. return;
  83. }
  84. if ( aliasInvoke === 'diff' && !wiki.isFandom(false) && args.join('') && !querystring.toString() && !fragment ) {
  85. return fn.diff(lang, msg, args, wiki, reaction, spoiler);
  86. }
  87. var noRedirect = ( querystring.getAll('redirect').pop() === 'no' || ( querystring.has('action') && querystring.getAll('action').pop() !== 'view' ) );
  88. got.get( wiki + 'api.php?action=query&meta=siteinfo&siprop=general|namespaces|specialpagealiases&iwurl=true' + ( noRedirect ? '' : '&redirects=true' ) + '&prop=categoryinfo|info|pageprops|pageimages|extracts&piprop=original|name&ppprop=description|displaytitle|page_image_free|infoboxes&explaintext=true&exsectionformat=raw&exlimit=1&converttitles=true&titles=%1F' + encodeURIComponent( ( aliasInvoke === 'search' ? full_title.split(' ').slice(1).join(' ') : title ).replace( /\x1F/g, '\ufffd' ) ) + '&format=json' ).then( response => {
  89. var body = response.body;
  90. if ( body && body.warnings ) log_warn(body.warnings);
  91. if ( response.statusCode !== 200 || !body || body.batchcomplete === undefined || !body.query ) {
  92. if ( body?.query?.general?.generator === 'MediaWiki 1.19.24' && wiki.isFandom(false) ) {
  93. return this.fandom(lang, msg, title, wiki, cmd, reaction, spoiler, querystring, fragment, selfcall);
  94. }
  95. else if ( interwiki ) msg.sendChannel( spoiler + ' ' + interwiki + ' ' + spoiler );
  96. else if ( wiki.noWiki(response.url, response.statusCode) ) {
  97. console.log( '- This wiki doesn\'t exist!' );
  98. msg.reactEmoji('nowiki');
  99. }
  100. else {
  101. console.log( '- ' + response.statusCode + ': Error while getting the search results: ' + ( body && body.error && body.error.info ) );
  102. msg.sendChannelError( spoiler + '<' + wiki.toLink( ( querystring.toString() || fragment || !title ? title : 'Special:Search' ), ( querystring.toString() || fragment || !title ? querystring : {search:title} ), fragment) + '>' + spoiler );
  103. }
  104. if ( reaction ) reaction.removeEmoji();
  105. return;
  106. }
  107. wiki.updateWiki(body.query.general);
  108. if ( aliasInvoke === 'search' ) {
  109. logging(wiki, msg.guild?.id, 'search');
  110. return fn.search(lang, msg, full_title.split(' ').slice(1).join(' '), wiki, body.query, reaction, spoiler);
  111. }
  112. if ( aliasInvoke === 'diff' && args.join('') && !querystring.toString() && !fragment ) {
  113. return fn.diff(lang, msg, args, wiki, reaction, spoiler);
  114. }
  115. if ( aliasInvoke === 'discussion' && wiki.isFandom(false) && !querystring.toString() && !fragment ) {
  116. logging(wiki, msg.guild?.id, 'discussion');
  117. return fn.discussion(lang, msg, wiki, args.join(' '), body.query.general.sitename, reaction, spoiler);
  118. }
  119. if ( body.query.pages ) {
  120. var querypages = Object.values(body.query.pages);
  121. var querypage = querypages[0];
  122. if ( body.query.redirects && body.query.redirects[0].from.split(':')[0] === body.query.namespaces['-1']['*'] && body.query.specialpagealiases.filter( sp => ['Mypage','Mytalk','MyLanguage'].includes( sp.realname ) ).map( sp => sp.aliases[0] ).includes( body.query.redirects[0].from.split(':').slice(1).join(':').split('/')[0].replace( / /g, '_' ) ) ) {
  123. querypage.title = body.query.redirects[0].from;
  124. delete body.query.redirects[0].tofragment;
  125. delete querypage.missing;
  126. querypage.ns = -1;
  127. querypage.special = '';
  128. }
  129. var contribs = body.query.namespaces['-1']['*'] + ':' + body.query.specialpagealiases.find( sp => sp.realname === 'Contributions' ).aliases[0] + '/';
  130. if ( ( querypage.ns === 2 || querypage.ns === 202 || querypage.ns === 1200 ) && ( !querypage.title.includes( '/' ) || /^[^:]+:(?:(?:\d{1,3}\.){3}\d{1,3}\/\d{2}|(?:[\dA-F]{1,4}:){7}[\dA-F]{1,4}\/\d{2,3})$/.test(querypage.title) ) ) {
  131. var userparts = querypage.title.split(':');
  132. querypage.noRedirect = noRedirect;
  133. return fn.user(lang, msg, userparts[0] + ':', userparts.slice(1).join(':'), wiki, querystring, fragment, querypage, contribs, reaction, spoiler);
  134. }
  135. if ( querypage.ns === -1 && querypage.title.startsWith( contribs ) && querypage.title.length > contribs.length ) {
  136. var username = querypage.title.split('/').slice(1).join('/');
  137. return got.get( wiki + 'api.php?action=query&titles=User:' + encodeURIComponent( username ) + '&format=json' ).then( uresponse => {
  138. var ubody = uresponse.body;
  139. if ( uresponse.statusCode !== 200 || !ubody || ubody.batchcomplete === undefined || !ubody.query ) {
  140. console.log( '- ' + uresponse.statusCode + ': Error while getting the user: ' + ( ubody && ubody.error && ubody.error.info ) );
  141. msg.sendChannelError( spoiler + '<' + wiki.toLink(contribs + username, querystring, fragment) + '>' + spoiler );
  142. if ( reaction ) reaction.removeEmoji();
  143. }
  144. else {
  145. querypage = Object.values(ubody.query.pages)[0];
  146. if ( querypage.ns === 2 ) {
  147. username = querypage.title.split(':').slice(1).join(':');
  148. querypage.title = contribs + username;
  149. delete querypage.missing;
  150. querypage.ns = -1;
  151. querypage.special = '';
  152. querypage.noRedirect = noRedirect;
  153. fn.user(lang, msg, contribs, username, wiki, querystring, fragment, querypage, contribs, reaction, spoiler);
  154. }
  155. else {
  156. msg.reactEmoji('error');
  157. if ( reaction ) reaction.removeEmoji();
  158. }
  159. }
  160. }, error => {
  161. console.log( '- Error while getting the user: ' + error );
  162. msg.sendChannelError( spoiler + '<' + wiki.toLink(contribs + username, querystring, fragment) + '>' + spoiler );
  163. if ( reaction ) reaction.removeEmoji();
  164. } );
  165. }
  166. if ( wiki.isMiraheze() && querypage.ns === 0 && /^Mh:[a-z\d]+:/.test(querypage.title) ) {
  167. logging(wiki, msg.guild?.id, 'interwiki', 'miraheze');
  168. var iw_parts = querypage.title.split(':');
  169. var iw = new Wiki('https://' + iw_parts[1] + '.miraheze.org/w/');
  170. var iw_link = iw.toLink(iw_parts.slice(2).join(':'), querystring, fragment);
  171. var maxselfcall = interwikiLimit[( patreons[msg.guild?.id] ? 'patreon' : 'default' )];
  172. if ( selfcall < maxselfcall ) {
  173. selfcall++;
  174. return this.general(lang, msg, iw_parts.slice(2).join(':'), iw, '!!' + iw.hostname + ' ', reaction, spoiler, querystring, fragment, iw_link, selfcall);
  175. }
  176. msg.sendChannel( spoiler + ' ' + iw_link + ' ' + spoiler ).then( message => {
  177. if ( message && selfcall === maxselfcall ) message.reactEmoji('⚠️');
  178. } );
  179. if ( reaction ) reaction.removeEmoji();
  180. return;
  181. }
  182. if ( ( querypage.missing !== undefined && querypage.known === undefined && !( noRedirect || querypage.categoryinfo ) ) || querypage.invalid !== undefined ) return got.get( wiki + 'api.php?action=query&prop=categoryinfo|info|pageprops|pageimages|extracts&piprop=original|name&ppprop=description|displaytitle|page_image_free|infoboxes&explaintext=true&exsectionformat=raw&exlimit=1&generator=search&gsrnamespace=4|12|14|' + querypage.ns + '|' + Object.values(body.query.namespaces).filter( ns => ns.content !== undefined ).map( ns => ns.id ).join('|') + '&gsrlimit=1&gsrsearch=' + encodeURIComponent( title ) + '&format=json' ).then( srresponse => {
  183. var srbody = srresponse.body;
  184. if ( srbody && srbody.warnings ) log_warn(srbody.warnings);
  185. if ( srresponse.statusCode !== 200 || !srbody || srbody.batchcomplete === undefined ) {
  186. console.log( '- ' + srresponse.statusCode + ': Error while getting the search results: ' + ( srbody && srbody.error && srbody.error.info ) );
  187. msg.sendChannelError( spoiler + '<' + wiki.toLink('Special:Search', {search:title}) + '>' + spoiler );
  188. return;
  189. }
  190. if ( !srbody.query ) {
  191. msg.reactEmoji('🤷');
  192. }
  193. else {
  194. querypage = Object.values(srbody.query.pages)[0];
  195. var pagelink = wiki.toLink(querypage.title, querystring, fragment);
  196. var text = '';
  197. var embed = new MessageEmbed().setAuthor( body.query.general.sitename ).setTitle( querypage.title.escapeFormatting() ).setURL( pagelink );
  198. if ( querypage.pageprops && querypage.pageprops.displaytitle ) {
  199. var displaytitle = htmlToDiscord( querypage.pageprops.displaytitle );
  200. if ( displaytitle.length > 250 ) displaytitle = displaytitle.substring(0, 250) + '\u2026';
  201. embed.setTitle( displaytitle );
  202. }
  203. if ( querypage.extract ) {
  204. var extract = extract_desc(querypage.extract, fragment);
  205. embed.backupDescription = extract[0];
  206. if ( extract[1].length && extract[2].length ) {
  207. embed.backupField = {name: extract[1], value: extract[2]};
  208. }
  209. }
  210. if ( querypage.pageprops && querypage.pageprops.description ) {
  211. var description = htmlToPlain( querypage.pageprops.description );
  212. if ( description.length > 1000 ) description = description.substring(0, 1000) + '\u2026';
  213. embed.backupDescription = description;
  214. }
  215. if ( querypage.ns === 6 ) {
  216. var pageimage = ( querypage?.original?.source || wiki.toLink('Special:FilePath/' + querypage.title, {version:Date.now()}) );
  217. if ( msg.showEmbed() && /\.(?:png|jpg|jpeg|gif)$/.test(querypage.title.toLowerCase()) ) embed.setImage( pageimage );
  218. else if ( msg.uploadFiles() ) embed.attachFiles( [{attachment:pageimage,name:( spoiler ? 'SPOILER ' : '' ) + querypage.title}] );
  219. }
  220. else if ( querypage.title === body.query.general.mainpage ) {
  221. embed.setThumbnail( new URL(body.query.general.logo, wiki).href );
  222. }
  223. else if ( querypage.pageimage && querypage.original ) {
  224. embed.setThumbnail( querypage.original.source );
  225. }
  226. else if ( querypage.pageprops && querypage.pageprops.page_image_free ) {
  227. embed.setThumbnail( wiki.toLink('Special:FilePath/' + querypage.pageprops.page_image_free, {version:Date.now()}) );
  228. }
  229. else embed.setThumbnail( new URL(body.query.general.logo, wiki).href );
  230. var prefix = ( msg.channel.isGuild() && patreons[msg.guild.id] || process.env.prefix );
  231. var linksuffix = ( querystring.toString() ? '?' + querystring : '' ) + ( fragment ? '#' + fragment : '' );
  232. if ( title.replace( /[_-]/g, ' ' ).toLowerCase() === querypage.title.replace( /-/g, ' ' ).toLowerCase() ) {
  233. text = '';
  234. }
  235. else if ( !srbody.continue ) {
  236. text = '\n' + lang.get('search.infopage', '`' + prefix + cmd + ( lang.localNames.page || 'page' ) + ' ' + title + linksuffix + '`');
  237. }
  238. else {
  239. text = '\n' + lang.get('search.infosearch', '`' + prefix + cmd + ( lang.localNames.page || 'page' ) + ' ' + title + linksuffix + '`', '`' + prefix + cmd + ( lang.localNames.search || 'search' ) + ' ' + title + linksuffix + '`');
  240. }
  241. if ( querypage.categoryinfo ) {
  242. var category = [lang.get('search.category.content')];
  243. if ( querypage.categoryinfo.size === 0 ) {
  244. category.push(lang.get('search.category.empty'));
  245. }
  246. if ( querypage.categoryinfo.pages > 0 ) {
  247. category.push(lang.get('search.category.pages', querypage.categoryinfo.pages.toLocaleString(lang.get('dateformat')), querypage.categoryinfo.pages));
  248. }
  249. if ( querypage.categoryinfo.files > 0 ) {
  250. category.push(lang.get('search.category.files', querypage.categoryinfo.files.toLocaleString(lang.get('dateformat')), querypage.categoryinfo.files));
  251. }
  252. if ( querypage.categoryinfo.subcats > 0 ) {
  253. category.push(lang.get('search.category.subcats', querypage.categoryinfo.subcats.toLocaleString(lang.get('dateformat')), querypage.categoryinfo.subcats));
  254. }
  255. if ( msg.showEmbed() ) embed.addField( category[0], category.slice(1).join('\n') );
  256. else text += '\n\n' + category.join('\n');
  257. }
  258. if ( !fragment && !embed.fields.length && querypage.pageprops && querypage.pageprops.infoboxes ) {
  259. try {
  260. var infobox = JSON.parse(querypage.pageprops.infoboxes)?.[0];
  261. parse_infobox(infobox, embed, new URL(body.query.general.logo, wiki).href, wiki.articleURL.href);
  262. }
  263. catch ( error ) {
  264. console.log( '- Failed to parse the infobox: ' + error );
  265. }
  266. }
  267. msg.sendChannel( spoiler + '<' + pagelink + '>' + text + spoiler, {embed} ).then( message => parse_page(message, querypage, embed, wiki, ( querypage.title === body.query.general.mainpage ? '' : new URL(body.query.general.logo, wiki).href ), fragment) );
  268. }
  269. }, error => {
  270. console.log( '- Error while getting the search results: ' + error );
  271. msg.sendChannelError( spoiler + '<' + wiki.toLink('Special:Search', {search:title}) + '>' + spoiler );
  272. } ).finally( () => {
  273. logging(wiki, msg.guild?.id, 'general', 'search');
  274. if ( reaction ) reaction.removeEmoji();
  275. } );
  276. if ( querypage.ns === -1 ) {
  277. var pagelink = wiki.toLink(querypage.title, querystring, fragment);
  278. var embed = new MessageEmbed().setAuthor( body.query.general.sitename ).setTitle( querypage.title.escapeFormatting() ).setURL( pagelink ).setThumbnail( new URL(body.query.general.logo, wiki).href );
  279. var specialpage = body.query.specialpagealiases.find( sp => body.query.namespaces['-1']['*'] + ':' + sp.aliases[0].replace( /\_/g, ' ' ) === querypage.title.split('/')[0] );
  280. specialpage = ( specialpage ? specialpage.realname : querypage.title.replace( body.query.namespaces['-1']['*'] + ':', '' ).split('/')[0] ).toLowerCase();
  281. return fn.special_page(lang, msg, querypage.title, specialpage, embed, wiki, reaction, spoiler);
  282. }
  283. if ( querypage.ns === -2 ) {
  284. logging(wiki, msg.guild?.id, 'general', 'media');
  285. var filepath = body.query.specialpagealiases.find( sp => sp.realname === 'Filepath' );
  286. var pagelink = wiki.toLink(body.query.namespaces['-1']['*'] + ':' + ( filepath?.aliases?.[0] || 'FilePath' ) + querypage.title.replace( body.query.namespaces['-2']['*'] + ':', '/' ), querystring, fragment);
  287. var embed = new MessageEmbed().setAuthor( body.query.general.sitename ).setTitle( querypage.title.escapeFormatting() ).setURL( pagelink ).setDescription( '[' + lang.get('search.media') + '](' + wiki.toLink(querypage.title, '', '', true) + ')' );
  288. if ( msg.showEmbed() && /\.(?:png|jpg|jpeg|gif)$/.test(querypage.title.toLowerCase()) ) embed.setImage( pagelink );
  289. else if ( msg.uploadFiles() ) embed.attachFiles( [{attachment:pagelink,name:( spoiler ? 'SPOILER ' : '' ) + querypage.title}] );
  290. msg.sendChannel( spoiler + '<' + pagelink + '>' + spoiler, {embed} );
  291. if ( reaction ) reaction.removeEmoji();
  292. return;
  293. }
  294. logging(wiki, msg.guild?.id, 'general');
  295. var pagelink = wiki.toLink(querypage.title, querystring, ( fragment || ( body.query.redirects && body.query.redirects[0].tofragment ) || '' ));
  296. var text = '';
  297. var embed = new MessageEmbed().setAuthor( body.query.general.sitename ).setTitle( querypage.title.escapeFormatting() ).setURL( pagelink );
  298. if ( querypage.pageprops && querypage.pageprops.displaytitle ) {
  299. var displaytitle = htmlToDiscord( querypage.pageprops.displaytitle );
  300. if ( displaytitle.length > 250 ) displaytitle = displaytitle.substring(0, 250) + '\u2026';
  301. embed.setTitle( displaytitle );
  302. }
  303. if ( querypage.extract ) {
  304. var extract = extract_desc(querypage.extract, ( fragment || ( body.query.redirects && body.query.redirects[0].tofragment ) || '' ));
  305. embed.backupDescription = extract[0];
  306. if ( extract[1].length && extract[2].length ) {
  307. embed.backupField = {name: extract[1], value: extract[2]};
  308. }
  309. }
  310. if ( querypage.pageprops && querypage.pageprops.description ) {
  311. var description = htmlToPlain( querypage.pageprops.description );
  312. if ( description.length > 1000 ) description = description.substring(0, 1000) + '\u2026';
  313. embed.backupDescription = description;
  314. }
  315. if ( querypage.ns === 6 ) {
  316. var pageimage = ( querypage?.original?.source || wiki.toLink('Special:FilePath/' + querypage.title, {version:Date.now()}) );
  317. if ( msg.showEmbed() && /\.(?:png|jpg|jpeg|gif)$/.test(querypage.title.toLowerCase()) ) embed.setImage( pageimage );
  318. else if ( msg.uploadFiles() ) embed.attachFiles( [{attachment:pageimage,name:( spoiler ? 'SPOILER ' : '' ) + querypage.title}] );
  319. }
  320. else if ( querypage.title === body.query.general.mainpage ) {
  321. embed.setThumbnail( new URL(body.query.general.logo, wiki).href );
  322. }
  323. else if ( querypage.pageimage && querypage.original ) {
  324. embed.setThumbnail( querypage.original.source );
  325. }
  326. else if ( querypage.pageprops && querypage.pageprops.page_image_free ) {
  327. embed.setThumbnail( wiki.toLink('Special:FilePath/' + querypage.pageprops.page_image_free, {version:Date.now()}) );
  328. }
  329. else embed.setThumbnail( new URL(body.query.general.logo, wiki).href );
  330. if ( querypage.categoryinfo ) {
  331. var category = [lang.get('search.category.content')];
  332. if ( querypage.categoryinfo.size === 0 ) {
  333. category.push(lang.get('search.category.empty'));
  334. }
  335. if ( querypage.categoryinfo.pages > 0 ) {
  336. category.push(lang.get('search.category.pages', querypage.categoryinfo.pages.toLocaleString(lang.get('dateformat')), querypage.categoryinfo.pages));
  337. }
  338. if ( querypage.categoryinfo.files > 0 ) {
  339. category.push(lang.get('search.category.files', querypage.categoryinfo.files.toLocaleString(lang.get('dateformat')), querypage.categoryinfo.files));
  340. }
  341. if ( querypage.categoryinfo.subcats > 0 ) {
  342. category.push(lang.get('search.category.subcats', querypage.categoryinfo.subcats.toLocaleString(lang.get('dateformat')), querypage.categoryinfo.subcats));
  343. }
  344. if ( msg.showEmbed() ) embed.addField( category[0], category.slice(1).join('\n') );
  345. else text += '\n\n' + category.join('\n');
  346. }
  347. if ( !( fragment || ( body.query.redirects && body.query.redirects[0].tofragment ) || '' ) && !embed.fields.length && querypage.pageprops && querypage.pageprops.infoboxes ) {
  348. try {
  349. var infobox = JSON.parse(querypage.pageprops.infoboxes)?.[0];
  350. parse_infobox(infobox, embed, new URL(body.query.general.logo, wiki).href, wiki.articleURL.href);
  351. }
  352. catch ( error ) {
  353. console.log( '- Failed to parse the infobox: ' + error );
  354. }
  355. }
  356. msg.sendChannel( spoiler + '<' + pagelink + '>' + text + spoiler, {embed} ).then( message => parse_page(message, querypage, embed, wiki, ( querypage.title === body.query.general.mainpage ? '' : new URL(body.query.general.logo, wiki).href ), ( fragment || ( body.query.redirects && body.query.redirects[0].tofragment ) || '' )) );
  357. if ( reaction ) reaction.removeEmoji();
  358. return;
  359. }
  360. if ( body.query.interwiki ) {
  361. if ( msg.channel.isGuild() && pause[msg.guild.id] ) {
  362. if ( reaction ) reaction.removeEmoji();
  363. console.log( '- Aborted, paused.' );
  364. return;
  365. }
  366. logging(wiki, msg.guild?.id, 'interwiki');
  367. var iw = new URL(body.query.interwiki[0].url.replace( /\\/g, '%5C' ).replace( /@(here|everyone)/g, '%40$1' ), wiki);
  368. querystring.forEach( (value, name) => {
  369. iw.searchParams.append(name, value);
  370. } );
  371. if ( fragment ) iw.hash = Wiki.toSection(fragment);
  372. else fragment = iw.hash.substring(1);
  373. var maxselfcall = interwikiLimit[( patreons[msg.guild?.id] ? 'patreon' : 'default' )];
  374. if ( selfcall < maxselfcall && ['http:','https:'].includes( iw.protocol ) ) {
  375. selfcall++;
  376. if ( iw.hostname.endsWith( '.gamepedia.com' ) ) {
  377. let iwtitle = decodeURIComponent( iw.pathname.substring(1) ).replace( /_/g, ' ' );
  378. cmd = '!' + iw.hostname.replace( '.gamepedia.com', ' ' );
  379. if ( cmd !== '!www ' ) return this.general(lang, msg, iwtitle, new Wiki(iw.origin), cmd, reaction, spoiler, iw.searchParams, fragment, iw.href, selfcall);
  380. }
  381. if ( iw.hostname.endsWith( '.fandom.com' ) || iw.hostname.endsWith( '.wikia.org' ) ) {
  382. let regex = iw.pathname.match( /^(\/(?!wiki\/)[a-z-]{2,12})?(?:\/wiki\/|\/?$)/ );
  383. if ( regex ) {
  384. let path = ( regex[1] || '' );
  385. let iwtitle = decodeURIComponent( iw.pathname.replace( regex[0], '' ) ).replace( /_/g, ' ' );
  386. cmd = ( iw.hostname.endsWith( '.wikia.org' ) ? '??' : '?' ) + ( path ? path.substring(1) + '.' : '' ) + iw.hostname.replace( /\.(?:fandom\.com|wikia\.org)/, ' ' );
  387. return this.general(lang, msg, iwtitle, new Wiki(iw.origin + path + '/'), cmd, reaction, spoiler, iw.searchParams, fragment, iw.href, selfcall);
  388. }
  389. }
  390. let project = wikiProjects.find( project => iw.hostname.endsWith( project.name ) );
  391. if ( project ) {
  392. let regex = ( iw.host + iw.pathname ).match( new RegExp( '^' + project.regex + '(?:' + project.articlePath + '|/?$)' ) );
  393. if ( regex ) {
  394. let iwtitle = decodeURIComponent( ( iw.host + iw.pathname ).replace( regex[0], '' ) ).replace( /_/g, ' ' );
  395. cmd = '!!' + regex[1] + ' ';
  396. return this.general(lang, msg, iwtitle, new Wiki('https://' + regex[1] + project.scriptPath), cmd, reaction, spoiler, iw.searchParams, fragment, iw.href, selfcall);
  397. }
  398. }
  399. }
  400. msg.sendChannel( spoiler + ' ' + iw + ' ' + spoiler ).then( message => {
  401. if ( message && selfcall === maxselfcall ) message.reactEmoji('⚠️');
  402. } );
  403. if ( reaction ) reaction.removeEmoji();
  404. return;
  405. }
  406. logging(wiki, msg.guild?.id, 'general');
  407. var querypage = {title: body.query.general.mainpage, contentmodel: 'wikitext'};
  408. var pagelink = wiki.toLink(querypage.title, querystring, fragment);
  409. var embed = new MessageEmbed().setAuthor( body.query.general.sitename ).setTitle( querypage.title.escapeFormatting() ).setURL( pagelink ).setThumbnail( new URL(body.query.general.logo, wiki).href );
  410. got.get( wiki + 'api.php?action=query' + ( noRedirect ? '' : '&redirects=true' ) + '&prop=info|pageprops|extracts&ppprop=description|displaytitle|infoboxes&explaintext=true&exsectionformat=raw&exlimit=1&titles=' + encodeURIComponent( querypage.title ) + '&format=json' ).then( mpresponse => {
  411. var mpbody = mpresponse.body;
  412. if ( mpbody && mpbody.warnings ) log_warn(body.warnings);
  413. if ( mpresponse.statusCode !== 200 || !mpbody || mpbody.batchcomplete === undefined || !mpbody.query ) {
  414. console.log( '- ' + mpresponse.statusCode + ': Error while getting the main page: ' + ( mpbody && mpbody.error && mpbody.error.info ) );
  415. return;
  416. }
  417. querypage = Object.values(mpbody.query.pages)[0];
  418. if ( querypage.pageprops && querypage.pageprops.displaytitle ) {
  419. var displaytitle = htmlToDiscord( querypage.pageprops.displaytitle );
  420. if ( displaytitle.length > 250 ) displaytitle = displaytitle.substring(0, 250) + '\u2026';
  421. embed.setTitle( displaytitle );
  422. }
  423. if ( querypage.extract ) {
  424. var extract = extract_desc(querypage.extract, fragment);
  425. embed.backupDescription = extract[0];
  426. if ( extract[1].length && extract[2].length ) {
  427. embed.backupField = {name: extract[1], value: extract[2]};
  428. }
  429. }
  430. if ( querypage.pageprops && querypage.pageprops.description ) {
  431. var description = htmlToPlain( querypage.pageprops.description );
  432. if ( description.length > 1000 ) description = description.substring(0, 1000) + '\u2026';
  433. embed.backupDescription = description;
  434. }
  435. if ( !fragment && !embed.fields.length && querypage.pageprops && querypage.pageprops.infoboxes ) {
  436. try {
  437. var infobox = JSON.parse(querypage.pageprops.infoboxes)?.[0];
  438. parse_infobox(infobox, embed, '', wiki.articleURL.href);
  439. }
  440. catch ( error ) {
  441. console.log( '- Failed to parse the infobox: ' + error );
  442. }
  443. }
  444. }, error => {
  445. console.log( '- Error while getting the main page: ' + error );
  446. } ).finally( () => {
  447. msg.sendChannel( spoiler + '<' + pagelink + '>' + spoiler, {embed} ).then( message => parse_page(message, querypage, embed, wiki, '', fragment) );
  448. if ( reaction ) reaction.removeEmoji();
  449. } );
  450. }, error => {
  451. if ( interwiki ) msg.sendChannel( spoiler + ' ' + interwiki + ' ' + spoiler );
  452. else if ( wiki.noWiki(error.message) ) {
  453. console.log( '- This wiki doesn\'t exist!' );
  454. msg.reactEmoji('nowiki');
  455. }
  456. else {
  457. console.log( '- Error while getting the search results: ' + error );
  458. msg.sendChannelError( spoiler + '<' + wiki.toLink( ( querystring.toString() || fragment || !title ? title : 'Special:Search' ), ( querystring.toString() || fragment || !title ? querystring : {search:title} ), fragment) + '>' + spoiler );
  459. }
  460. if ( reaction ) reaction.removeEmoji();
  461. } );
  462. }
  463. module.exports = gamepedia_check_wiki;