1
0

verification.js 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689
  1. const {limit: {verification: verificationLimit}, usergroups} = require('../util/default.json');
  2. const Lang = require('../util/i18n.js');
  3. const {got, db, sendMsg, createNotice, escapeText, hasPerm} = require('./util.js');
  4. const fieldset = {
  5. channel: '<label for="wb-settings-channel">Channel:</label>'
  6. + '<select id="wb-settings-channel" name="channel" required></select>'
  7. + '<button type="button" id="wb-settings-channel-more" class="addmore">Add more</button>',
  8. role: '<label for="wb-settings-role">Role:</label>'
  9. + '<select id="wb-settings-role" name="role" required></select>'
  10. + '<button type="button" id="wb-settings-role-more" class="addmore">Add more</button>',
  11. usergroup: '<label for="wb-settings-usergroup">Wiki user group:</label>'
  12. + '<input type="text" id="wb-settings-usergroup" name="usergroup" list="wb-settings-usergroup-list" autocomplete="on">'
  13. + '<datalist id="wb-settings-usergroup-list">'
  14. + usergroups.sorted.filter( group => group !== '__CUSTOM__' ).map( group => {
  15. return `<option value="${group}"></option>`
  16. } ).join('')
  17. + usergroups.global.filter( group => group !== '__CUSTOM__' ).map( group => {
  18. return `<option value="${group}"></option>`
  19. } ).join('')
  20. + '</datalist>'
  21. + '<div id="wb-settings-usergroup-multiple">'
  22. + '<label for="wb-settings-usergroup-and">Require all user groups:</label>'
  23. + '<input type="checkbox" id="wb-settings-usergroup-and" name="usergroup_and">'
  24. + '</div>',
  25. editcount: '<label for="wb-settings-editcount">Minimal edit count:</label>'
  26. + '<input type="number" id="wb-settings-editcount" name="editcount" min="0" max="1000000" required>',
  27. postcount: '<div id="wb-settings-postcount-input">'
  28. + '<label for="wb-settings-postcount">Minimal post count:</label>'
  29. + '<input type="number" id="wb-settings-postcount" name="postcount" min="0" max="1000000" required>'
  30. + '</div><div class="wb-settings-postcount">'
  31. + '<span>Only Fandom wikis:</span>'
  32. + '<input type="radio" id="wb-settings-postcount-and" name="posteditcount" value="and" required>'
  33. + '<label for="wb-settings-postcount-and">Require both edit and post count.</label>'
  34. + '</div><div class="wb-settings-postcount">'
  35. + '<input type="radio" id="wb-settings-postcount-or" name="posteditcount" value="or" required>'
  36. + '<label for="wb-settings-postcount-or">Require either edit or post count.</label>'
  37. + '</div><div class="wb-settings-postcount">'
  38. + '<input type="radio" id="wb-settings-postcount-both" name="posteditcount" value="both" required>'
  39. + '<label for="wb-settings-postcount-both">Require combined edit and post count.</label>'
  40. + '</div>',
  41. accountage: '<label for="wb-settings-accountage">Account age (in days):</label>'
  42. + '<input type="number" id="wb-settings-accountage" name="accountage" min="0" max="1000000" required>',
  43. rename: '<label for="wb-settings-rename">Rename users:</label>'
  44. + '<input type="checkbox" id="wb-settings-rename" name="rename">',
  45. save: '<input type="submit" id="wb-settings-save" name="save_settings">',
  46. delete: '<input type="submit" id="wb-settings-delete" name="delete_settings" formnovalidate>'
  47. };
  48. /**
  49. * Create a settings form
  50. * @param {import('cheerio')} $ - The response body
  51. * @param {String} header - The form header
  52. * @param {import('./i18n.js')} dashboardLang - The user language
  53. * @param {Object} settings - The current settings
  54. * @param {String} settings.channel
  55. * @param {String} settings.role
  56. * @param {String} settings.usergroup
  57. * @param {Number} settings.editcount
  58. * @param {Number} settings.postcount
  59. * @param {Number} settings.accountage
  60. * @param {Boolean} settings.rename
  61. * @param {String} [settings.defaultrole]
  62. * @param {import('./util.js').Channel[]} guildChannels - The guild channels
  63. * @param {import('./util.js').Role[]} guildRoles - The guild roles
  64. */
  65. function createForm($, header, dashboardLang, settings, guildChannels, guildRoles) {
  66. var readonly = ( process.env.READONLY ? true : false );
  67. var fields = [];
  68. let channel = $('<div>').append(fieldset.channel);
  69. channel.find('label').text(dashboardLang.get('verification.form.channel'));
  70. let curCat = null;
  71. channel.find('#wb-settings-channel').append(
  72. $('<option class="wb-settings-channel-default defaultSelect" hidden>').val('').text(dashboardLang.get('verification.form.select_channel')),
  73. ...guildChannels.filter( guildChannel => {
  74. return ( hasPerm(guildChannel.userPermissions, 'VIEW_CHANNEL') || guildChannel.isCategory || settings.channel.includes( '|' + guildChannel.id + '|' ) );
  75. } ).map( guildChannel => {
  76. if ( guildChannel.isCategory ) {
  77. curCat = $('<optgroup>').attr('label', guildChannel.name);
  78. return curCat;
  79. }
  80. var optionChannel = $(`<option class="wb-settings-channel-${guildChannel.id}">`).val(guildChannel.id).text(`${guildChannel.id} – #${guildChannel.name}`);
  81. if ( !hasPerm(guildChannel.userPermissions, 'VIEW_CHANNEL') ) {
  82. optionChannel.addClass('wb-settings-error');
  83. }
  84. if ( !curCat ) return optionChannel;
  85. optionChannel.appendTo(curCat);
  86. } ).filter( catChannel => {
  87. if ( !catChannel ) return false;
  88. if ( catChannel.is('optgroup') && !catChannel.children('option').length ) return false;
  89. return true;
  90. } )
  91. );
  92. if ( settings.channel ) {
  93. let settingsChannels = settings.channel.split('|').filter( guildChannel => guildChannel.length );
  94. channel.find('#wb-settings-channel').append(
  95. ...settingsChannels.filter( guildChannel => {
  96. return !channel.find(`.wb-settings-channel-${guildChannel}`).length;
  97. } ).map( guildChannel => {
  98. return $(`<option class="wb-settings-channel-${guildChannel}">`).val(guildChannel).text(`${guildChannel} – #UNKNOWN`).addClass('wb-settings-error');
  99. } )
  100. );
  101. if ( settingsChannels.length > 1 ) channel.find('#wb-settings-channel').after(
  102. ...settingsChannels.slice(1).map( guildChannel => {
  103. var additionalChannel = channel.find('#wb-settings-channel').clone();
  104. additionalChannel.addClass('wb-settings-additional-select');
  105. additionalChannel.find(`.wb-settings-channel-default`).removeAttr('hidden');
  106. additionalChannel.find(`.wb-settings-channel-${guildChannel}`).attr('selected', '');
  107. return additionalChannel.removeAttr('id').removeAttr('required');
  108. } )
  109. );
  110. channel.find(`#wb-settings-channel .wb-settings-channel-${settingsChannels[0]}`).attr('selected', '');
  111. }
  112. else {
  113. channel.find('.wb-settings-channel-default').attr('selected', '');
  114. channel.find('button.addmore').attr('hidden', '');
  115. }
  116. fields.push(channel);
  117. let role = $('<div>').append(fieldset.role);
  118. role.find('label').text(dashboardLang.get('verification.form.role'));
  119. role.find('#wb-settings-role').append(
  120. $('<option class="wb-settings-role-default defaultSelect" hidden>').val('').text(dashboardLang.get('verification.form.select_role')),
  121. ...guildRoles.filter( guildRole => {
  122. return guildRole.lower || settings.role.split('|').includes( guildRole.id );
  123. } ).map( guildRole => {
  124. var optionRole = $(`<option class="wb-settings-role-${guildRole.id}">`).val(guildRole.id);
  125. if ( !guildRole.lower ) optionRole.addClass('wb-settings-error');
  126. return optionRole.text(`${guildRole.id} – @${guildRole.name}`);
  127. } )
  128. );
  129. if ( settings.role ) {
  130. let settingsRoles = settings.role.split('|');
  131. role.find('#wb-settings-role').append(
  132. ...settingsRoles.filter( guildRole => {
  133. return !role.find(`.wb-settings-role-${guildRole}`).length;
  134. } ).map( guildRole => {
  135. return $(`<option class="wb-settings-role-${guildRole}">`).val(guildRole).text(`${guildRole} – @UNKNOWN`).addClass('wb-settings-error');
  136. } )
  137. );
  138. if ( settingsRoles.length > 1 ) role.find('#wb-settings-role').after(
  139. ...settingsRoles.slice(1).map( guildRole => {
  140. var additionalRole = role.find('#wb-settings-role').clone();
  141. additionalRole.addClass('wb-settings-additional-select');
  142. additionalRole.find(`.wb-settings-role-default`).removeAttr('hidden');
  143. additionalRole.find(`.wb-settings-role-${guildRole}`).attr('selected', '');
  144. return additionalRole.removeAttr('id').removeAttr('required');
  145. } )
  146. );
  147. role.find(`#wb-settings-role .wb-settings-role-${settingsRoles[0]}`).attr('selected', '');
  148. }
  149. else {
  150. if ( role.find(`.wb-settings-role-${settings.defaultrole}`).length ) {
  151. role.find(`.wb-settings-role-${settings.defaultrole}`).attr('selected', '');
  152. }
  153. else role.find('.wb-settings-role-default').attr('selected', '');
  154. role.find('button.addmore').attr('hidden', '');
  155. }
  156. fields.push(role);
  157. let usergroup = $('<div>').append(fieldset.usergroup);
  158. usergroup.find('label').eq(0).text(dashboardLang.get('verification.form.usergroup'));
  159. usergroup.find('label').eq(1).text(dashboardLang.get('verification.form.usergroup_and'));
  160. if ( settings.usergroup.startsWith( 'AND|' ) ) {
  161. settings.usergroup = settings.usergroup.substring(4);
  162. usergroup.find('#wb-settings-usergroup-and').attr('checked', '');
  163. }
  164. usergroup.find('#wb-settings-usergroup').val(settings.usergroup.split('|').join(', '));
  165. if ( !settings.usergroup.includes( '|' ) ) {
  166. usergroup.find('#wb-settings-usergroup-multiple').attr('style', 'display: none;');
  167. }
  168. fields.push(usergroup);
  169. let editcount = $('<div>').append(fieldset.editcount);
  170. editcount.find('label').text(dashboardLang.get('verification.form.editcount'));
  171. editcount.find('#wb-settings-editcount').val(settings.editcount);
  172. fields.push(editcount);
  173. let postcount = $('<div>').append(fieldset.postcount);
  174. postcount.find('label').eq(0).text(dashboardLang.get('verification.form.postcount'));
  175. postcount.find('span').text(dashboardLang.get('verification.form.postcount_fandom'));
  176. postcount.find('label').eq(1).text(dashboardLang.get('verification.form.postcount_and'));
  177. postcount.find('label').eq(2).text(dashboardLang.get('verification.form.postcount_or'));
  178. postcount.find('label').eq(3).text(dashboardLang.get('verification.form.postcount_both'));
  179. postcount.find('#wb-settings-postcount').val(Math.abs(settings.postcount));
  180. if ( settings.postcount === null ) {
  181. postcount.find('#wb-settings-postcount-both').attr('checked', '');
  182. postcount.find('#wb-settings-postcount-input').attr('style', 'display: none;');
  183. }
  184. else if ( settings.postcount < 0 ) postcount.find('#wb-settings-postcount-or').attr('checked', '');
  185. else postcount.find('#wb-settings-postcount-and').attr('checked', '');
  186. fields.push(postcount);
  187. let accountage = $('<div>').append(fieldset.accountage);
  188. accountage.find('label').text(dashboardLang.get('verification.form.accountage'));
  189. accountage.find('#wb-settings-accountage').val(settings.accountage);
  190. fields.push(accountage);
  191. if ( settings.rename || guildChannels.some( guildChannel => {
  192. return hasPerm(guildChannel.botPermissions, 'MANAGE_NICKNAMES');
  193. } ) ) {
  194. let rename = $('<div>').append(fieldset.rename);
  195. rename.find('label').text(dashboardLang.get('verification.form.rename'));
  196. if ( settings.rename ) rename.find('#wb-settings-rename').attr('checked', '');
  197. fields.push(rename);
  198. }
  199. fields.push($(fieldset.save).val(dashboardLang.get('general.save')));
  200. if ( settings.channel ) {
  201. fields.push($(fieldset.delete).val(dashboardLang.get('general.delete')).attr('onclick', `return confirm('${dashboardLang.get('verification.form.confirm').replace( /'/g, '\\$&' )}');`));
  202. }
  203. var form = $('<fieldset>').append(...fields);
  204. if ( readonly ) {
  205. form.find('input').attr('readonly', '');
  206. form.find('input[type="checkbox"], option, optgroup').attr('disabled', '');
  207. form.find('input[type="submit"], button.addmore').remove();
  208. }
  209. form.find('button.addmore').text(dashboardLang.get('verification.form.more'));
  210. return $('<form id="wb-settings" method="post" enctype="application/x-www-form-urlencoded">').append(
  211. $('<h2>').text(header),
  212. form
  213. );
  214. }
  215. /**
  216. * Let a user change verifications
  217. * @param {import('http').ServerResponse} res - The server response
  218. * @param {import('cheerio')} $ - The response body
  219. * @param {import('./util.js').Guild} guild - The current guild
  220. * @param {String[]} args - The url parts
  221. * @param {import('./i18n.js')} dashboardLang - The user language
  222. */
  223. function dashboard_verification(res, $, guild, args, dashboardLang) {
  224. db.query( 'SELECT wiki, discord.role defaultrole, prefix, configid, verification.channel, verification.role, editcount, postcount, usergroup, accountage, rename FROM discord LEFT JOIN verification ON discord.guild = verification.guild WHERE discord.guild = $1 AND discord.channel IS NULL ORDER BY configid ASC', [guild.id] ).then( ({rows}) => {
  225. if ( rows.length === 0 ) {
  226. createNotice($, 'nosettings', dashboardLang, [guild.id]);
  227. $('#text .description').html(dashboardLang.get('verification.explanation'));
  228. $('#text code.prefix').prepend(escapeText(process.env.prefix));
  229. $('.channel#verification').addClass('selected');
  230. let body = $.html();
  231. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  232. res.write( body );
  233. return res.end();
  234. }
  235. if ( !hasPerm(guild.botPermissions, 'MANAGE_ROLES') ) {
  236. createNotice($, 'missingperm', dashboardLang, ['Manage Roles']);
  237. $('#text .description').html(dashboardLang.get('verification.explanation'));
  238. $('#text code.prefix').prepend(escapeText(rows[0].prefix));
  239. $('.channel#verification').addClass('selected');
  240. let body = $.html();
  241. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  242. res.write( body );
  243. return res.end();
  244. }
  245. var wiki = rows[0].wiki;
  246. var defaultrole = rows[0].defaultrole;
  247. var prefix = rows[0].prefix;
  248. if ( rows.length === 1 && rows[0].configid === null ) rows.pop();
  249. $('<p>').html(dashboardLang.get('verification.desc', true, $('<code>').text(guild.name))).appendTo('#text .description');
  250. let suffix = ( args[0] === 'owner' ? '?owner=true' : '' );
  251. $('#channellist #verification').after(
  252. ...rows.map( row => {
  253. return $('<a class="channel">').attr('id', `channel-${row.configid}`).append(
  254. $('<img>').attr('src', '/src/channel.svg'),
  255. $('<div>').text(`${row.configid} - ${( guild.roles.find( role => {
  256. return role.id === row.role.split('|')[0];
  257. } )?.name || guild.channels.find( channel => {
  258. return channel.id === row.channel.split('|')[1];
  259. } )?.name || row.usergroup.split('|')[( row.usergroup.startsWith('AND|') ? 1 : 0 )] )}`)
  260. ).attr('href', `/guild/${guild.id}/verification/${row.configid}${suffix}`);
  261. } ),
  262. ( process.env.READONLY || rows.length >= verificationLimit[( guild.patreon ? 'patreon' : 'default' )] ? '' :
  263. $('<a class="channel" id="channel-new">').append(
  264. $('<img>').attr('src', '/src/channel.svg'),
  265. $('<div>').text(dashboardLang.get('verification.new'))
  266. ).attr('href', `/guild/${guild.id}/verification/new${suffix}`) )
  267. );
  268. if ( args[4] === 'new' && !( process.env.READONLY || rows.length >= verificationLimit[( guild.patreon ? 'patreon' : 'default' )] ) ) {
  269. $('.channel#channel-new').addClass('selected');
  270. createForm($, dashboardLang.get('verification.form.new'), dashboardLang, {
  271. channel: '', role: '', usergroup: 'user',
  272. editcount: 0, postcount: 0, accountage: 0,
  273. rename: false, defaultrole
  274. }, guild.channels, guild.roles).attr('action', `/guild/${guild.id}/verification/new`).appendTo('#text');
  275. }
  276. else if ( rows.some( row => row.configid.toString() === args[4] ) ) {
  277. let row = rows.find( row => row.configid.toString() === args[4] );
  278. $(`.channel#channel-${row.configid}`).addClass('selected');
  279. createForm($, dashboardLang.get('verification.form.entry', false, row.configid), dashboardLang, row, guild.channels, guild.roles).attr('action', `/guild/${guild.id}/verification/${row.configid}`).appendTo('#text');
  280. }
  281. else {
  282. $('.channel#verification').addClass('selected');
  283. $('#text .description').html(dashboardLang.get('verification.explanation'));
  284. $('#text code.prefix').prepend(escapeText(prefix));
  285. }
  286. let body = $.html();
  287. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  288. res.write( body );
  289. return res.end();
  290. }, dberror => {
  291. console.log( '- Dashboard: Error while getting the verifications: ' + dberror );
  292. createNotice($, 'error', dashboardLang);
  293. $('#text .description').html(dashboardLang.get('verification.explanation'));
  294. $('#text code.prefix').prepend(escapeText(process.env.prefix));
  295. $('.channel#verification').addClass('selected');
  296. let body = $.html();
  297. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  298. res.write( body );
  299. return res.end();
  300. } );
  301. }
  302. /**
  303. * Change verifications
  304. * @param {Function} res - The server response
  305. * @param {import('./util.js').Settings} userSettings - The settings of the user
  306. * @param {String} guild - The id of the guild
  307. * @param {String|Number} type - The setting to change
  308. * @param {Object} settings - The new settings
  309. * @param {String[]} settings.channel
  310. * @param {String[]} settings.role
  311. * @param {String[]} [settings.usergroup]
  312. * @param {String} [settings.usergroup_and]
  313. * @param {Number} settings.editcount
  314. * @param {Number} [settings.postcount]
  315. * @param {String} settings.posteditcount
  316. * @param {Number} settings.accountage
  317. * @param {String} [settings.rename]
  318. * @param {String} [settings.save_settings]
  319. * @param {String} [settings.delete_settings]
  320. */
  321. function update_verification(res, userSettings, guild, type, settings) {
  322. if ( type === 'default' ) {
  323. return res(`/guild/${guild}/verification`, 'savefail');
  324. }
  325. if ( !settings.save_settings === !settings.delete_settings ) {
  326. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  327. }
  328. if ( settings.save_settings ) {
  329. if ( !/^[\d|]+ [\d|]+$/.test(`${settings.channel} ${settings.role}`) ) {
  330. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  331. }
  332. if ( !/^\d+ \d+$/.test(`${settings.editcount} ${settings.accountage}`) ) {
  333. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  334. }
  335. if ( !( ['and','or','both'].includes( settings.posteditcount ) && ( /^\d+$/.test(settings.postcount) || settings.posteditcount === 'both' ) ) ) {
  336. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  337. }
  338. settings.channel = settings.channel.split('|').filter( (channel, i, self) => {
  339. return ( channel.length && self.indexOf(channel) === i );
  340. } );
  341. if ( !settings.channel.length || settings.channel.length > 10 ) {
  342. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  343. }
  344. settings.role = settings.role.split('|').filter( (role, i, self) => {
  345. return ( role.length && self.indexOf(role) === i );
  346. } );
  347. if ( !settings.role.length || settings.role.length > 10 ) {
  348. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  349. }
  350. if ( !settings.usergroup ) settings.usergroup = 'user';
  351. settings.usergroup = settings.usergroup.replace( /_/g, ' ' ).trim().toLowerCase();
  352. settings.usergroup = settings.usergroup.split(/\s*[,|]\s*/).map( usergroup => {
  353. if ( usergroup === '*' ) return 'user';
  354. return usergroup.replace( / /g, '_' );
  355. } ).filter( (usergroup, i, self) => {
  356. return ( usergroup.length && self.indexOf(usergroup) === i );
  357. } );
  358. if ( !settings.usergroup.length ) settings.usergroup.push('user');
  359. if ( settings.usergroup.length > 10 || settings.usergroup.some( usergroup => {
  360. return ( usergroup.length > 100 );
  361. } ) ) return res(`/guild/${guild}/verification/${type}`, 'invalidusergroup');
  362. settings.editcount = parseInt(settings.editcount, 10);
  363. settings.accountage = parseInt(settings.accountage, 10);
  364. if ( settings.editcount > 1000000 || settings.accountage > 1000000 ) {
  365. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  366. }
  367. if ( settings.posteditcount === 'both' ) settings.postcount = null;
  368. else settings.postcount = parseInt(settings.postcount, 10);
  369. if ( settings.posteditcount === 'or' ) settings.postcount = settings.postcount * -1;
  370. if ( settings.postcount > 1000000 || settings.postcount < -1000000 ) {
  371. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  372. }
  373. if ( type === 'new' ) {
  374. let curGuild = userSettings.guilds.isMember.get(guild);
  375. if ( settings.channel.some( channel => {
  376. return !curGuild.channels.some( guildChannel => {
  377. return ( guildChannel.id === channel && !guildChannel.isCategory );
  378. } );
  379. } ) || settings.role.some( role => {
  380. return !curGuild.roles.some( guildRole => {
  381. return ( guildRole.id === role && guildRole.lower );
  382. } );
  383. } ) ) return res(`/guild/${guild}/verification/new`, 'savefail');
  384. }
  385. }
  386. if ( settings.delete_settings && type === 'new' ) {
  387. return res(`/guild/${guild}/verification/new`, 'savefail');
  388. }
  389. if ( type !== 'new' ) type = parseInt(type, 10);
  390. sendMsg( {
  391. type: 'getMember',
  392. member: userSettings.user.id,
  393. guild: guild
  394. } ).then( response => {
  395. if ( !response ) {
  396. userSettings.guilds.notMember.set(guild, userSettings.guilds.isMember.get(guild));
  397. userSettings.guilds.isMember.delete(guild);
  398. return res(`/guild/${guild}`, 'savefail');
  399. }
  400. if ( response === 'noMember' || !hasPerm(response.userPermissions, 'MANAGE_GUILD') ) {
  401. userSettings.guilds.isMember.delete(guild);
  402. return res('/', 'savefail');
  403. }
  404. if ( settings.delete_settings ) return db.query( 'DELETE FROM verification WHERE guild = $1 AND configid = $2 RETURNING channel, role, editcount, postcount, usergroup, accountage, rename', [guild, type] ).then( ({rows:[row]}) => {
  405. console.log( `- Dashboard: Verification successfully removed: ${guild}#${type}` );
  406. res(`/guild/${guild}/verification`, 'save');
  407. if ( row ) db.query( 'SELECT lang FROM discord WHERE guild = $1 AND channel IS NULL', [guild] ).then( ({rows:[channel]}) => {
  408. var lang = new Lang(channel.lang);
  409. var text = lang.get('verification.dashboard.removed', `<@${userSettings.user.id}>`, type);
  410. if ( row ) {
  411. text += '\n' + lang.get('verification.channel') + ' <#' + row.channel.split('|').filter( channel => channel.length ).join('>, <#') + '>';
  412. text += '\n' + lang.get('verification.role') + ' <@&' + row.role.split('|').join('>, <@&') + '>';
  413. if ( row.postcount === null ) {
  414. text += '\n' + lang.get('verification.posteditcount') + ' `' + row.editcount + '`';
  415. }
  416. else {
  417. text += '\n' + lang.get('verification.editcount') + ' `' + row.editcount + '`';
  418. text += '\n' + lang.get('verification.postcount') + ' `' + Math.abs(row.postcount) + '`';
  419. if ( row.postcount < 0 ) text += ' ' + lang.get('verification.postcount_or');
  420. }
  421. text += '\n' + lang.get('verification.usergroup') + ' `' + ( row.usergroup.startsWith( 'AND|' ) ? row.usergroup.split('|').slice(1).join('` ' + lang.get('verification.and') + ' `') : row.usergroup.split('|').join('` ' + lang.get('verification.or') + ' `') ) + '`';
  422. text += '\n' + lang.get('verification.accountage') + ' `' + row.accountage + '` ' + lang.get('verification.indays');
  423. text += '\n' + lang.get('verification.rename') + ' *`' + lang.get('verification.' + ( row.rename ? 'enabled' : 'disabled')) + '`*';
  424. }
  425. text += `\n<${new URL(`/guild/${guild}/verification`, process.env.dashboard).href}>`;
  426. sendMsg( {
  427. type: 'notifyGuild', guild, text
  428. } ).catch( error => {
  429. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  430. } );
  431. }, dberror => {
  432. console.log( '- Dashboard: Error while notifying the guild: ' + dberror );
  433. } );
  434. }, dberror => {
  435. console.log( '- Dashboard: Error while removing the verification: ' + dberror );
  436. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  437. } );
  438. if ( !hasPerm(response.botPermissions, 'MANAGE_ROLES') ) {
  439. return res(`/guild/${guild}/verification`, 'savefail');
  440. }
  441. if ( type === 'new' ) return db.query( 'SELECT wiki, lang, ARRAY_REMOVE(ARRAY_AGG(configid ORDER BY configid), NULL) count FROM discord LEFT JOIN verification ON discord.guild = verification.guild WHERE discord.guild = $1 AND discord.channel IS NULL GROUP BY wiki, lang', [guild] ).then( ({rows:[row]}) => {
  442. if ( !row ) return res(`/guild/${guild}/verification`, 'savefail');
  443. if ( row.count.length >= verificationLimit[( response.patreon ? 'patreon' : 'default' )] ) {
  444. return res(`/guild/${guild}/verification`, 'savefail');
  445. }
  446. return got.get( row.wiki + 'api.php?action=query&meta=allmessages&amprefix=group-&amincludelocal=true&amenableparser=true&format=json' ).then( gresponse => {
  447. var body = gresponse.body;
  448. if ( gresponse.statusCode !== 200 || body?.batchcomplete === undefined || !body?.query?.allmessages ) {
  449. console.log( '- Dashboard: ' + gresponse.statusCode + ': Error while getting the usergroups: ' + body?.error?.info );
  450. return;
  451. }
  452. var groups = body.query.allmessages.filter( group => {
  453. if ( group.name === 'group-all' ) return false;
  454. if ( group.name === 'group-membership-link-with-expiry' ) return false;
  455. if ( group.name.endsWith( '.css' ) || group.name.endsWith( '.js' ) ) return false;
  456. return true;
  457. } ).map( group => {
  458. return {
  459. name: group.name.replace( /^group-/, '' ).replace( /-member$/, '' ),
  460. content: group['*'].replace( / /g, '_' ).toLowerCase()
  461. };
  462. } );
  463. settings.usergroup = settings.usergroup.map( usergroup => {
  464. if ( groups.some( group => group.name === usergroup ) ) return usergroup;
  465. if ( groups.some( group => group.content === usergroup ) ) {
  466. return groups.find( group => group.content === usergroup ).name;
  467. }
  468. if ( /^admins?$/.test(usergroup) ) return 'sysop';
  469. if ( usergroup === '*' ) return 'user';
  470. return usergroup;
  471. } );
  472. }, error => {
  473. console.log( '- Dashboard: Error while getting the usergroups: ' + error );
  474. } ).finally( () => {
  475. if ( settings.usergroup_and ) settings.usergroup.unshift('AND');
  476. var configid = 1;
  477. for ( let i of row.count ) {
  478. if ( configid === i ) configid++;
  479. else break;
  480. }
  481. db.query( 'INSERT INTO verification(guild, configid, channel, role, editcount, postcount, usergroup, accountage, rename) VALUES($1, $2, $3, $4, $5, $6, $7, $8, $9)', [guild, configid, '|' + settings.channel.join('|') + '|', settings.role.join('|'), settings.editcount, settings.postcount, settings.usergroup.join('|'), settings.accountage, ( settings.rename ? 1 : 0 )] ).then( () => {
  482. console.log( `- Dashboard: Verification successfully added: ${guild}#${configid}` );
  483. res(`/guild/${guild}/verification/${configid}`, 'save');
  484. var lang = new Lang(row.lang);
  485. var text = lang.get('verification.dashboard.added', `<@${userSettings.user.id}>`, configid);
  486. text += '\n' + lang.get('verification.channel') + ' <#' + settings.channel.join('>, <#') + '>';
  487. text += '\n' + lang.get('verification.role') + ' <@&' + settings.role.join('>, <@&') + '>';
  488. if ( settings.postcount === null ) {
  489. text += '\n' + lang.get('verification.posteditcount') + ' `' + settings.editcount + '`';
  490. }
  491. else {
  492. text += '\n' + lang.get('verification.editcount') + ' `' + settings.editcount + '`';
  493. text += '\n' + lang.get('verification.postcount') + ' `' + Math.abs(settings.postcount) + '`';
  494. if ( settings.postcount < 0 ) text += ' ' + lang.get('verification.postcount_or');
  495. }
  496. text += '\n' + lang.get('verification.usergroup') + ' `' + ( settings.usergroup_and ? settings.usergroup.slice(1).join('` ' + lang.get('verification.and') + ' `') : settings.usergroup.join('` ' + lang.get('verification.or') + ' `') ) + '`';
  497. text += '\n' + lang.get('verification.accountage') + ' `' + settings.accountage + '` ' + lang.get('verification.indays');
  498. text += '\n' + lang.get('verification.rename') + ' *`' + lang.get('verification.' + ( settings.rename ? 'enabled' : 'disabled')) + '`*';
  499. text += `\n<${new URL(`/guild/${guild}/verification/${configid}`, process.env.dashboard).href}>`;
  500. if ( settings.rename && !hasPerm(response.botPermissions, 'MANAGE_NICKNAMES') ) {
  501. text += '\n\n' + lang.get('verification.rename_no_permission', `<@${process.env.bot}>`);
  502. }
  503. if ( settings.role.some( role => {
  504. return !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  505. return ( guildRole.id === role && guildRole.lower );
  506. } );
  507. } ) ) {
  508. text += '\n';
  509. settings.role.forEach( role => {
  510. if ( !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  511. return ( guildRole.id === role );
  512. } ) ) {
  513. text += '\n' + lang.get('verification.role_deleted', `<@&${role}>`);
  514. }
  515. else if ( userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  516. return ( guildRole.id === role && !guildRole.lower );
  517. } ) ) {
  518. text += '\n' + lang.get('verification.role_too_high', `<@&${role}>`, `<@${process.env.bot}>`);
  519. }
  520. } );
  521. }
  522. sendMsg( {
  523. type: 'notifyGuild', guild, text
  524. } ).catch( error => {
  525. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  526. } );
  527. }, dberror => {
  528. console.log( '- Dashboard: Error while adding the verification: ' + dberror );
  529. return res(`/guild/${guild}/verification/new`, 'savefail');
  530. } );
  531. } );
  532. }, dberror => {
  533. console.log( '- Dashboard: Error while checking for verifications: ' + dberror );
  534. return res(`/guild/${guild}/verification/new`, 'savefail');
  535. } );
  536. return db.query( 'SELECT wiki, lang, verification.channel, verification.role, editcount, postcount, usergroup, accountage, rename FROM discord LEFT JOIN verification ON discord.guild = verification.guild AND verification.configid = $1 WHERE discord.guild = $2 AND discord.channel IS NULL', [type, guild] ).then( ({rows:[row]}) => {
  537. if ( !row?.channel ) return res(`/guild/${guild}/verification`, 'savefail');
  538. row.channel = row.channel.split('|').filter( channel => channel.length );
  539. var newChannel = settings.channel.filter( channel => !row.channel.includes( channel ) );
  540. row.role = row.role.split('|');
  541. var newRole = settings.role.filter( role => !row.role.includes( role ) );
  542. row.usergroup = row.usergroup.split('|');
  543. var newUsergroup = settings.usergroup.filter( group => !row.usergroup.includes( group ) );
  544. if ( newChannel.length || newRole.length ) {
  545. let curGuild = userSettings.guilds.isMember.get(guild);
  546. if ( newChannel.some( channel => {
  547. return !curGuild.channels.some( guildChannel => {
  548. return ( guildChannel.id === channel && !guildChannel.isCategory );
  549. } );
  550. } ) || newRole.some( role => {
  551. return !curGuild.roles.some( guildRole => {
  552. return ( guildRole.id === role && guildRole.lower );
  553. } );
  554. } ) ) return res(`/guild/${guild}/verification/${type}`, 'savefail');
  555. }
  556. ( newUsergroup.length ? got.get( row.wiki + 'api.php?action=query&meta=allmessages&amprefix=group-&amincludelocal=true&amenableparser=true&format=json' ).then( gresponse => {
  557. var body = gresponse.body;
  558. if ( gresponse.statusCode !== 200 || body?.batchcomplete === undefined || !body?.query?.allmessages ) {
  559. console.log( '- Dashboard: ' + gresponse.statusCode + ': Error while getting the usergroups: ' + body?.error?.info );
  560. return;
  561. }
  562. var groups = body.query.allmessages.filter( group => {
  563. if ( group.name === 'group-all' ) return false;
  564. if ( group.name === 'group-membership-link-with-expiry' ) return false;
  565. if ( group.name.endsWith( '.css' ) || group.name.endsWith( '.js' ) ) return false;
  566. return true;
  567. } ).map( group => {
  568. return {
  569. name: group.name.replace( /^group-/, '' ).replace( /-member$/, '' ),
  570. content: group['*'].replace( / /g, '_' ).toLowerCase()
  571. };
  572. } );
  573. settings.usergroup = settings.usergroup.map( usergroup => {
  574. if ( groups.some( group => group.name === usergroup ) ) return usergroup;
  575. if ( groups.some( group => group.content === usergroup ) ) {
  576. return groups.find( group => group.content === usergroup ).name;
  577. }
  578. if ( /^admins?$/.test(usergroup) ) return 'sysop';
  579. if ( usergroup === '*' ) return 'user';
  580. return usergroup;
  581. } );
  582. }, error => {
  583. console.log( '- Dashboard: Error while getting the usergroups: ' + error );
  584. } ) : Promise.resolve() ).finally( () => {
  585. if ( settings.usergroup_and ) settings.usergroup.unshift('AND');
  586. var lang = new Lang(row.lang);
  587. var diff = [];
  588. if ( newChannel.length || row.channel.some( channel => {
  589. return !settings.channel.includes( channel );
  590. } ) ) {
  591. diff.push(lang.get('verification.channel') + ` ~~<#${row.channel.join('>, <#')}>~~ → <#${settings.channel.join('>, <#')}>`);
  592. }
  593. if ( newRole.length || row.role.some( role => {
  594. return !settings.role.includes( role );
  595. } ) ) {
  596. diff.push(lang.get('verification.role') + ` ~~<@&${row.role.join('>, <@&')}>~~ → <@&${settings.role.join('>, <@&')}>`);
  597. }
  598. if ( row.postcount !== settings.postcount && ( row.postcount === null || settings.postcount === null ) ) {
  599. if ( row.postcount === null ) {
  600. diff.push('~~' + lang.get('verification.posteditcount') + ` \`${row.editcount}\`~~`);
  601. diff.push('→ ' + lang.get('verification.editcount') + ` \`${settings.editcount}\``);
  602. diff.push('→ ' + lang.get('verification.postcount') + ` \`${Math.abs(settings.postcount)}\`` + ( settings.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ));
  603. }
  604. if ( settings.postcount === null ) {
  605. diff.push('~~' + lang.get('verification.editcount') + ` \`${row.editcount}\`~~`);
  606. diff.push('~~' + lang.get('verification.postcount') + ` \`${Math.abs(row.postcount)}\`` + ( row.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ) + '~~');
  607. diff.push('→ ' + lang.get('verification.posteditcount') + ` \`${settings.editcount}\``);
  608. }
  609. }
  610. else {
  611. if ( row.editcount !== settings.editcount ) {
  612. diff.push(lang.get('verification.editcount') + ` ~~\`${row.editcount}\`~~ → \`${settings.editcount}\``);
  613. }
  614. if ( row.postcount !== settings.postcount ) {
  615. if ( ( row.postcount >= 0 && settings.postcount < 0 ) || ( row.postcount < 0 && settings.postcount >= 0 ) ) {
  616. diff.push('~~' + lang.get('verification.postcount') + ` \`${Math.abs(row.postcount)}\`` + ( row.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ) + '~~');
  617. diff.push('→ ' + lang.get('verification.postcount') + ` \`${Math.abs(settings.postcount)}\`` + ( settings.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ));
  618. }
  619. else diff.push(lang.get('verification.postcount') + ` ~~\`${Math.abs(row.postcount)}\`~~ → \`${Math.abs(settings.postcount)}\`` + ( settings.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ));
  620. }
  621. }
  622. if ( newUsergroup.length || row.usergroup.some( usergroup => {
  623. return !settings.usergroup.includes( usergroup );
  624. } ) ) {
  625. diff.push(lang.get('verification.usergroup') + ' ~~`' + ( row.usergroup[0] === 'AND' ? row.usergroup.slice(1).join('` ' + lang.get('verification.and') + ' `') : row.usergroup.join('` ' + lang.get('verification.or') + ' `') ) + '`~~ → `' + ( settings.usergroup_and ? settings.usergroup.slice(1).join('` ' + lang.get('verification.and') + ' `') : settings.usergroup.join('` ' + lang.get('verification.or') + ' `') ) + '`');
  626. }
  627. if ( row.accountage !== settings.accountage ) {
  628. diff.push(lang.get('verification.accountage') + ` ~~\`${row.accountage}\`~~ → \`${settings.accountage}\``);
  629. }
  630. if ( row.rename !== ( settings.rename ? 1 : 0 ) ) {
  631. diff.push(lang.get('verification.rename') + ` ~~*\`${lang.get('verification.' + ( row.rename ? 'enabled' : 'disabled'))}\`*~~ → *\`${lang.get('verification.' + ( settings.rename ? 'enabled' : 'disabled'))}\`*`);
  632. }
  633. if ( !diff.length ) return res(`/guild/${guild}/verification/${type}`, 'save');
  634. db.query( 'UPDATE verification SET channel = $1, role = $2, editcount = $3, postcount = $4, usergroup = $5, accountage = $6, rename = $7 WHERE guild = $8 AND configid = $9', ['|' + settings.channel.join('|') + '|', settings.role.join('|'), settings.editcount, settings.postcount, settings.usergroup.join('|'), settings.accountage, ( settings.rename ? 1 : 0 ), guild, type] ).then( () => {
  635. console.log( `- Dashboard: Verification successfully updated: ${guild}#${type}` );
  636. res(`/guild/${guild}/verification/${type}`, 'save');
  637. var text = lang.get('verification.dashboard.updated', `<@${userSettings.user.id}>`, type);
  638. text += '\n' + diff.join('\n');
  639. text += `\n<${new URL(`/guild/${guild}/verification/${type}`, process.env.dashboard).href}>`;
  640. if ( settings.rename && !hasPerm(response.botPermissions, 'MANAGE_NICKNAMES') ) {
  641. text += '\n\n' + lang.get('verification.rename_no_permission', `<@${process.env.bot}>`);
  642. }
  643. if ( settings.role.some( role => {
  644. return !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  645. return ( guildRole.id === role && guildRole.lower );
  646. } );
  647. } ) ) {
  648. text += '\n';
  649. settings.role.forEach( role => {
  650. if ( !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  651. return ( guildRole.id === role );
  652. } ) ) {
  653. text += '\n' + lang.get('verification.role_deleted', `<@&${role}>`);
  654. }
  655. else if ( userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  656. return ( guildRole.id === role && !guildRole.lower );
  657. } ) ) {
  658. text += '\n' + lang.get('verification.role_too_high', `<@&${role}>`, `<@${process.env.bot}>`);
  659. }
  660. } );
  661. }
  662. sendMsg( {
  663. type: 'notifyGuild', guild, text
  664. } ).catch( error => {
  665. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  666. } );
  667. }, dberror => {
  668. console.log( '- Dashboard: Error while updating the verification: ' + dberror );
  669. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  670. } );
  671. } );
  672. }, dberror => {
  673. console.log( '- Dashboard: Error while checking for verifications: ' + dberror );
  674. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  675. } );
  676. }, error => {
  677. console.log( '- Dashboard: Error while getting the member: ' + error );
  678. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  679. } );
  680. }
  681. module.exports = {
  682. get: dashboard_verification,
  683. post: update_verification
  684. };