diff.js 14 KB

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