overview.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. const {MessageEmbed} = require('discord.js');
  2. const fandom_overview = require('./fandom/overview.js').run;
  3. const {timeoptions} = require('../../util/default.json');
  4. const {toFormatting, toPlaintext} = require('../../util/functions.js');
  5. var allSites = [];
  6. const getAllSites = require('../../util/allSites.js');
  7. getAllSites.then( sites => allSites = sites );
  8. /**
  9. * Sends a Gamepedia wiki overview.
  10. * @param {import('../../util/i18n.js')} lang - The user language.
  11. * @param {import('discord.js').Message} msg - The Discord message.
  12. * @param {import('../../util/wiki.js')} wiki - The wiki for the overview.
  13. * @param {import('discord.js').MessageReaction} reaction - The reaction on the message.
  14. * @param {String} spoiler - If the response is in a spoiler.
  15. */
  16. function gamepedia_overview(lang, msg, wiki, reaction, spoiler) {
  17. if ( !allSites.length ) getAllSites.update();
  18. got.get( wiki + 'api.php?action=query&meta=allmessages|siteinfo&ammessages=custom-Wiki_Manager|custom-GamepediaNotice|custom-FandomMergeNotice&amenableparser=true&siprop=general|statistics|languages|rightsinfo&siinlanguagecode=' + lang.lang + '&list=logevents&ledir=newer&lelimit=1&leprop=timestamp&titles=Special:Statistics&format=json' ).then( response => {
  19. var body = response.body;
  20. if ( body && body.warnings ) log_warn(body.warnings);
  21. if ( response.statusCode !== 200 || !body || body.batchcomplete === undefined || !body.query || !body.query.pages ) {
  22. if ( wiki.noWiki(response.url, response.statusCode) ) {
  23. console.log( '- This wiki doesn\'t exist!' );
  24. msg.reactEmoji('nowiki');
  25. }
  26. else if ( body?.query?.general?.generator === 'MediaWiki 1.19.24' && wiki.isFandom(false) ) {
  27. return fandom_overview(lang, msg, wiki, reaction, spoiler);
  28. }
  29. else {
  30. console.log( '- ' + response.statusCode + ': Error while getting the statistics: ' + ( body && body.error && body.error.info ) );
  31. msg.sendChannelError( spoiler + '<' + wiki.toLink('Special:Statistics') + '>' + spoiler );
  32. }
  33. if ( reaction ) reaction.removeEmoji();
  34. }
  35. else {
  36. wiki.updateWiki(body.query.general);
  37. var site = null;
  38. if ( allSites.some( site => site.wiki_domain === wiki.hostname ) ) {
  39. site = allSites.find( site => site.wiki_domain === wiki.hostname );
  40. var name = [lang.get('overview.name'), site.wiki_display_name];
  41. var manager = [lang.get('overview.manager'), site.wiki_managers];
  42. var official = [lang.get('overview.official'), lang.get('overview.' + ( site.official_wiki ? 'yes' : 'no' ))];
  43. var crossover = [lang.get('overview.crossover'), ( site.wiki_crossover ? '<https://' + site.wiki_crossover + '/>' : '' )];
  44. var description = [lang.get('overview.description'), site.wiki_description];
  45. var image = [lang.get('overview.image'), site.wiki_image];
  46. if ( description[1] ) {
  47. description[1] = description[1].escapeFormatting();
  48. if ( description[1].length > 1000 ) description[1] = description[1].substring(0, 1000) + '\u2026';
  49. }
  50. if ( image[1] && image[1].startsWith( '/' ) ) image[1] = new URL(image[1], wiki).href;
  51. }
  52. var version = [lang.get('overview.version'), body.query.general.generator];
  53. var creation_date = null;
  54. var created = [lang.get('overview.created'), lang.get('overview.unknown')];
  55. if ( body.query.logevents?.[0]?.timestamp ) {
  56. creation_date = new Date(body.query.logevents[0].timestamp);
  57. created[1] = creation_date.toLocaleString(lang.get('dateformat'), timeoptions);
  58. }
  59. var language = [lang.get('overview.lang'), body.query.languages.find( language => {
  60. return language.code === body.query.general.lang;
  61. } )['*']];
  62. var rtl = [lang.get('overview.rtl'), ( body.query.general.rtl !== undefined ? lang.get('overview.yes') : undefined )];
  63. var articles = [lang.get('overview.articles'), body.query.statistics.articles];
  64. var pages = [lang.get('overview.pages'), body.query.statistics.pages];
  65. var edits = [lang.get('overview.edits'), body.query.statistics.edits];
  66. var users = [lang.get('overview.users'), body.query.statistics.activeusers];
  67. var license = [lang.get('overview.license'), lang.get('overview.unknown')];
  68. if ( body.query.rightsinfo.url ) {
  69. let licenseurl = body.query.rightsinfo.url
  70. if ( /^(?:https?:\/)?\//.test(licenseurl) ) licenseurl = new URL(licenseurl, wiki).href;
  71. else licenseurl = wiki.toLink(licenseurl, '', '', true);
  72. if ( body.query.rightsinfo.text ) {
  73. let licensetext = body.query.rightsinfo.text;
  74. if ( msg.showEmbed() ) {
  75. license[1] = '[' + toPlaintext(licensetext, true) + '](' + licenseurl + ')';
  76. }
  77. else license[1] = toPlaintext(licensetext, true) + ' (<' + licenseurl + '>)';
  78. }
  79. else license[1] = '<' + licenseurl + '>';
  80. }
  81. else if ( body.query.rightsinfo.text ) {
  82. license[1] = toFormatting(body.query.rightsinfo.text, msg.showEmbed(), wiki, '', true);
  83. }
  84. var misermode = [lang.get('overview.misermode'), lang.get('overview.' + ( body.query.general.misermode !== undefined ? 'yes' : 'no' ))];
  85. var readonly = [lang.get('overview.readonly')];
  86. if ( body.query.general.readonly !== undefined ) {
  87. if ( body.query.general.readonlyreason ) {
  88. let readonlyreason = body.query.general.readonlyreason;
  89. readonly.push(toFormatting(readonlyreason, msg.showEmbed(), wiki, '', true));
  90. }
  91. else readonly = ['\u200b', '**' + lang.get('overview.readonly') + '**'];
  92. }
  93. var title = body.query.pages['-1'].title;
  94. var pagelink = wiki.toLink(title);
  95. if ( msg.showEmbed() ) {
  96. var text = '<' + pagelink + '>';
  97. var embed = new MessageEmbed().setAuthor( body.query.general.sitename ).setTitle( title.escapeFormatting() ).setURL( pagelink ).setThumbnail( new URL(body.query.general.logo, wiki).href );
  98. }
  99. else {
  100. var embed = {};
  101. var text = '<' + pagelink + '>\n\n';
  102. }
  103. if ( wiki.isFandom(false) ) got.get( 'https://community.fandom.com/api/v1/Wikis/ByString?expand=true&includeDomain=true&limit=10&string=' + body.query.general.servername + body.query.general.scriptpath + '&format=json&cache=' + Date.now() ).then( ovresponse => {
  104. var manager = [lang.get('overview.manager'), body.query.allmessages[0]['*']];
  105. var crossover = [lang.get('overview.crossover')];
  106. if ( body.query.allmessages[1]['*'] ) {
  107. crossover.push('<https://' + body.query.allmessages[1]['*'] + '.gamepedia.com/>');
  108. }
  109. else if ( body.query.allmessages[2]['*'] ) {
  110. let merge = body.query.allmessages[2]['*'].split('/');
  111. crossover.push('<https://' + merge[0] + '.fandom.com/' + ( merge[1] ? merge[1] + '/' : '' ) + '>');
  112. }
  113. var ovbody = ovresponse.body;
  114. if ( ovresponse.statusCode !== 200 || !ovbody || ovbody.exception || !ovbody.items || !ovbody.items.length ) {
  115. console.log( '- ' + ovresponse.statusCode + ': Error while getting the wiki details: ' + ( ovbody && ovbody.exception && ovbody.exception.details ) );
  116. if ( msg.showEmbed() ) {
  117. embed.addField( version[0], version[1], true ).addField( created[0], created[1], true ).addField( language[0], language[1], true );
  118. if ( rtl[1] ) embed.addField( rtl[0], rtl[1], true );
  119. embed.addField( articles[0], articles[1], true ).addField( pages[0], pages[1], true ).addField( edits[0], edits[1], true ).addField( users[0], users[1], true ).addField( license[0], license[1], true ).addField( misermode[0], misermode[1], true ).setFooter( lang.get('overview.inaccurate') );
  120. if ( readonly[1] ) embed.addField( readonly[0], readonly[1] );
  121. }
  122. else {
  123. text += version.join(' ') + '\n' + created.join(' ') + '\n' + language.join(' ') + '\n';
  124. if ( rtl[1] ) text += rtl.join(' ') + '\n';
  125. text += articles.join(' ') + '\n' + pages.join(' ') + '\n' + edits.join(' ') + '\n' + users.join(' ') + '\n' + license.join(' ') + '\n' + misermode.join(' ');
  126. if ( readonly[1] ) text += '\n\n' + ( readonly[0] === '\u200b' ? readonly[1] : readonly.join('\n') );
  127. text += '\n\n*' + lang.get('overview.inaccurate') + '*';
  128. }
  129. msg.sendChannelError( spoiler + text + spoiler, {embed} );
  130. if ( reaction ) reaction.removeEmoji();
  131. }
  132. else if ( ovbody.items.some( site => new URL(site.url).href === wiki.href ) ) {
  133. site = ovbody.items.find( site => new URL(site.url).href === wiki.href );
  134. var vertical = [lang.get('overview.vertical'), site.hub];
  135. var topic = [lang.get('overview.topic'), site.topic];
  136. var founder = [lang.get('overview.founder'), site.founding_user_id];
  137. if ( site.creation_date && creation_date > new Date(site.creation_date) ) {
  138. creation_date = new Date(site.creation_date);
  139. created[1] = creation_date.toLocaleString(lang.get('dateformat'), timeoptions);
  140. }
  141. var posts = [lang.get('overview.posts')];
  142. var walls = [lang.get('overview.walls')];
  143. var comments = [lang.get('overview.comments')];
  144. var description = [lang.get('overview.description'), site.desc];
  145. var image = [lang.get('overview.image'), site.image];
  146. if ( description[1] ) {
  147. description[1] = description[1].escapeFormatting();
  148. if ( description[1].length > 1000 ) description[1] = description[1].substring(0, 1000) + '\u2026';
  149. }
  150. if ( image[1] && image[1].startsWith( '/' ) ) image[1] = new URL(image[1], wiki).href;
  151. if ( msg.showEmbed() ) {
  152. embed.addField( vertical[0], vertical[1], true );
  153. if ( topic[1] ) embed.addField( topic[0], topic[1], true );
  154. }
  155. else text += vertical.join(' ') + ( topic[1] ? '\n' + topic.join(' ') : '' );
  156. Promise.all([
  157. ( founder[1] > 0 ? got.get( wiki + 'api.php?action=query&list=users&usprop=&ususerids=' + founder[1] + '&format=json' ).then( usresponse => {
  158. var usbody = usresponse.body;
  159. if ( usbody && usbody.warnings ) log_warn(usbody.warnings);
  160. if ( usresponse.statusCode !== 200 || !usbody || !usbody.query || !usbody.query.users || !usbody.query.users[0] ) {
  161. console.log( '- ' + usresponse.statusCode + ': Error while getting the wiki founder: ' + ( usbody && usbody.error && usbody.error.info ) );
  162. founder[1] = 'ID: ' + founder[1];
  163. }
  164. else {
  165. var user = usbody.query.users[0].name;
  166. if ( msg.showEmbed() ) founder[1] = '[' + user + '](' + wiki.toLink('User:' + user, '', '', true) + ')';
  167. else founder[1] = user;
  168. }
  169. }, error => {
  170. console.log( '- Error while getting the wiki founder: ' + error );
  171. founder[1] = 'ID: ' + founder[1];
  172. } ) : founder[1] = lang.get('overview.none') ),
  173. got.get( 'https://services.fandom.com/discussion/' + site.id + '/posts?limit=1&format=json&cache=' + Date.now(), {
  174. headers: {
  175. Accept: 'application/hal+json'
  176. }
  177. } ).then( dsresponse => {
  178. var dsbody = dsresponse.body;
  179. if ( dsresponse.statusCode !== 200 || !dsbody || dsbody.title ) {
  180. if ( dsbody?.title !== 'site doesn\'t exists' ) console.log( '- ' + dsresponse.statusCode + ': Error while getting discussions stats: ' + dsbody?.title );
  181. }
  182. else {
  183. let counts = dsbody?._embedded?.count?.[0];
  184. if ( counts?.FORUM || counts?.WALL || counts?.ARTICLE_COMMENT ) {
  185. if ( counts?.FORUM ) posts.push(counts.FORUM);
  186. if ( counts?.WALL ) walls.push(counts.WALL);
  187. if ( counts?.ARTICLE_COMMENT ) comments.push(counts.ARTICLE_COMMENT);
  188. }
  189. else if ( counts?.total ) posts.push(counts.total);
  190. }
  191. }, error => {
  192. console.log( '- Error while getting discussions stats: ' + error );
  193. } )
  194. ]).finally( () => {
  195. if ( msg.showEmbed() ) {
  196. embed.addField( founder[0], founder[1], true );
  197. if ( manager[1] ) embed.addField( manager[0], '[' + manager[1] + '](' + wiki.toLink('User:' + manager[1], '', '', true) + ') ([' + lang.get('overview.talk') + '](' + wiki.toLink('User talk:' + manager[1], '', '', true) + '))', true );
  198. embed.addField( version[0], version[1], true ).addField( created[0], created[1], true ).addField( language[0], language[1], true );
  199. if ( rtl[1] ) embed.addField( rtl[0], rtl[1], true );
  200. embed.addField( articles[0], articles[1], true ).addField( pages[0], pages[1], true ).addField( edits[0], edits[1], true );
  201. if ( posts[1] ) embed.addField( posts[0], posts[1], true );
  202. if ( walls[1] ) embed.addField( walls[0], walls[1], true );
  203. if ( comments[1] ) embed.addField( comments[0], comments[1], true );
  204. embed.addField( users[0], users[1], true ).addField( license[0], license[1], true ).addField( misermode[0], misermode[1], true ).setFooter( lang.get('overview.inaccurate') + ' • ' + lang.get('overview.wikiid') + ' ' + site.id );
  205. if ( crossover[1] ) {
  206. var crossoverSite = allSites.find( site => '<https://' + site.wiki_domain + '/>' === crossover[1] );
  207. if ( crossoverSite ) embed.addField( crossover[0], '[' + crossoverSite.wiki_display_name + '](' + crossover[1] + ')', true );
  208. else embed.addField( crossover[0], crossover[1], true );
  209. }
  210. if ( description[1] ) embed.addField( description[0], description[1] );
  211. if ( image[1] ) embed.addField( image[0], image[1] ).setImage( image[1] );
  212. if ( readonly[1] ) embed.addField( readonly[0], readonly[1] );
  213. }
  214. else {
  215. text += '\n' + founder.join(' ');
  216. if ( manager[1] ) text += '\n' + manager.join(' ');
  217. text += '\n' + version.join(' ') + '\n' + created.join(' ') + '\n' + language.join(' ');
  218. if ( rtl[1] ) text += '\n' + rtl.join(' ');
  219. text += '\n' + articles.join(' ') + '\n' + pages.join(' ') + '\n' + edits.join(' ');
  220. if ( posts[1] ) text += '\n' + posts.join(' ');
  221. if ( walls[1] ) text += '\n' + walls.join(' ');
  222. if ( comments[1] ) text += '\n' + comments.join(' ');
  223. text += '\n' + users.join(' ') + '\n' + license.join(' ') + '\n' + misermode.join(' ');
  224. if ( crossover[1] ) text += '\n' + crossover.join(' ');
  225. if ( description[1] ) text += '\n' + description.join(' ');
  226. if ( image[1] ) {
  227. text += '\n' + image.join(' ');
  228. if ( msg.uploadFiles() ) embed.files = [{attachment:image[1],name:( spoiler ? 'SPOILER ' : '' ) + body.query.general.sitename + image[1].substring(image[1].lastIndexOf('.'))}];
  229. }
  230. if ( readonly[1] ) text += '\n\n' + ( readonly[0] === '\u200b' ? readonly[1] : readonly.join('\n') );
  231. text += '\n\n*' + lang.get('overview.inaccurate') + '*';
  232. }
  233. msg.sendChannel( spoiler + text + spoiler, {embed} );
  234. if ( reaction ) reaction.removeEmoji();
  235. } );
  236. }
  237. else {
  238. if ( msg.showEmbed() ) {
  239. if ( manager[1] ) embed.addField( manager[0], '[' + manager[1] + '](' + wiki.toLink('User:' + manager[1], '', '', true) + ') ([' + lang.get('overview.talk') + '](' + wiki.toLink('User talk:' + manager[1], '', '', true) + '))', true );
  240. embed.addField( version[0], version[1], true ).addField( created[0], created[1], true ).addField( language[0], language[1], true );
  241. if ( rtl[1] ) embed.addField( rtl[0], rtl[1], true );
  242. embed.addField( articles[0], articles[1], true ).addField( pages[0], pages[1], true ).addField( edits[0], edits[1], true ).addField( users[0], users[1], true ).addField( license[0], license[1], true ).addField( misermode[0], misermode[1], true ).setFooter( lang.get('overview.inaccurate') );
  243. if ( crossover[1] ) {
  244. var crossoverSite = allSites.find( site => '<https://' + site.wiki_domain + '/>' === crossover[1] );
  245. if ( crossoverSite ) embed.addField( crossover[0], '[' + crossoverSite.wiki_display_name + '](' + crossover[1] + ')', true );
  246. else embed.addField( crossover[0], crossover[1], true );
  247. }
  248. if ( readonly[1] ) embed.addField( readonly[0], readonly[1] );
  249. }
  250. else {
  251. if ( manager[1] ) text += manager.join(' ') + '\n';
  252. text += version.join(' ') + '\n' + created.join(' ') + '\n' + language.join(' ') + '\n';
  253. if ( rtl[1] ) text += rtl.join(' ') + '\n';
  254. text += articles.join(' ') + '\n' + pages.join(' ') + '\n' + edits.join(' ') + '\n' + users.join(' ') + '\n' + license.join(' ') + '\n' + misermode.join(' ');
  255. if ( crossover[1] ) text += '\n' + crossover.join(' ');
  256. if ( readonly[1] ) text += '\n\n' + ( readonly[0] === '\u200b' ? readonly[1] : readonly.join('\n') );
  257. text += '\n\n*' + lang.get('overview.inaccurate') + '*';
  258. }
  259. msg.sendChannel( spoiler + text + spoiler, {embed} );
  260. if ( reaction ) reaction.removeEmoji();
  261. }
  262. }, error => {
  263. console.log( '- Error while getting the wiki details: ' + error );
  264. if ( msg.showEmbed() ) {
  265. embed.addField( version[0], version[1], true ).addField( created[0], created[1], true ).addField( language[0], language[1], true );
  266. if ( rtl[1] ) embed.addField( rtl[0], rtl[1], true );
  267. embed.addField( articles[0], articles[1], true ).addField( pages[0], pages[1], true ).addField( edits[0], edits[1], true ).addField( users[0], users[1], true ).addField( license[0], license[1], true ).addField( misermode[0], misermode[1], true ).setFooter( lang.get('overview.inaccurate') );
  268. if ( readonly[1] ) embed.addField( readonly[0], readonly[1] );
  269. }
  270. else {
  271. text += version.join(' ') + '\n' + created.join(' ') + '\n' + language.join(' ') + '\n';
  272. if ( rtl[1] ) text += rtl.join(' ') + '\n';
  273. text += articles.join(' ') + '\n' + pages.join(' ') + '\n' + edits.join(' ') + '\n' + users.join(' ') + '\n' + license.join(' ') + '\n' + misermode.join(' ');
  274. if ( readonly[1] ) text += '\n\n' + ( readonly[0] === '\u200b' ? readonly[1] : readonly.join('\n') );
  275. text += '\n\n*' + lang.get('overview.inaccurate') + '*';
  276. }
  277. msg.sendChannelError( spoiler + text + spoiler, {embed} );
  278. if ( reaction ) reaction.removeEmoji();
  279. } );
  280. else {
  281. if ( msg.showEmbed() ) {
  282. if ( site ) {
  283. var managerlist = manager[1].map( wm => '[' + wm + '](' + wiki.toLink('User:' + wm, '', '', true) + ') ([' + lang.get('overview.talk') + '](' + wiki.toLink('User talk:' + wm, '', '', true) + '))' ).join('\n');
  284. embed.addField( name[0], name[1], true ).addField( manager[0], ( managerlist || lang.get('overview.none') ), true ).addField( official[0], official[1], true ).setTimestamp( msg.client.readyTimestamp );
  285. }
  286. embed.addField( version[0], version[1], true ).addField( created[0], created[1], true ).addField( language[0], language[1], true );
  287. if ( rtl[1] ) embed.addField( rtl[0], rtl[1], true );
  288. embed.addField( articles[0], articles[1], true ).addField( pages[0], pages[1], true ).addField( edits[0], edits[1], true ).addField( users[0], users[1], true ).addField( license[0], license[1], true ).addField( misermode[0], misermode[1], true ).setFooter( lang.get('overview.inaccurate') );
  289. if ( site ) {
  290. if ( crossover[1] ) embed.addField( crossover[0], crossover[1], true );
  291. if ( description[1] ) embed.addField( description[0], description[1] );
  292. if ( image[1] ) embed.addField( image[0], image[1] ).setImage( image[1] );
  293. }
  294. if ( readonly[1] ) embed.addField( readonly[0], readonly[1] );
  295. }
  296. else {
  297. if ( site ) text += name.join(' ') + '\n' + manager[0] + ' ' + ( manager[1].join(', ') || lang.get('overview.none') ) + '\n' + official.join(' ') + '\n';
  298. text += version.join(' ') + '\n' + created.join(' ') + '\n' + language.join(' ') + '\n';
  299. if ( rtl[1] ) text += rtl.join(' ') + '\n';
  300. text += articles.join(' ') + '\n' + pages.join(' ') + '\n' + edits.join(' ') + '\n' + users.join(' ') + '\n' + license.join(' ') + '\n' + misermode.join(' ');
  301. if ( site ) {
  302. if ( crossover[1] ) text += '\n' + crossover.join(' ');
  303. if ( description[1] ) text += '\n' + description.join(' ');
  304. if ( image[1] ) {
  305. text += '\n' + image.join(' ');
  306. if ( msg.uploadFiles() ) embed.files = [{attachment:image[1],name:( spoiler ? 'SPOILER ' : '' ) + name[1] + image[1].substring(image[1].lastIndexOf('.'))}];
  307. }
  308. }
  309. if ( readonly[1] ) text += '\n\n' + ( readonly[0] === '\u200b' ? readonly[1] : readonly.join('\n') );
  310. text += '\n\n*' + lang.get('overview.inaccurate') + '*';
  311. }
  312. msg.sendChannel( spoiler + text + spoiler, {embed} );
  313. if ( reaction ) reaction.removeEmoji();
  314. }
  315. }
  316. }, error => {
  317. if ( wiki.noWiki(error.message) ) {
  318. console.log( '- This wiki doesn\'t exist!' );
  319. msg.reactEmoji('nowiki');
  320. }
  321. else {
  322. console.log( '- Error while getting the statistics: ' + error );
  323. msg.sendChannelError( spoiler + '<' + wiki.toLink('Special:Statistics') + '>' + spoiler );
  324. }
  325. if ( reaction ) reaction.removeEmoji();
  326. } );
  327. }
  328. module.exports = {
  329. name: 'overview',
  330. run: gamepedia_overview
  331. };