diff.js 13 KB

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