general.js 30 KB

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