verification.js 50 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936
  1. const {limit: {verification: verificationLimit}, usergroups} = require('../util/default.json');
  2. const Lang = require('../util/i18n.js');
  3. const {got, db, slashCommands, sendMsg, createNotice, escapeText, hasPerm} = require('./util.js');
  4. const slashCommand = slashCommands.find( slashCommand => slashCommand.name === 'verify' );
  5. const fieldset = {
  6. channel: '<label for="wb-settings-channel">Channel:</label>'
  7. + '<select id="wb-settings-channel" name="channel" required></select>'
  8. + '<button type="button" id="wb-settings-channel-more" class="addmore">Add more</button>',
  9. role: '<label for="wb-settings-role">Role:</label>'
  10. + '<select id="wb-settings-role" name="role" required></select>'
  11. + '<button type="button" id="wb-settings-role-more" class="addmore">Add more</button>',
  12. usergroup: '<label for="wb-settings-usergroup">Wiki user group:</label>'
  13. + '<input type="text" id="wb-settings-usergroup" name="usergroup" list="wb-settings-usergroup-list" autocomplete="on">'
  14. + '<datalist id="wb-settings-usergroup-list">'
  15. + usergroups.sorted.filter( group => group !== '__CUSTOM__' ).map( group => {
  16. return `<option value="${group}"></option>`
  17. } ).join('')
  18. + usergroups.global.filter( group => group !== '__CUSTOM__' ).map( group => {
  19. return `<option value="${group}"></option>`
  20. } ).join('')
  21. + '</datalist>'
  22. + '<div id="wb-settings-usergroup-multiple">'
  23. + '<label for="wb-settings-usergroup-and">Require all user groups:</label>'
  24. + '<input type="checkbox" id="wb-settings-usergroup-and" name="usergroup_and">'
  25. + '</div>',
  26. editcount: '<label for="wb-settings-editcount">Minimal edit count:</label>'
  27. + '<input type="number" id="wb-settings-editcount" name="editcount" min="0" max="1000000" required>',
  28. postcount: '<div id="wb-settings-postcount-input">'
  29. + '<label for="wb-settings-postcount">Minimal post count:</label>'
  30. + '<input type="number" id="wb-settings-postcount" name="postcount" min="0" max="1000000" required>'
  31. + '</div><div class="wb-settings-postcount">'
  32. + '<span>Only Fandom wikis:</span>'
  33. + '<input type="radio" id="wb-settings-postcount-and" name="posteditcount" value="and" required>'
  34. + '<label for="wb-settings-postcount-and">Require both edit and post count.</label>'
  35. + '</div><div class="wb-settings-postcount">'
  36. + '<input type="radio" id="wb-settings-postcount-or" name="posteditcount" value="or" required>'
  37. + '<label for="wb-settings-postcount-or">Require either edit or post count.</label>'
  38. + '</div><div class="wb-settings-postcount">'
  39. + '<input type="radio" id="wb-settings-postcount-both" name="posteditcount" value="both" required>'
  40. + '<label for="wb-settings-postcount-both">Require combined edit and post count.</label>'
  41. + '</div>',
  42. accountage: '<label for="wb-settings-accountage">Account age (in days):</label>'
  43. + '<input type="number" id="wb-settings-accountage" name="accountage" min="0" max="1000000" required>',
  44. rename: '<label for="wb-settings-rename">Rename users:</label>'
  45. + '<input type="checkbox" id="wb-settings-rename" name="rename">',
  46. save: '<input type="submit" id="wb-settings-save" name="save_settings">',
  47. delete: '<input type="submit" id="wb-settings-delete" name="delete_settings" formnovalidate>'
  48. };
  49. /**
  50. * Create a settings form
  51. * @param {import('cheerio')} $ - The response body
  52. * @param {String} header - The form header
  53. * @param {import('./i18n.js')} dashboardLang - The user language
  54. * @param {Object} settings - The current settings
  55. * @param {String} settings.channel
  56. * @param {String} settings.role
  57. * @param {String} settings.usergroup
  58. * @param {Number} settings.editcount
  59. * @param {Number} settings.postcount
  60. * @param {Number} settings.accountage
  61. * @param {Boolean} settings.rename
  62. * @param {String} [settings.defaultrole]
  63. * @param {import('./util.js').Channel[]} guildChannels - The guild channels
  64. * @param {import('./util.js').Role[]} guildRoles - The guild roles
  65. */
  66. function createForm($, header, dashboardLang, settings, guildChannels, guildRoles) {
  67. var readonly = ( process.env.READONLY ? true : false );
  68. var fields = [];
  69. let channel = $('<div>').append(fieldset.channel);
  70. channel.find('label').text(dashboardLang.get('verification.form.channel'));
  71. let curCat = null;
  72. channel.find('#wb-settings-channel').append(
  73. $('<option class="wb-settings-channel-default defaultSelect" hidden>').val('').text(dashboardLang.get('verification.form.select_channel')),
  74. ...guildChannels.filter( guildChannel => {
  75. return ( hasPerm(guildChannel.userPermissions, 'VIEW_CHANNEL') || guildChannel.isCategory || settings.channel.includes( '|' + guildChannel.id + '|' ) );
  76. } ).map( guildChannel => {
  77. if ( guildChannel.isCategory ) {
  78. curCat = $('<optgroup>').attr('label', guildChannel.name);
  79. return curCat;
  80. }
  81. var optionChannel = $(`<option class="wb-settings-channel-${guildChannel.id}">`).val(guildChannel.id).text(`${guildChannel.id} – #${guildChannel.name}`);
  82. if ( !hasPerm(guildChannel.userPermissions, 'VIEW_CHANNEL') ) {
  83. optionChannel.addClass('wb-settings-error');
  84. }
  85. if ( !curCat ) return optionChannel;
  86. optionChannel.appendTo(curCat);
  87. } ).filter( catChannel => {
  88. if ( !catChannel ) return false;
  89. if ( catChannel.is('optgroup') && !catChannel.children('option').length ) return false;
  90. return true;
  91. } )
  92. );
  93. if ( settings.channel ) {
  94. let settingsChannels = settings.channel.split('|').filter( guildChannel => guildChannel.length );
  95. channel.find('#wb-settings-channel').append(
  96. ...settingsChannels.filter( guildChannel => {
  97. return !channel.find(`.wb-settings-channel-${guildChannel}`).length;
  98. } ).map( guildChannel => {
  99. return $(`<option class="wb-settings-channel-${guildChannel}">`).val(guildChannel).text(`${guildChannel} – #UNKNOWN`).addClass('wb-settings-error');
  100. } )
  101. );
  102. if ( settingsChannels.length > 1 ) channel.find('#wb-settings-channel').after(
  103. ...settingsChannels.slice(1).map( guildChannel => {
  104. var additionalChannel = channel.find('#wb-settings-channel').clone();
  105. additionalChannel.addClass('wb-settings-additional-select');
  106. additionalChannel.find(`.wb-settings-channel-default`).removeAttr('hidden');
  107. additionalChannel.find(`.wb-settings-channel-${guildChannel}`).attr('selected', '');
  108. return additionalChannel.removeAttr('id').removeAttr('required');
  109. } )
  110. );
  111. channel.find(`#wb-settings-channel .wb-settings-channel-${settingsChannels[0]}`).attr('selected', '');
  112. }
  113. else {
  114. channel.find('.wb-settings-channel-default').attr('selected', '');
  115. channel.find('button.addmore').attr('hidden', '');
  116. }
  117. fields.push(channel);
  118. let role = $('<div>').append(fieldset.role);
  119. role.find('label').text(dashboardLang.get('verification.form.role'));
  120. role.find('#wb-settings-role').append(
  121. $('<option class="wb-settings-role-default defaultSelect" hidden>').val('').text(dashboardLang.get('verification.form.select_role')),
  122. ...guildRoles.filter( guildRole => {
  123. return guildRole.lower || settings.role.split('|').includes( guildRole.id );
  124. } ).map( guildRole => {
  125. var optionRole = $(`<option class="wb-settings-role-${guildRole.id}">`).val(guildRole.id);
  126. if ( !guildRole.lower ) optionRole.addClass('wb-settings-error');
  127. return optionRole.text(`${guildRole.id} – @${guildRole.name}`);
  128. } )
  129. );
  130. if ( settings.role ) {
  131. let settingsRoles = settings.role.split('|');
  132. role.find('#wb-settings-role').append(
  133. ...settingsRoles.filter( guildRole => {
  134. return !role.find(`.wb-settings-role-${guildRole}`).length;
  135. } ).map( guildRole => {
  136. return $(`<option class="wb-settings-role-${guildRole}">`).val(guildRole).text(`${guildRole} – @UNKNOWN`).addClass('wb-settings-error');
  137. } )
  138. );
  139. if ( settingsRoles.length > 1 ) role.find('#wb-settings-role').after(
  140. ...settingsRoles.slice(1).map( guildRole => {
  141. var additionalRole = role.find('#wb-settings-role').clone();
  142. additionalRole.addClass('wb-settings-additional-select');
  143. additionalRole.find(`.wb-settings-role-default`).removeAttr('hidden');
  144. additionalRole.find(`.wb-settings-role-${guildRole}`).attr('selected', '');
  145. return additionalRole.removeAttr('id').removeAttr('required');
  146. } )
  147. );
  148. role.find(`#wb-settings-role .wb-settings-role-${settingsRoles[0]}`).attr('selected', '');
  149. }
  150. else {
  151. if ( role.find(`.wb-settings-role-${settings.defaultrole}`).length ) {
  152. role.find(`.wb-settings-role-${settings.defaultrole}`).attr('selected', '');
  153. }
  154. else role.find('.wb-settings-role-default').attr('selected', '');
  155. role.find('button.addmore').attr('hidden', '');
  156. }
  157. fields.push(role);
  158. let usergroup = $('<div>').append(fieldset.usergroup);
  159. usergroup.find('label').eq(0).text(dashboardLang.get('verification.form.usergroup'));
  160. usergroup.find('label').eq(1).text(dashboardLang.get('verification.form.usergroup_and'));
  161. if ( settings.usergroup.startsWith( 'AND|' ) ) {
  162. settings.usergroup = settings.usergroup.substring(4);
  163. usergroup.find('#wb-settings-usergroup-and').attr('checked', '');
  164. }
  165. usergroup.find('#wb-settings-usergroup').val(settings.usergroup.split('|').join(', '));
  166. if ( !settings.usergroup.includes( '|' ) ) {
  167. usergroup.find('#wb-settings-usergroup-multiple').attr('style', 'display: none;');
  168. }
  169. fields.push(usergroup);
  170. let editcount = $('<div>').append(fieldset.editcount);
  171. editcount.find('label').text(dashboardLang.get('verification.form.editcount'));
  172. editcount.find('#wb-settings-editcount').val(settings.editcount);
  173. fields.push(editcount);
  174. let postcount = $('<div>').append(fieldset.postcount);
  175. postcount.find('label').eq(0).text(dashboardLang.get('verification.form.postcount'));
  176. postcount.find('span').text(dashboardLang.get('verification.form.postcount_fandom'));
  177. postcount.find('label').eq(1).text(dashboardLang.get('verification.form.postcount_and'));
  178. postcount.find('label').eq(2).text(dashboardLang.get('verification.form.postcount_or'));
  179. postcount.find('label').eq(3).text(dashboardLang.get('verification.form.postcount_both'));
  180. postcount.find('#wb-settings-postcount').val(Math.abs(settings.postcount));
  181. if ( settings.postcount === null ) {
  182. postcount.find('#wb-settings-postcount-both').attr('checked', '');
  183. postcount.find('#wb-settings-postcount-input').attr('style', 'display: none;');
  184. }
  185. else if ( settings.postcount < 0 ) postcount.find('#wb-settings-postcount-or').attr('checked', '');
  186. else postcount.find('#wb-settings-postcount-and').attr('checked', '');
  187. fields.push(postcount);
  188. let accountage = $('<div>').append(fieldset.accountage);
  189. accountage.find('label').text(dashboardLang.get('verification.form.accountage'));
  190. accountage.find('#wb-settings-accountage').val(settings.accountage);
  191. fields.push(accountage);
  192. if ( settings.rename || guildChannels.some( guildChannel => {
  193. return hasPerm(guildChannel.botPermissions, 'MANAGE_NICKNAMES');
  194. } ) ) {
  195. let rename = $('<div>').append(fieldset.rename);
  196. rename.find('label').text(dashboardLang.get('verification.form.rename'));
  197. if ( settings.rename ) rename.find('#wb-settings-rename').attr('checked', '');
  198. fields.push(rename);
  199. }
  200. fields.push($(fieldset.save).val(dashboardLang.get('general.save')));
  201. if ( settings.channel ) {
  202. fields.push($(fieldset.delete).val(dashboardLang.get('general.delete')).attr('onclick', `return confirm('${dashboardLang.get('verification.form.confirm').replace( /'/g, '\\$&' )}');`));
  203. }
  204. var form = $('<fieldset>').append(...fields);
  205. if ( readonly ) {
  206. form.find('input').attr('readonly', '');
  207. form.find('input[type="checkbox"], option, optgroup').attr('disabled', '');
  208. form.find('input[type="submit"], button.addmore').remove();
  209. }
  210. form.find('button.addmore').text(dashboardLang.get('verification.form.more'));
  211. return $('<form id="wb-settings" method="post" enctype="application/x-www-form-urlencoded">').append(
  212. $('<h2>').text(header),
  213. form
  214. );
  215. }
  216. /**
  217. * Let a user change verifications
  218. * @param {import('http').ServerResponse} res - The server response
  219. * @param {import('cheerio')} $ - The response body
  220. * @param {import('./util.js').Guild} guild - The current guild
  221. * @param {String[]} args - The url parts
  222. * @param {import('./i18n.js')} dashboardLang - The user language
  223. */
  224. function dashboard_verification(res, $, guild, args, dashboardLang) {
  225. 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}) => {
  226. if ( rows.length === 0 ) {
  227. createNotice($, 'nosettings', dashboardLang, [guild.id]);
  228. $('#text .description').html(dashboardLang.get('verification.explanation'));
  229. $('#text code.prefix').prepend(escapeText(process.env.prefix));
  230. $('.channel#verification').addClass('selected');
  231. let body = $.html();
  232. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  233. res.write( body );
  234. return res.end();
  235. }
  236. if ( !hasPerm(guild.botPermissions, 'MANAGE_ROLES') ) {
  237. createNotice($, 'missingperm', dashboardLang, ['Manage Roles']);
  238. $('#text .description').html(dashboardLang.get('verification.explanation'));
  239. $('#text code.prefix').prepend(escapeText(rows[0].prefix));
  240. $('.channel#verification').addClass('selected');
  241. let body = $.html();
  242. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  243. res.write( body );
  244. return res.end();
  245. }
  246. var wiki = rows[0].wiki;
  247. var defaultrole = rows[0].defaultrole;
  248. var prefix = rows[0].prefix;
  249. if ( rows.length === 1 && rows[0].configid === null ) rows.pop();
  250. $('<p>').html(dashboardLang.get('verification.desc', true, $('<code>').text(guild.name))).appendTo('#text .description');
  251. let suffix = ( args[0] === 'owner' ? '?owner=true' : '' );
  252. $('#channellist #verification').after(
  253. ...rows.map( row => {
  254. let text = `${row.configid} - ${( guild.roles.find( role => {
  255. return role.id === row.role.split('|')[0];
  256. } )?.name || guild.channels.find( channel => {
  257. return channel.id === row.channel.split('|')[1];
  258. } )?.name || row.usergroup.split('|')[( row.usergroup.startsWith('AND|') ? 1 : 0 )] )}`;
  259. return $('<a class="channel">').attr('id', `channel-${row.configid}`).append(
  260. $('<img>').attr('src', '/src/channel.svg'),
  261. $('<div>').text(text)
  262. ).attr('title', text).attr('href', `/guild/${guild.id}/verification/${row.configid}${suffix}`);
  263. } ),
  264. ( process.env.READONLY || rows.length >= verificationLimit[( guild.patreon ? 'patreon' : 'default' )] ? '' :
  265. $('<a class="channel" id="channel-new">').append(
  266. $('<img>').attr('src', '/src/channel.svg'),
  267. $('<div>').text(dashboardLang.get('verification.new'))
  268. ).attr('href', `/guild/${guild.id}/verification/new${suffix}`) ),
  269. ( !rows.length ? '' :
  270. $('<a class="channel" id="channel-notice">').append(
  271. $('<img>').attr('src', '/src/channel.svg'),
  272. $('<div>').text(dashboardLang.get('verification.notice'))
  273. ).attr('href', `/guild/${guild.id}/verification/notice${suffix}`) )
  274. );
  275. if ( args[4] === 'new' && !( process.env.READONLY || rows.length >= verificationLimit[( guild.patreon ? 'patreon' : 'default' )] ) ) {
  276. $('.channel#channel-new').addClass('selected');
  277. createForm($, dashboardLang.get('verification.form.new'), dashboardLang, {
  278. channel: '', role: '', usergroup: 'user',
  279. editcount: 0, postcount: 0, accountage: 0,
  280. rename: false, defaultrole
  281. }, guild.channels, guild.roles).attr('action', `/guild/${guild.id}/verification/new`).appendTo('#text');
  282. }
  283. else if ( rows.some( row => row.configid.toString() === args[4] ) ) {
  284. let row = rows.find( row => row.configid.toString() === args[4] );
  285. $(`.channel#channel-${row.configid}`).addClass('selected');
  286. 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');
  287. }
  288. else if ( args[4] === 'notice' && rows.length ) {
  289. $(`.channel#channel-notice`).addClass('selected');
  290. return db.query( 'SELECT logchannel, onsuccess, onmatch FROM verifynotice WHERE guild = $1', [guild.id] ).then( ({rows:[row]}) => {
  291. let curCat = null;
  292. $('<form id="wb-settings" method="post" enctype="application/x-www-form-urlencoded">').append(
  293. $('<h2>').text(dashboardLang.get('verification.form.notice')),
  294. $('<fieldset>').append(
  295. $('<div>').append(
  296. $('<label for="wb-settings-channel">').text(dashboardLang.get('verification.form.logging')),
  297. $('<select id="wb-settings-channel" name="channel">').append(
  298. $('<option class="wb-settings-channel-default defaultSelect">').val('').text(dashboardLang.get('verification.form.select_channel')),
  299. ...guild.channels.filter( guildChannel => {
  300. return ( ( hasPerm(guildChannel.botPermissions, 'VIEW_CHANNEL', 'SEND_MESSAGES') && hasPerm(guildChannel.userPermissions, 'VIEW_CHANNEL') ) || guildChannel.isCategory );
  301. } ).map( guildChannel => {
  302. if ( guildChannel.isCategory ) {
  303. curCat = $('<optgroup>').attr('label', guildChannel.name);
  304. return curCat;
  305. }
  306. var optionChannel = $(`<option class="wb-settings-channel-${guildChannel.id}">`).val(guildChannel.id).text(`${guildChannel.id} – #${guildChannel.name}`);
  307. if ( guildChannel.id === row?.logchannel ) optionChannel.attr('selected', '');
  308. if ( !curCat ) return optionChannel;
  309. optionChannel.appendTo(curCat);
  310. } ).filter( catChannel => {
  311. if ( !catChannel ) return false;
  312. if ( catChannel.is('optgroup') && !catChannel.children('option').length ) return false;
  313. return true;
  314. } )
  315. )
  316. ),
  317. $('<div>').append(
  318. $('<label for="wb-settings-success">').text(dashboardLang.get('verification.form.success')).append(
  319. $('<div>').html('&nbsp;')
  320. ),
  321. $('<textarea id="wb-settings-success" name="success" spellcheck="true" maxlength="1000" cols="65">').attr('rows', ( row?.onsuccess || '' ).split('\n').length + 3).attr('placeholder', dashboardLang.get('verification.form.success_placeholder')).text(row?.onsuccess || '')
  322. ),
  323. $('<div>').append(
  324. $('<label for="wb-settings-match">').text(dashboardLang.get('verification.form.match')).append(
  325. $('<div>').html('&nbsp;')
  326. ),
  327. $('<textarea id="wb-settings-match" name="match" spellcheck="true" maxlength="1000" cols="65">').attr('rows', ( row?.onmatch || '' ).split('\n').length + 3).attr('placeholder', dashboardLang.get('verification.form.match_placeholder')).text(row?.onmatch || '')
  328. ),
  329. $('<input type="submit" id="wb-settings-save" name="save_settings">').val(dashboardLang.get('general.save'))
  330. )
  331. ).attr('action', `/guild/${guild.id}/verification/notice`).appendTo('#text');
  332. if ( process.env.READONLY ) {
  333. $('input, textarea').attr('readonly', '');
  334. $('textarea, option, optgroup').attr('disabled', '');
  335. $('input[type="submit"]').remove();
  336. }
  337. $('<div class="description">').html(dashboardLang.get('verification.help_notice')).appendTo('#text');
  338. let body = $.html();
  339. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  340. res.write( body );
  341. return res.end();
  342. }, dberror => {
  343. console.log( '- Dashboard: Error while getting the verification notices: ' + dberror );
  344. createNotice($, 'error', dashboardLang);
  345. $('#text .description').html(dashboardLang.get('verification.explanation'));
  346. $('#text code.prefix').prepend(escapeText(process.env.prefix));
  347. $('.channel#verification').addClass('selected');
  348. let body = $.html();
  349. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  350. res.write( body );
  351. return res.end();
  352. } );
  353. }
  354. else {
  355. $('.channel#verification').addClass('selected');
  356. $('#text .description').html(dashboardLang.get('verification.explanation'));
  357. $('#text code.prefix').prepend(escapeText(prefix));
  358. }
  359. let body = $.html();
  360. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  361. res.write( body );
  362. return res.end();
  363. }, dberror => {
  364. console.log( '- Dashboard: Error while getting the verifications: ' + dberror );
  365. createNotice($, 'error', dashboardLang);
  366. $('#text .description').html(dashboardLang.get('verification.explanation'));
  367. $('#text code.prefix').prepend(escapeText(process.env.prefix));
  368. $('.channel#verification').addClass('selected');
  369. let body = $.html();
  370. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  371. res.write( body );
  372. return res.end();
  373. } );
  374. }
  375. /**
  376. * Change verifications
  377. * @param {Function} res - The server response
  378. * @param {import('./util.js').Settings} userSettings - The settings of the user
  379. * @param {String} guild - The id of the guild
  380. * @param {String|Number} type - The setting to change
  381. * @param {Object} settings - The new settings
  382. * @param {String[]} settings.channel
  383. * @param {String[]} settings.role
  384. * @param {String[]} [settings.usergroup]
  385. * @param {String} [settings.usergroup_and]
  386. * @param {Number} settings.editcount
  387. * @param {Number} [settings.postcount]
  388. * @param {String} settings.posteditcount
  389. * @param {Number} settings.accountage
  390. * @param {String} [settings.rename]
  391. * @param {String} [settings.save_settings]
  392. * @param {String} [settings.delete_settings]
  393. */
  394. function update_verification(res, userSettings, guild, type, settings) {
  395. if ( type === 'default' ) {
  396. return res(`/guild/${guild}/verification`, 'savefail');
  397. }
  398. if ( type === 'notice' ) return update_notices(res, userSettings, guild, type, settings);
  399. if ( !settings.save_settings === !settings.delete_settings ) {
  400. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  401. }
  402. if ( settings.save_settings ) {
  403. if ( !/^[\d|]+ [\d|]+$/.test(`${settings.channel} ${settings.role}`) ) {
  404. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  405. }
  406. if ( !/^\d+ \d+$/.test(`${settings.editcount} ${settings.accountage}`) ) {
  407. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  408. }
  409. if ( !( ['and','or','both'].includes( settings.posteditcount ) && ( /^\d+$/.test(settings.postcount) || settings.posteditcount === 'both' ) ) ) {
  410. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  411. }
  412. settings.channel = settings.channel.split('|').filter( (channel, i, self) => {
  413. return ( channel.length && self.indexOf(channel) === i );
  414. } );
  415. if ( !settings.channel.length || settings.channel.length > 10 ) {
  416. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  417. }
  418. settings.role = settings.role.split('|').filter( (role, i, self) => {
  419. return ( role.length && self.indexOf(role) === i );
  420. } );
  421. if ( !settings.role.length || settings.role.length > 10 ) {
  422. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  423. }
  424. if ( !settings.usergroup ) settings.usergroup = 'user';
  425. settings.usergroup = settings.usergroup.replace( /_/g, ' ' ).trim().toLowerCase();
  426. settings.usergroup = settings.usergroup.split(/\s*[,|]\s*/).map( usergroup => {
  427. if ( usergroup === '*' ) return 'user';
  428. return usergroup.replace( / /g, '_' );
  429. } ).filter( (usergroup, i, self) => {
  430. return ( usergroup.length && self.indexOf(usergroup) === i );
  431. } );
  432. if ( !settings.usergroup.length ) settings.usergroup.push('user');
  433. if ( settings.usergroup.length > 10 || settings.usergroup.some( usergroup => {
  434. return ( usergroup.length > 100 );
  435. } ) ) return res(`/guild/${guild}/verification/${type}`, 'invalidusergroup');
  436. settings.editcount = parseInt(settings.editcount, 10);
  437. settings.accountage = parseInt(settings.accountage, 10);
  438. if ( settings.editcount > 1000000 || settings.accountage > 1000000 ) {
  439. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  440. }
  441. if ( settings.posteditcount === 'both' ) settings.postcount = null;
  442. else settings.postcount = parseInt(settings.postcount, 10);
  443. if ( settings.posteditcount === 'or' ) settings.postcount = settings.postcount * -1;
  444. if ( settings.postcount > 1000000 || settings.postcount < -1000000 ) {
  445. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  446. }
  447. if ( type === 'new' ) {
  448. let curGuild = userSettings.guilds.isMember.get(guild);
  449. if ( settings.channel.some( channel => {
  450. return !curGuild.channels.some( guildChannel => {
  451. return ( guildChannel.id === channel && !guildChannel.isCategory );
  452. } );
  453. } ) || settings.role.some( role => {
  454. return !curGuild.roles.some( guildRole => {
  455. return ( guildRole.id === role && guildRole.lower );
  456. } );
  457. } ) ) return res(`/guild/${guild}/verification/new`, 'savefail');
  458. }
  459. }
  460. if ( settings.delete_settings && type === 'new' ) {
  461. return res(`/guild/${guild}/verification/new`, 'savefail');
  462. }
  463. if ( type !== 'new' ) type = parseInt(type, 10);
  464. sendMsg( {
  465. type: 'getMember',
  466. member: userSettings.user.id,
  467. guild: guild
  468. } ).then( response => {
  469. if ( !response ) {
  470. userSettings.guilds.notMember.set(guild, userSettings.guilds.isMember.get(guild));
  471. userSettings.guilds.isMember.delete(guild);
  472. return res(`/guild/${guild}`, 'savefail');
  473. }
  474. if ( response === 'noMember' || !hasPerm(response.userPermissions, 'MANAGE_GUILD') ) {
  475. userSettings.guilds.isMember.delete(guild);
  476. return res('/', 'savefail');
  477. }
  478. 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]}) => {
  479. console.log( `- Dashboard: Verification successfully removed: ${guild}#${type}` );
  480. res(`/guild/${guild}/verification`, 'save');
  481. if ( slashCommand?.id ) db.query( 'SELECT COUNT(1) FROM verification WHERE guild = $1', [guild] ).then( ({rows:[{count}]}) => {
  482. if ( count > 0 ) return;
  483. got.put( 'https://discord.com/api/v8/applications/' + process.env.bot + '/guilds/' + guild + '/commands/' + slashCommand.id + '/permissions', {
  484. headers:{
  485. Authorization: 'Bot ' + process.env.token
  486. },
  487. json: {
  488. permissions: []
  489. },
  490. timeout: 10000
  491. } ).then( response=> {
  492. if ( response.statusCode !== 200 || !response.body ) {
  493. console.log( '- Dashboard: ' + response.statusCode + ': Error while disabling the slash command: ' + response.body?.message );
  494. return;
  495. }
  496. console.log( '- Dashboard: Slash command successfully disabled.' );
  497. }, error => {
  498. console.log( '- Dashboard: Error while disabling the slash command: ' + error );
  499. } );
  500. }, dberror => {
  501. console.log( '- Dashboard: Error while disabling the slash command: ' + dberror );
  502. } );
  503. if ( row ) db.query( 'SELECT lang FROM discord WHERE guild = $1 AND channel IS NULL', [guild] ).then( ({rows:[channel]}) => {
  504. var lang = new Lang(channel.lang);
  505. var text = lang.get('verification.dashboard.removed', `<@${userSettings.user.id}>`, type);
  506. if ( row ) {
  507. text += '\n' + lang.get('verification.channel') + ' <#' + row.channel.split('|').filter( channel => channel.length ).join('>, <#') + '>';
  508. text += '\n' + lang.get('verification.role') + ' <@&' + row.role.split('|').join('>, <@&') + '>';
  509. if ( row.postcount === null ) {
  510. text += '\n' + lang.get('verification.posteditcount') + ' `' + row.editcount + '`';
  511. }
  512. else {
  513. text += '\n' + lang.get('verification.editcount') + ' `' + row.editcount + '`';
  514. text += '\n' + lang.get('verification.postcount') + ' `' + Math.abs(row.postcount) + '`';
  515. if ( row.postcount < 0 ) text += ' ' + lang.get('verification.postcount_or');
  516. }
  517. 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') + ' `') ) + '`';
  518. text += '\n' + lang.get('verification.accountage') + ' `' + row.accountage + '` ' + lang.get('verification.indays');
  519. text += '\n' + lang.get('verification.rename') + ' *`' + lang.get('verification.' + ( row.rename ? 'enabled' : 'disabled')) + '`*';
  520. }
  521. text += `\n<${new URL(`/guild/${guild}/verification`, process.env.dashboard).href}>`;
  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 notifying the guild: ' + dberror );
  529. } );
  530. }, dberror => {
  531. console.log( '- Dashboard: Error while removing the verification: ' + dberror );
  532. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  533. } );
  534. if ( !hasPerm(response.botPermissions, 'MANAGE_ROLES') ) {
  535. return res(`/guild/${guild}/verification`, 'savefail');
  536. }
  537. 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]}) => {
  538. if ( !row ) return res(`/guild/${guild}/verification`, 'savefail');
  539. if ( row.count.length >= verificationLimit[( response.patreon ? 'patreon' : 'default' )] ) {
  540. return res(`/guild/${guild}/verification`, 'savefail');
  541. }
  542. return got.get( row.wiki + 'api.php?action=query&meta=allmessages&amprefix=group-&amincludelocal=true&amenableparser=true&format=json' ).then( gresponse => {
  543. var body = gresponse.body;
  544. if ( gresponse.statusCode !== 200 || body?.batchcomplete === undefined || !body?.query?.allmessages ) {
  545. console.log( '- Dashboard: ' + gresponse.statusCode + ': Error while getting the usergroups: ' + body?.error?.info );
  546. return;
  547. }
  548. var groups = body.query.allmessages.filter( group => {
  549. if ( group.name === 'group-all' ) return false;
  550. if ( group.name === 'group-membership-link-with-expiry' ) return false;
  551. if ( group.name.endsWith( '.css' ) || group.name.endsWith( '.js' ) ) return false;
  552. return true;
  553. } ).map( group => {
  554. return {
  555. name: group.name.replace( /^group-/, '' ).replace( /-member$/, '' ),
  556. content: group['*'].replace( / /g, '_' ).toLowerCase()
  557. };
  558. } );
  559. settings.usergroup = settings.usergroup.map( usergroup => {
  560. if ( groups.some( group => group.name === usergroup ) ) return usergroup;
  561. if ( groups.some( group => group.content === usergroup ) ) {
  562. return groups.find( group => group.content === usergroup ).name;
  563. }
  564. if ( /^admins?$/.test(usergroup) ) return 'sysop';
  565. if ( usergroup === '*' ) return 'user';
  566. return usergroup;
  567. } );
  568. }, error => {
  569. console.log( '- Dashboard: Error while getting the usergroups: ' + error );
  570. } ).finally( () => {
  571. if ( settings.usergroup_and ) settings.usergroup.unshift('AND');
  572. var configid = 1;
  573. for ( let i of row.count ) {
  574. if ( configid === i ) configid++;
  575. else break;
  576. }
  577. 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( () => {
  578. console.log( `- Dashboard: Verification successfully added: ${guild}#${configid}` );
  579. res(`/guild/${guild}/verification/${configid}`, 'save');
  580. if ( !row.count.length && slashCommand?.id ) got.put( 'https://discord.com/api/v8/applications/' + process.env.bot + '/guilds/' + guild + '/commands/' + slashCommand.id + '/permissions', {
  581. headers:{
  582. Authorization: 'Bot ' + process.env.token
  583. },
  584. json: {
  585. permissions: [
  586. {
  587. id: guild,
  588. type: 1,
  589. permission: true
  590. }
  591. ]
  592. },
  593. timeout: 10000
  594. } ).then( response=> {
  595. if ( response.statusCode !== 200 || !response.body ) {
  596. console.log( '- Dashboard: ' + response.statusCode + ': Error while enabling the slash command: ' + response.body?.message );
  597. return;
  598. }
  599. console.log( '- Dashboard: Slash command successfully enabled.' );
  600. }, error => {
  601. console.log( '- Dashboard: Error while enabling the slash command: ' + error );
  602. } );
  603. var lang = new Lang(row.lang);
  604. var text = lang.get('verification.dashboard.added', `<@${userSettings.user.id}>`, configid);
  605. text += '\n' + lang.get('verification.channel') + ' <#' + settings.channel.join('>, <#') + '>';
  606. text += '\n' + lang.get('verification.role') + ' <@&' + settings.role.join('>, <@&') + '>';
  607. if ( settings.postcount === null ) {
  608. text += '\n' + lang.get('verification.posteditcount') + ' `' + settings.editcount + '`';
  609. }
  610. else {
  611. text += '\n' + lang.get('verification.editcount') + ' `' + settings.editcount + '`';
  612. text += '\n' + lang.get('verification.postcount') + ' `' + Math.abs(settings.postcount) + '`';
  613. if ( settings.postcount < 0 ) text += ' ' + lang.get('verification.postcount_or');
  614. }
  615. 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') + ' `') ) + '`';
  616. text += '\n' + lang.get('verification.accountage') + ' `' + settings.accountage + '` ' + lang.get('verification.indays');
  617. text += '\n' + lang.get('verification.rename') + ' *`' + lang.get('verification.' + ( settings.rename ? 'enabled' : 'disabled')) + '`*';
  618. text += `\n<${new URL(`/guild/${guild}/verification/${configid}`, process.env.dashboard).href}>`;
  619. if ( settings.rename && !hasPerm(response.botPermissions, 'MANAGE_NICKNAMES') ) {
  620. text += '\n\n' + lang.get('verification.rename_no_permission', `<@${process.env.bot}>`);
  621. }
  622. if ( settings.role.some( role => {
  623. return !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  624. return ( guildRole.id === role && guildRole.lower );
  625. } );
  626. } ) ) {
  627. text += '\n';
  628. settings.role.forEach( role => {
  629. if ( !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  630. return ( guildRole.id === role );
  631. } ) ) {
  632. text += '\n' + lang.get('verification.role_deleted', `<@&${role}>`);
  633. }
  634. else if ( userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  635. return ( guildRole.id === role && !guildRole.lower );
  636. } ) ) {
  637. text += '\n' + lang.get('verification.role_too_high', `<@&${role}>`, `<@${process.env.bot}>`);
  638. }
  639. } );
  640. }
  641. sendMsg( {
  642. type: 'notifyGuild', guild, text
  643. } ).catch( error => {
  644. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  645. } );
  646. }, dberror => {
  647. console.log( '- Dashboard: Error while adding the verification: ' + dberror );
  648. return res(`/guild/${guild}/verification/new`, 'savefail');
  649. } );
  650. } );
  651. }, dberror => {
  652. console.log( '- Dashboard: Error while checking for verifications: ' + dberror );
  653. return res(`/guild/${guild}/verification/new`, 'savefail');
  654. } );
  655. 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]}) => {
  656. if ( !row?.channel ) return res(`/guild/${guild}/verification`, 'savefail');
  657. row.channel = row.channel.split('|').filter( channel => channel.length );
  658. var newChannel = settings.channel.filter( channel => !row.channel.includes( channel ) );
  659. row.role = row.role.split('|');
  660. var newRole = settings.role.filter( role => !row.role.includes( role ) );
  661. row.usergroup = row.usergroup.split('|');
  662. var newUsergroup = settings.usergroup.filter( group => !row.usergroup.includes( group ) );
  663. if ( newChannel.length || newRole.length ) {
  664. let curGuild = userSettings.guilds.isMember.get(guild);
  665. if ( newChannel.some( channel => {
  666. return !curGuild.channels.some( guildChannel => {
  667. return ( guildChannel.id === channel && !guildChannel.isCategory );
  668. } );
  669. } ) || newRole.some( role => {
  670. return !curGuild.roles.some( guildRole => {
  671. return ( guildRole.id === role && guildRole.lower );
  672. } );
  673. } ) ) return res(`/guild/${guild}/verification/${type}`, 'savefail');
  674. }
  675. ( newUsergroup.length ? got.get( row.wiki + 'api.php?action=query&meta=allmessages&amprefix=group-&amincludelocal=true&amenableparser=true&format=json' ).then( gresponse => {
  676. var body = gresponse.body;
  677. if ( gresponse.statusCode !== 200 || body?.batchcomplete === undefined || !body?.query?.allmessages ) {
  678. console.log( '- Dashboard: ' + gresponse.statusCode + ': Error while getting the usergroups: ' + body?.error?.info );
  679. return;
  680. }
  681. var groups = body.query.allmessages.filter( group => {
  682. if ( group.name === 'group-all' ) return false;
  683. if ( group.name === 'group-membership-link-with-expiry' ) return false;
  684. if ( group.name.endsWith( '.css' ) || group.name.endsWith( '.js' ) ) return false;
  685. return true;
  686. } ).map( group => {
  687. return {
  688. name: group.name.replace( /^group-/, '' ).replace( /-member$/, '' ),
  689. content: group['*'].replace( / /g, '_' ).toLowerCase()
  690. };
  691. } );
  692. settings.usergroup = settings.usergroup.map( usergroup => {
  693. if ( groups.some( group => group.name === usergroup ) ) return usergroup;
  694. if ( groups.some( group => group.content === usergroup ) ) {
  695. return groups.find( group => group.content === usergroup ).name;
  696. }
  697. if ( /^admins?$/.test(usergroup) ) return 'sysop';
  698. if ( usergroup === '*' ) return 'user';
  699. return usergroup;
  700. } );
  701. }, error => {
  702. console.log( '- Dashboard: Error while getting the usergroups: ' + error );
  703. } ) : Promise.resolve() ).finally( () => {
  704. if ( settings.usergroup_and ) settings.usergroup.unshift('AND');
  705. var lang = new Lang(row.lang);
  706. var diff = [];
  707. if ( newChannel.length || row.channel.some( channel => {
  708. return !settings.channel.includes( channel );
  709. } ) ) {
  710. diff.push(lang.get('verification.channel') + ` ~~<#${row.channel.join('>, <#')}>~~ → <#${settings.channel.join('>, <#')}>`);
  711. }
  712. if ( newRole.length || row.role.some( role => {
  713. return !settings.role.includes( role );
  714. } ) ) {
  715. diff.push(lang.get('verification.role') + ` ~~<@&${row.role.join('>, <@&')}>~~ → <@&${settings.role.join('>, <@&')}>`);
  716. }
  717. if ( row.postcount !== settings.postcount && ( row.postcount === null || settings.postcount === null ) ) {
  718. if ( row.postcount === null ) {
  719. diff.push('~~' + lang.get('verification.posteditcount') + ` \`${row.editcount}\`~~`);
  720. diff.push('→ ' + lang.get('verification.editcount') + ` \`${settings.editcount}\``);
  721. diff.push('→ ' + lang.get('verification.postcount') + ` \`${Math.abs(settings.postcount)}\`` + ( settings.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ));
  722. }
  723. if ( settings.postcount === null ) {
  724. diff.push('~~' + lang.get('verification.editcount') + ` \`${row.editcount}\`~~`);
  725. diff.push('~~' + lang.get('verification.postcount') + ` \`${Math.abs(row.postcount)}\`` + ( row.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ) + '~~');
  726. diff.push('→ ' + lang.get('verification.posteditcount') + ` \`${settings.editcount}\``);
  727. }
  728. }
  729. else {
  730. if ( row.editcount !== settings.editcount ) {
  731. diff.push(lang.get('verification.editcount') + ` ~~\`${row.editcount}\`~~ → \`${settings.editcount}\``);
  732. }
  733. if ( row.postcount !== settings.postcount ) {
  734. if ( ( row.postcount >= 0 && settings.postcount < 0 ) || ( row.postcount < 0 && settings.postcount >= 0 ) ) {
  735. diff.push('~~' + lang.get('verification.postcount') + ` \`${Math.abs(row.postcount)}\`` + ( row.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ) + '~~');
  736. diff.push('→ ' + lang.get('verification.postcount') + ` \`${Math.abs(settings.postcount)}\`` + ( settings.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ));
  737. }
  738. else diff.push(lang.get('verification.postcount') + ` ~~\`${Math.abs(row.postcount)}\`~~ → \`${Math.abs(settings.postcount)}\`` + ( settings.postcount < 0 ? ' ' + lang.get('verification.postcount_or') : '' ));
  739. }
  740. }
  741. if ( newUsergroup.length || row.usergroup.some( usergroup => {
  742. return !settings.usergroup.includes( usergroup );
  743. } ) ) {
  744. 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') + ' `') ) + '`');
  745. }
  746. if ( row.accountage !== settings.accountage ) {
  747. diff.push(lang.get('verification.accountage') + ` ~~\`${row.accountage}\`~~ → \`${settings.accountage}\``);
  748. }
  749. if ( row.rename !== ( settings.rename ? 1 : 0 ) ) {
  750. diff.push(lang.get('verification.rename') + ` ~~*\`${lang.get('verification.' + ( row.rename ? 'enabled' : 'disabled'))}\`*~~ → *\`${lang.get('verification.' + ( settings.rename ? 'enabled' : 'disabled'))}\`*`);
  751. }
  752. if ( !diff.length ) return res(`/guild/${guild}/verification/${type}`, 'save');
  753. 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( () => {
  754. console.log( `- Dashboard: Verification successfully updated: ${guild}#${type}` );
  755. res(`/guild/${guild}/verification/${type}`, 'save');
  756. var text = lang.get('verification.dashboard.updated', `<@${userSettings.user.id}>`, type);
  757. text += '\n' + diff.join('\n');
  758. text += `\n<${new URL(`/guild/${guild}/verification/${type}`, process.env.dashboard).href}>`;
  759. if ( settings.rename && !hasPerm(response.botPermissions, 'MANAGE_NICKNAMES') ) {
  760. text += '\n\n' + lang.get('verification.rename_no_permission', `<@${process.env.bot}>`);
  761. }
  762. if ( settings.role.some( role => {
  763. return !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  764. return ( guildRole.id === role && guildRole.lower );
  765. } );
  766. } ) ) {
  767. text += '\n';
  768. settings.role.forEach( role => {
  769. if ( !userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  770. return ( guildRole.id === role );
  771. } ) ) {
  772. text += '\n' + lang.get('verification.role_deleted', `<@&${role}>`);
  773. }
  774. else if ( userSettings.guilds.isMember.get(guild).roles.some( guildRole => {
  775. return ( guildRole.id === role && !guildRole.lower );
  776. } ) ) {
  777. text += '\n' + lang.get('verification.role_too_high', `<@&${role}>`, `<@${process.env.bot}>`);
  778. }
  779. } );
  780. }
  781. sendMsg( {
  782. type: 'notifyGuild', guild, text
  783. } ).catch( error => {
  784. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  785. } );
  786. }, dberror => {
  787. console.log( '- Dashboard: Error while updating the verification: ' + dberror );
  788. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  789. } );
  790. } );
  791. }, dberror => {
  792. console.log( '- Dashboard: Error while checking for verifications: ' + dberror );
  793. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  794. } );
  795. }, error => {
  796. console.log( '- Dashboard: Error while getting the member: ' + error );
  797. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  798. } );
  799. }
  800. /**
  801. * Change verification notices
  802. * @param {Function} res - The server response
  803. * @param {import('./util.js').Settings} userSettings - The settings of the user
  804. * @param {String} guild - The id of the guild
  805. * @param {String} type - The setting to change
  806. * @param {Object} settings - The new settings
  807. * @param {String} [settings.channel]
  808. * @param {String} [settings.success]
  809. * @param {String} [settings.match]
  810. * @param {String} settings.save_settings
  811. */
  812. function update_notices(res, userSettings, guild, type, settings) {
  813. if ( !settings.save_settings ) {
  814. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  815. }
  816. if ( settings.channel && !/^\d+$/.test(settings.channel) ) {
  817. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  818. }
  819. if ( settings.success && settings.success.trim().length > 1000 ) {
  820. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  821. }
  822. if ( settings.match && settings.match.trim().length > 1000 ) {
  823. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  824. }
  825. settings.channel = ( settings.channel || null );
  826. settings.success = ( settings.success?.trim().replace( /`ˋ`/g, '```' ) || null );
  827. settings.match = ( settings.match?.trim().replace( /`ˋ`/g, '```' ) || null );
  828. sendMsg( {
  829. type: 'getMember',
  830. member: userSettings.user.id,
  831. guild: guild,
  832. newchannel: settings.channel
  833. } ).then( response => {
  834. if ( !response ) {
  835. userSettings.guilds.notMember.set(guild, userSettings.guilds.isMember.get(guild));
  836. userSettings.guilds.isMember.delete(guild);
  837. return res(`/guild/${guild}`, 'savefail');
  838. }
  839. if ( response === 'noMember' || !hasPerm(response.userPermissions, 'MANAGE_GUILD') ) {
  840. userSettings.guilds.isMember.delete(guild);
  841. return res('/', 'savefail');
  842. }
  843. if ( settings.channel && ( response.message === 'noChannel' || !( hasPerm(response.botPermissionsNew, 'VIEW_CHANNEL', 'SEND_MESSAGES') && hasPerm(response.userPermissions, 'VIEW_CHANNEL') ) ) ) {
  844. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  845. }
  846. return db.connect().then( client => {
  847. return client.query( 'SELECT logchannel, onsuccess, onmatch FROM verifynotice WHERE guild = $1', [guild] ).then( ({rows:[row]}) => {
  848. if ( !row ) {
  849. if ( !( settings.channel || settings.success || settings.match ) ) {
  850. return res(`/guild/${guild}/verification/${type}`, 'save');
  851. }
  852. return client.query( 'INSERT INTO verifynotice(guild, logchannel, onsuccess, onmatch) VALUES($1, $2, $3, $4)', [guild, settings.channel, settings.success, settings.match] ).then( () => {
  853. console.log( `- Dashboard: Verification notices successfully added: ${guild}` );
  854. res(`/guild/${guild}/verification/${type}`, 'save');
  855. return client.query( 'SELECT lang FROM discord WHERE guild = $1 AND channel IS NULL', [guild] ).then( ({rows:[channel]}) => {
  856. var lang = new Lang(channel?.lang);
  857. var text = lang.get('verification.dashboard.added_notice', `<@${userSettings.user.id}>`) + '\n';
  858. if ( settings.channel ) text += `${lang.get('verification.logging')} <#${settings.channel}>\n`;
  859. if ( settings.success ) text += `${lang.get('verification.success')} \`\`\`md\n${settings.success.replace( /```/g, '`ˋ`' )}\n\`\`\``;
  860. if ( settings.match ) text += `${lang.get('verification.match')} \`\`\`md\n${settings.match.replace( /```/g, '`ˋ`' )}\n\`\`\``;
  861. text += `<${new URL(`/guild/${guild}/verification/${type}`, process.env.dashboard).href}>`;
  862. if ( settings.success?.includes( '](' ) || settings.match?.includes( '](' ) ) {
  863. text += '\n\n' + lang.get('verification.notice_embed');
  864. }
  865. sendMsg( {
  866. type: 'notifyGuild', guild, text
  867. } ).catch( error => {
  868. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  869. } );
  870. }, dberror => {
  871. console.log( '- Dashboard: Error while notifying the guild: ' + dberror );
  872. } );
  873. }, dberror => {
  874. console.log( '- Dashboard: Error while adding the verification notices: ' + dberror );
  875. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  876. } );
  877. }
  878. if ( settings.channel === row.logchannel && settings.success === row.onsuccess && settings.match === row.onmatch ) {
  879. return res(`/guild/${guild}/verification/${type}`, 'save');
  880. }
  881. return client.query( 'UPDATE verifynotice SET logchannel = $1, onsuccess = $2, onmatch = $3 WHERE guild = $4', [settings.channel, settings.success, settings.match, guild] ).then( () => {
  882. console.log( `- Dashboard: Verification notices successfully updated: ${guild}` );
  883. res(`/guild/${guild}/verification/${type}`, 'save');
  884. return client.query( 'SELECT lang FROM discord WHERE guild = $1 AND channel IS NULL', [guild] ).then( ({rows:[channel]}) => {
  885. var lang = new Lang(channel?.lang);
  886. var text = lang.get('verification.dashboard.updated_notice', `<@${userSettings.user.id}>`) + '\n';
  887. if ( settings.channel !== row.logchannel ) {
  888. text += lang.get('verification.logging') + ' ~~' + ( row.logchannel ? `<#${row.logchannel}>` : `*\`${lang.get('verification.disabled')}\`*` ) + '~~ → ' + ( settings.channel ? `<#${settings.channel}>` : `*\`${lang.get('verification.disabled')}\`*` ) + '\n';
  889. }
  890. if ( settings.success !== row.onsuccess ) {
  891. text += lang.get('verification.success') + ' ' + ( row.onsuccess ? '~~```md\n' + row.onsuccess.replace( /\\/g, '\\$&' ).replace( /```/g, '`ˋ`' ) + '\n```~~' : `~~*\`${lang.get('verification.disabled')}\`*~~ → ` ) + ( settings.success ? '```md\n' + settings.success.replace( /\\/g, '\\$&' ).replace( /```/g, '`ˋ`' ) + '\n```' : ` → *\`${lang.get('verification.disabled')}\`*\n` );
  892. }
  893. if ( settings.match !== row.onmatch ) {
  894. text += lang.get('verification.match') + ' ' + ( row.onmatch ? '~~```md\n' + row.onmatch.replace( /\\/g, '\\$&' ).replace( /```/g, '`ˋ`' ) + '\n```~~' : `~~*\`${lang.get('verification.disabled')}\`*~~ → ` ) + ( settings.match ? '```md\n' + settings.match.replace( /\\/g, '\\$&' ).replace( /```/g, '`ˋ`' ) + '\n```' : ` → *\`${lang.get('verification.disabled')}\`*\n` );
  895. }
  896. text += `<${new URL(`/guild/${guild}/verification/${type}`, process.env.dashboard).href}>`;
  897. if ( settings.success?.includes( '](' ) || settings.match?.includes( '](' ) ) {
  898. text += '\n\n' + lang.get('verification.notice_embed');
  899. }
  900. sendMsg( {
  901. type: 'notifyGuild', guild, text
  902. } ).catch( error => {
  903. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  904. } );
  905. }, dberror => {
  906. console.log( '- Dashboard: Error while notifying the guild: ' + dberror );
  907. } );
  908. }, dberror => {
  909. console.log( '- Dashboard: Error while updating the verification notices: ' + dberror );
  910. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  911. } );
  912. }, dberror => {
  913. console.log( '- Dashboard: Error while getting the current verification notices: ' + dberror );
  914. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  915. } ).finally( () => {
  916. client.release();
  917. } );
  918. }, dberror => {
  919. console.log( '- Error while connecting to the database client: ' + dberror );
  920. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  921. } );
  922. }, error => {
  923. console.log( '- Dashboard: Error while getting the member: ' + error );
  924. return res(`/guild/${guild}/verification/${type}`, 'savefail');
  925. } );
  926. }
  927. module.exports = {
  928. get: dashboard_verification,
  929. post: update_verification
  930. };