diff.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. import { MessageEmbed } from 'discord.js';
  2. import logging from '../../util/logging.js';
  3. import { got, htmlToPlain, htmlToDiscord, escapeFormatting } from '../../util/functions.js';
  4. import diffParser from '../../util/edit_diff.js';
  5. import { createRequire } from 'module';
  6. const require = createRequire(import.meta.url);
  7. const {timeoptions} = require('../../util/default.json');
  8. /**
  9. * Processes a Gamepedia edit.
  10. * @param {import('../../util/i18n.js').default} lang - The user language.
  11. * @param {import('discord.js').Message} msg - The Discord message.
  12. * @param {String[]} args - The command arguments.
  13. * @param {import('../../util/wiki.js').default} wiki - The wiki for the edit.
  14. * @param {import('discord.js').MessageReaction} reaction - The reaction on the message.
  15. * @param {String} spoiler - If the response is in a spoiler.
  16. * @param {Boolean} noEmbed - If the response should be without an embed.
  17. * @param {MessageEmbed} [embed] - The embed for the page.
  18. */
  19. export default function gamepedia_diff(lang, msg, args, wiki, reaction, spoiler, noEmbed, embed) {
  20. if ( args[0] ) {
  21. var error = false;
  22. var title = '';
  23. var revision = 0;
  24. var diff = 0;
  25. var relative = 'prev';
  26. if ( /^\d+$/.test(args[0]) ) {
  27. revision = parseInt(args[0], 10);
  28. if ( args[1] ) {
  29. if ( /^\d+$/.test(args[1]) ) {
  30. diff = parseInt(args[1], 10);
  31. }
  32. else if ( args[1] === 'prev' || args[1] === 'next' || args[1] === 'cur' ) {
  33. relative = args[1];
  34. }
  35. else error = true;
  36. }
  37. }
  38. else if ( args[0] === 'prev' || args[0] === 'next' || args[0] === 'cur' ) {
  39. relative = args[0];
  40. if ( args[1] ) {
  41. if ( /^\d+$/.test(args[1]) ) {
  42. revision = parseInt(args[1], 10);
  43. }
  44. else error = true;
  45. }
  46. else error = true;
  47. }
  48. else title = args.join(' ');
  49. if ( error ) {
  50. msg.reactEmoji('error');
  51. if ( reaction ) reaction.removeEmoji();
  52. }
  53. else if ( diff ) {
  54. gamepedia_diff_send(lang, msg, [diff, revision], wiki, reaction, spoiler, noEmbed);
  55. }
  56. else {
  57. got.get( wiki + 'api.php?action=compare&prop=ids|diff' + ( title ? '&fromtitle=' + encodeURIComponent( title ) : '&fromrev=' + revision ) + '&torelative=' + relative + '&format=json' ).then( response => {
  58. var body = response.body;
  59. if ( body && body.warnings ) log_warning(body.warnings);
  60. if ( response.statusCode !== 200 || !body || !body.compare ) {
  61. var noerror = false;
  62. if ( body && body.error ) {
  63. switch ( body.error.code ) {
  64. case 'nosuchrevid':
  65. noerror = true;
  66. break;
  67. case 'missingtitle':
  68. noerror = true;
  69. break;
  70. case 'invalidtitle':
  71. noerror = true;
  72. break;
  73. case 'missingcontent':
  74. noerror = true;
  75. break;
  76. default:
  77. noerror = false;
  78. }
  79. }
  80. if ( wiki.noWiki(response.url, response.statusCode) ) {
  81. console.log( '- This wiki doesn\'t exist!' );
  82. msg.reactEmoji('nowiki');
  83. }
  84. else if ( noerror ) {
  85. msg.replyMsg( {content: lang.get('diff.badrev'), allowedMentions: {repliedUser: false}} );
  86. }
  87. else {
  88. console.log( '- ' + response.statusCode + ': Error while getting the search results: ' + ( body && body.error && body.error.info ) );
  89. msg.sendChannelError( spoiler + '<' + wiki.toLink(title, ( title ? {diff} : {diff,oldid:revision} )) + '>' + spoiler );
  90. }
  91. if ( reaction ) reaction.removeEmoji();
  92. }
  93. else {
  94. if ( body.compare.fromarchive !== undefined || body.compare.toarchive !== undefined ) {
  95. msg.reactEmoji('error');
  96. if ( reaction ) reaction.removeEmoji();
  97. } else {
  98. var argids = [];
  99. var ids = body.compare;
  100. if ( ids.fromrevid && !ids.torevid ) argids = [ids.fromrevid];
  101. else if ( !ids.fromrevid && ids.torevid ) argids = [ids.torevid];
  102. else {
  103. argids = [ids.torevid, ids.fromrevid];
  104. var compare = ['', ''];
  105. if ( ids.fromtexthidden === undefined && ids.totexthidden === undefined && ids['*'] !== undefined ) {
  106. let more = '\n__' + lang.get('diff.info.more') + '__';
  107. let whitespace = '__' + lang.get('diff.info.whitespace') + '__';
  108. compare = diffParser( ids['*'], more, whitespace );
  109. }
  110. else if ( ids.fromtexthidden !== undefined ) compare[0] = '__' + lang.get('diff.hidden') + '__';
  111. else if ( ids.totexthidden !== undefined ) compare[1] = '__' + lang.get('diff.hidden') + '__';
  112. }
  113. gamepedia_diff_send(lang, msg, argids, wiki, reaction, spoiler, noEmbed, compare);
  114. }
  115. }
  116. }, error => {
  117. if ( wiki.noWiki(error.message) ) {
  118. console.log( '- This wiki doesn\'t exist!' );
  119. msg.reactEmoji('nowiki');
  120. }
  121. else {
  122. console.log( '- Error while getting the search results: ' + error );
  123. msg.sendChannelError( spoiler + '<' + wiki.toLink(title, 'diff=' + relative + ( title ? '' : '&oldid=' + revision )) + '>' + spoiler );
  124. }
  125. if ( reaction ) reaction.removeEmoji();
  126. } );
  127. }
  128. }
  129. else {
  130. if ( embed ) msg.sendChannel( {content: spoiler + '<' + embed.url + '>' + spoiler, embeds: ( noEmbed ? [] : [embed] )} );
  131. else msg.reactEmoji('error');
  132. if ( reaction ) reaction.removeEmoji();
  133. }
  134. }
  135. /**
  136. * Sends a Gamepedia edit.
  137. * @param {import('../../util/i18n.js').default} lang - The user language.
  138. * @param {import('discord.js').Message} msg - The Discord message.
  139. * @param {String[]} args - The command arguments.
  140. * @param {import('../../util/wiki.js').default} wiki - The wiki for the edit.
  141. * @param {import('discord.js').MessageReaction} reaction - The reaction on the message.
  142. * @param {String} spoiler - If the response is in a spoiler.
  143. * @param {Boolean} noEmbed - If the response should be without an embed.
  144. * @param {String[]} [compare] - The edit difference.
  145. */
  146. function gamepedia_diff_send(lang, msg, args, wiki, reaction, spoiler, noEmbed, compare) {
  147. got.get( wiki + 'api.php?uselang=' + lang.lang + '&action=query&meta=siteinfo&siprop=general&list=tags&tglimit=500&tgprop=displayname&prop=revisions&rvslots=main&rvprop=ids|timestamp|flags|user|size|parsedcomment|tags' + ( args.length === 1 || args[0] === args[1] ? '|content' : '' ) + '&revids=' + args.join('|') + '&format=json' ).then( response => {
  148. var body = response.body;
  149. if ( body && body.warnings ) log_warning(body.warnings);
  150. if ( response.statusCode !== 200 || !body || body.batchcomplete === undefined || !body.query ) {
  151. if ( wiki.noWiki(response.url, response.statusCode) ) {
  152. console.log( '- This wiki doesn\'t exist!' );
  153. msg.reactEmoji('nowiki');
  154. }
  155. else {
  156. console.log( '- ' + response.statusCode + ': Error while getting the search results: ' + ( body && body.error && body.error.info ) );
  157. msg.sendChannelError( spoiler + '<' + wiki.toLink('Special:Diff/' + ( args[1] ? args[1] + '/' : '' ) + args[0]) + '>' + spoiler );
  158. }
  159. if ( reaction ) reaction.removeEmoji();
  160. }
  161. else if ( body.query.badrevids ) {
  162. msg.replyMsg( {content: lang.get('diff.badrev'), allowedMentions: {repliedUser: false}} );
  163. if ( reaction ) reaction.removeEmoji();
  164. }
  165. else if ( body.query.pages && !body.query.pages['-1'] ) {
  166. wiki.updateWiki(body.query.general);
  167. logging(wiki, msg.guildId, 'diff');
  168. var pages = Object.values(body.query.pages);
  169. if ( pages.length !== 1 ) {
  170. msg.sendChannel( spoiler + '<' + wiki.toLink('Special:Diff/' + ( args[1] ? args[1] + '/' : '' ) + args[0]) + '>' + spoiler );
  171. if ( reaction ) reaction.removeEmoji();
  172. return;
  173. }
  174. var title = pages[0].title;
  175. var revisions = pages[0].revisions.sort( (first, second) => Date.parse(second.timestamp) - Date.parse(first.timestamp) );
  176. var diff = revisions[0].revid;
  177. var oldid = ( revisions[1] ? revisions[1].revid : 0 );
  178. var editor = [lang.get('diff.info.editor'), ( revisions[0].userhidden !== undefined ? lang.get('diff.hidden') : ( msg.showEmbed() && !noEmbed ? '[' + escapeFormatting(revisions[0].user) + '](' + wiki.toLink(( revisions[0].anon !== undefined ? 'Special:Contributions/' : 'User:' ) + revisions[0].user, '', '', true) + ')' : escapeFormatting(revisions[0].user) ) )];
  179. try {
  180. var dateformat = new Intl.DateTimeFormat(lang.get('dateformat'), Object.assign({
  181. timeZone: body.query.general.timezone
  182. }, timeoptions));
  183. }
  184. catch ( error ) {
  185. var dateformat = new Intl.DateTimeFormat(lang.get('dateformat'), Object.assign({
  186. timeZone: 'UTC'
  187. }, timeoptions));
  188. }
  189. var editDate = new Date(revisions[0].timestamp);
  190. var timestamp = [lang.get('diff.info.timestamp'), dateformat.format(editDate), '<t:' + Math.trunc(editDate.getTime() / 1000) + ':R>'];
  191. var difference = revisions[0].size - ( revisions[1] ? revisions[1].size : 0 );
  192. var size = [lang.get('diff.info.size'), lang.get('diff.info.bytes', ( difference > 0 ? '+' : '' ) + difference.toLocaleString(lang.get('dateformat')), difference) + ( revisions[0].minor !== undefined ? lang.get('diff.info.minor').replace( /_/g, ' ' ) : '' )];
  193. var comment = [lang.get('diff.info.comment'), ( revisions[0].commenthidden !== undefined ? lang.get('diff.hidden') : ( revisions[0].parsedcomment ? ( msg.showEmbed() && !noEmbed ? htmlToDiscord(revisions[0].parsedcomment, wiki.toLink(title), true) : htmlToPlain(revisions[0].parsedcomment) ) : lang.get('diff.nocomment') ) )];
  194. if ( revisions[0].tags.length ) var tags = [lang.get('diff.info.tags'), body.query.tags.filter( tag => tag.displayname && revisions[0].tags.includes( tag.name ) ).map( tag => tag.displayname || tag.name ).join(', ')];
  195. var pagelink = wiki.toLink(title, {diff,oldid});
  196. var text = '<' + pagelink + '>';
  197. if ( msg.showEmbed() && !noEmbed ) {
  198. var embed = new MessageEmbed().setAuthor( {name: body.query.general.sitename} ).setTitle( escapeFormatting( title + '?diff=' + diff + '&oldid=' + oldid ) ).setURL( pagelink ).addField( editor[0], editor[1], true ).addField( size[0], size[1], true ).addField( timestamp[0], timestamp[1] + '\n' + timestamp[2], true ).addField( comment[0], comment[1] ).setTimestamp( editDate );
  199. var more = '\n__' + lang.get('diff.info.more') + '__';
  200. var whitespace = '__' + lang.get('diff.info.whitespace') + '__';
  201. if ( !compare && oldid ) got.get( wiki + 'api.php?action=compare&prop=diff&fromrev=' + oldid + '&torev=' + diff + '&format=json' ).then( cpresponse => {
  202. var cpbody = cpresponse.body;
  203. if ( cpbody && cpbody.warnings ) log_warning(cpbody.warnings);
  204. if ( cpresponse.statusCode !== 200 || !cpbody || !cpbody.compare || cpbody.compare['*'] === undefined ) {
  205. var noerror = false;
  206. if ( cpbody && cpbody.error ) {
  207. switch ( cpbody.error.code ) {
  208. case 'nosuchrevid':
  209. noerror = true;
  210. break;
  211. case 'missingcontent':
  212. noerror = true;
  213. break;
  214. default:
  215. noerror = false;
  216. }
  217. }
  218. if ( !noerror ) console.log( '- ' + cpresponse.statusCode + ': Error while getting the diff: ' + ( cpbody && cpbody.error && cpbody.error.info ) );
  219. }
  220. else if ( cpbody.compare.fromtexthidden === undefined && cpbody.compare.totexthidden === undefined && cpbody.compare.fromarchive === undefined && cpbody.compare.toarchive === undefined ) {
  221. let edit_diff = diffParser( cpbody.compare['*'], more, whitespace )
  222. if ( edit_diff[0].length ) {
  223. embed.addField( lang.get('diff.info.removed'), edit_diff[0], true );
  224. }
  225. if ( edit_diff[1].length ) {
  226. embed.addField( lang.get('diff.info.added'), edit_diff[1], true );
  227. }
  228. }
  229. else if ( cpbody.compare.fromtexthidden !== undefined ) {
  230. embed.addField( lang.get('diff.info.removed'), '__' + lang.get('diff.hidden') + '__', true );
  231. }
  232. else if ( cpbody.compare.totexthidden !== undefined ) {
  233. embed.addField( lang.get('diff.info.added'), '__' + lang.get('diff.hidden') + '__', true );
  234. }
  235. }, error => {
  236. console.log( '- Error while getting the diff: ' + error );
  237. } ).finally( () => {
  238. if ( tags?.[1] ) embed.addField( tags[0], htmlToDiscord(tags[1], pagelink) );
  239. msg.sendChannel( {content: spoiler + text + spoiler, embeds: [embed]} );
  240. if ( reaction ) reaction.removeEmoji();
  241. } );
  242. else {
  243. if ( compare ) {
  244. if ( compare[0].length ) embed.addField( lang.get('diff.info.removed'), compare[0], true );
  245. if ( compare[1].length ) embed.addField( lang.get('diff.info.added'), compare[1], true );
  246. }
  247. else if ( ( revisions[0]?.slots?.main || revisions[0] )['*'] ) {
  248. var content = escapeFormatting( ( revisions[0]?.slots?.main || revisions[0] )['*'] );
  249. if ( content.trim().length ) {
  250. if ( content.length <= 1000 ) content = '**' + content + '**';
  251. else {
  252. content = content.substring(0, 1000 - more.length);
  253. content = '**' + content.substring(0, content.lastIndexOf('\n')) + '**' + more;
  254. }
  255. embed.addField( lang.get('diff.info.added'), content, true );
  256. } else embed.addField( lang.get('diff.info.added'), whitespace, true );
  257. }
  258. if ( tags?.[1] ) embed.addField( tags[0], htmlToDiscord(tags[1], pagelink) );
  259. msg.sendChannel( {content: spoiler + text + spoiler, embeds: [embed]} );
  260. if ( reaction ) reaction.removeEmoji();
  261. }
  262. }
  263. else {
  264. text += '\n\n' + editor.join(' ') + '\n' + timestamp.join(' ') + '\n' + size.join(' ') + '\n' + comment.join(' ');
  265. if ( tags?.[1] ) text += htmlToDiscord( '\n' + tags.join(' ') );
  266. msg.sendChannel( spoiler + text + spoiler );
  267. if ( reaction ) reaction.removeEmoji();
  268. }
  269. }
  270. else {
  271. msg.reactEmoji('error');
  272. if ( reaction ) reaction.removeEmoji();
  273. }
  274. }, error => {
  275. if ( wiki.noWiki(error.message) ) {
  276. console.log( '- This wiki doesn\'t exist!' );
  277. msg.reactEmoji('nowiki');
  278. }
  279. else {
  280. console.log( '- Error while getting the search results: ' + error );
  281. msg.sendChannelError( spoiler + '<' + wiki.toLink('Special:Diff/' + ( args[1] ? args[1] + '/' : '' ) + args[0]) + '>' + spoiler );
  282. }
  283. if ( reaction ) reaction.removeEmoji();
  284. } );
  285. }