slash.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. import Lang from '../util/i18n.js';
  2. import { got, db, slashCommands, sendMsg, createNotice, hasPerm } from './util.js';
  3. const fieldset = {
  4. role: '<label for="wb-settings-addrole">Role:</label>'
  5. + '<select id="wb-settings-addrole" name="role"></select>'
  6. + '<button type="button" id="wb-settings-addrole-add">Add</button>'
  7. + '<hr>',
  8. permission: '<span title="@UNKNOWN">@UNKNOWN:</span>'
  9. + '<div class="wb-settings-permission">'
  10. + '<input type="radio" id="wb-settings-permission-0" name="permission" value="0" required>'
  11. + '<label for="wb-settings-permission-0" class="wb-settings-permission-deny radio-label">Deny</label>'
  12. + '</div><div class="wb-settings-permission">'
  13. + '<input type="radio" id="wb-settings-permission-1" name="permission" value="1" required>'
  14. + '<label for="wb-settings-permission-1" class="wb-settings-permission-allow radio-label">Allow</label>'
  15. + '</div><div class="wb-settings-permission">'
  16. + '<input type="radio" id="wb-settings-permission-default" name="permission" value="" required>'
  17. + '<label for="wb-settings-permission-default" class="wb-settings-permission-default radio-label">Default</label>'
  18. + '</div>',
  19. save: '<input type="submit" id="wb-settings-save" name="save_settings">'
  20. };
  21. /**
  22. * Create a settings form
  23. * @param {import('cheerio').default} $ - The response body
  24. * @param {slashCommands[0]} slashCommand - The slash command
  25. * @param {import('./i18n.js').default} dashboardLang - The user language
  26. * @param {Object[]} permissions - The current permissions
  27. * @param {String} permissions.id
  28. * @param {Number} permissions.type
  29. * @param {Boolean} permissions.permission
  30. * @param {String} guildId - The guild id
  31. * @param {import('./util.js').Role[]} guildRoles - The guild roles
  32. */
  33. function createForm($, slashCommand, dashboardLang, permissions, guildId, guildRoles) {
  34. var readonly = ( process.env.READONLY ? true : false );
  35. var fields = [];
  36. if ( !readonly ) {
  37. $('<script>').text(`const i18nSlashPermission = ${JSON.stringify({
  38. allow: dashboardLang.get('slash.form.allow'),
  39. deny: dashboardLang.get('slash.form.deny'),
  40. default: dashboardLang.get('slash.form.default')
  41. })};`).insertBefore('script#indexjs');
  42. let role = $('<div>').append(fieldset.role);
  43. role.find('label').text(dashboardLang.get('slash.form.role'));
  44. role.find('#wb-settings-addrole').append(
  45. $(`<option id="wb-settings-channel-default" selected hidden>`).val('').text(dashboardLang.get('slash.form.select_role')),
  46. ...guildRoles.filter( guildRole => !permissions.some( perm => perm.id === guildRole.id ) ).map( guildRole => {
  47. return $(`<option id="wb-settings-addrole-${guildRole.id}">`).val(guildRole.id).text(`${guildRole.id} – @${guildRole.name}`)
  48. } ),
  49. ( permissions.some( perm => perm.id === guildId ) ? '' : $(`<option id="wb-settings-addrole-${guildId}">`).val(guildId).text(`@everyone`) )
  50. );
  51. role.find('#wb-settings-addrole-add').text(dashboardLang.get('slash.form.add'));
  52. fields.push(role);
  53. }
  54. let perms = permissions.sort( (a, b) => {
  55. if ( a.id === guildId ) return 1;
  56. if ( b.id === guildId ) return -1;
  57. return guildRoles.findIndex( guildRole => guildRole.id === a.id ) - guildRoles.findIndex( guildRole => guildRole.id === b.id );
  58. } ).map( perm => {
  59. let permission = $('<div>').append(fieldset.permission);
  60. let span = permission.find('span').attr('title', perm.id);
  61. if ( perm.id === guildId ) span.text('@everyone').attr('title', '@everyone');
  62. else span.text(`@${( guildRoles.find( guildRole => guildRole.id === perm.id )?.name || 'UNKNOWN' )}`);
  63. permission.find('input[name="permission"]').attr('name', `permission-${perm.id}`);
  64. permission.find('input#wb-settings-permission-0').attr('id', `wb-settings-permission-${perm.id}-0`);
  65. permission.find('label[for="wb-settings-permission-0"]').attr('for', `wb-settings-permission-${perm.id}-0`);
  66. permission.find('input#wb-settings-permission-1').attr('id', `wb-settings-permission-${perm.id}-1`);
  67. permission.find('label[for="wb-settings-permission-1"]').attr('for', `wb-settings-permission-${perm.id}-1`);
  68. permission.find('input#wb-settings-permission-default').attr('id', `wb-settings-permission-${perm.id}-default`);
  69. permission.find('label[for="wb-settings-permission-default"]').attr('for', `wb-settings-permission-${perm.id}-default`);
  70. permission.find(`#wb-settings-permission-${perm.id}-${( perm.permission ? '1' : '0' )}`).attr('checked', '');
  71. return permission;
  72. } );
  73. fields.push(...perms);
  74. fields.push($(fieldset.save).val(dashboardLang.get('general.save')));
  75. var form = $('<fieldset>').append(
  76. $('<legend>').text(( slashCommand.default_permission ? dashboardLang.get('slash.form.default_allow') : dashboardLang.get('slash.form.default_deny') )),
  77. ...fields);
  78. if ( readonly ) {
  79. form.find('input').attr('readonly', '');
  80. form.find('input[type="radio"]:not(:checked), option').attr('disabled', '');
  81. form.find('input[type="submit"]').remove();
  82. }
  83. form.find('label.wb-settings-permission-deny').text(dashboardLang.get('slash.form.deny'));
  84. form.find('label.wb-settings-permission-allow').text(dashboardLang.get('slash.form.allow'));
  85. form.find('label.wb-settings-permission-default').text(dashboardLang.get('slash.form.default'));
  86. return $('<form id="wb-settings" method="post" enctype="application/x-www-form-urlencoded">').append(
  87. $('<h2>').html(dashboardLang.get('slash.form.entry', true, $('<code>').text('/' + slashCommand.name))),
  88. form
  89. );
  90. }
  91. /**
  92. * Let a user change slashs
  93. * @param {import('http').ServerResponse} res - The server response
  94. * @param {import('cheerio').default} $ - The response body
  95. * @param {import('./util.js').Guild} guild - The current guild
  96. * @param {String[]} args - The url parts
  97. * @param {import('./i18n.js').default} dashboardLang - The user language
  98. */
  99. function dashboard_slash(res, $, guild, args, dashboardLang) {
  100. let suffix = ( args[0] === 'owner' ? '?owner=true' : '' );
  101. $('#channellist #slash').after(
  102. ...slashCommands.filter( slashCommand => slashCommand.id ).map( slashCommand => {
  103. return $('<a class="channel">').attr('id', `channel-${slashCommand.id}`).append(
  104. $('<img>').attr('src', '/src/channel.svg'),
  105. $('<div>').text(slashCommand.name)
  106. ).attr('title', slashCommand.name).attr('href', `/guild/${guild.id}/slash/${slashCommand.id}${suffix}`);
  107. } )
  108. );
  109. if ( args[4] ) {
  110. let slashCommand = slashCommands.find( slashCommand => args[4] === slashCommand.id );
  111. if ( slashCommand ) return got.get( `https://discord.com/api/v8/applications/${process.env.bot}/guilds/${guild.id}/commands/${slashCommand.id}/permissions`, {
  112. headers: {
  113. Authorization: `Bot ${process.env.token}`
  114. },
  115. timeout: {
  116. request: 10000
  117. }
  118. } ).then( response=> {
  119. var permissions = [];
  120. if ( response.statusCode !== 200 || !response.body ) {
  121. if ( response.statusCode !== 404 || response.body?.message !== 'Unknown application command permissions' ) {
  122. if ( response.statusCode === 403 && response.body?.message === 'Missing Access' ) {
  123. createNotice($, 'noslash', dashboardLang, [guild.id]);
  124. }
  125. else {
  126. console.log( '- Dashboard: ' + response.statusCode + ': Error while getting the slash command permissions: ' + response.body?.message );
  127. createNotice($, 'error', dashboardLang);
  128. }
  129. $('#text .description').html(dashboardLang.get('slash.explanation'));
  130. $('.channel#slash').addClass('selected');
  131. return;
  132. }
  133. else if ( slashCommand.name === 'verify' ) return db.query( 'SELECT 1 FROM verification WHERE guild = $1 LIMIT 1', [guild.id] ).then( ({rows}) => {
  134. if ( rows.length ) {
  135. $('<p>').html(dashboardLang.get('slash.desc', true, $('<code>').text(guild.name))).appendTo('#text .description');
  136. $(`.channel#channel-${slashCommand.id}`).addClass('selected');
  137. createForm($, slashCommand, dashboardLang, permissions, guild.id, guild.roles).attr('action', `/guild/${guild.id}/slash/${slashCommand.id}`).appendTo('#text');
  138. return;
  139. }
  140. res.writeHead(302, {Location: `/guild/${guild.id}/verification/new${suffix}` + ( suffix ? '&' : '?' ) + 'slash=noverify'});
  141. res.end();
  142. return true;
  143. }, dberror => {
  144. console.log( '- Dashboard: Error while checking for verifications: ' + dberror );
  145. res.writeHead(302, {Location: `/guild/${guild.id}/verification/new${suffix}` + ( suffix ? '&' : '?' ) + 'slash=noverify'});
  146. res.end();
  147. return true;
  148. } );
  149. }
  150. else permissions = response.body.permissions;
  151. $('<p>').html(dashboardLang.get('slash.desc', true, $('<code>').text(guild.name))).appendTo('#text .description');
  152. $(`.channel#channel-${slashCommand.id}`).addClass('selected');
  153. createForm($, slashCommand, dashboardLang, permissions, guild.id, guild.roles).attr('action', `/guild/${guild.id}/slash/${slashCommand.id}`).appendTo('#text');
  154. }, error => {
  155. console.log( '- Dashboard: Error while getting the slash command permissions: ' + error );
  156. createNotice($, 'error', dashboardLang);
  157. $('#text .description').html(dashboardLang.get('slash.explanation'));
  158. $('.channel#slash').addClass('selected');
  159. } ).then( isRedirected => {
  160. if ( isRedirected ) return;
  161. let body = $.html();
  162. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  163. res.write( body );
  164. return res.end();
  165. } );
  166. }
  167. $('#text .description').html(dashboardLang.get('slash.explanation'));
  168. $('.channel#slash').addClass('selected');
  169. let body = $.html();
  170. res.writeHead(200, {'Content-Length': Buffer.byteLength(body)});
  171. res.write( body );
  172. return res.end();
  173. }
  174. /**
  175. * Change slashs
  176. * @param {Function} res - The server response
  177. * @param {import('./util.js').Settings} userSettings - The settings of the user
  178. * @param {String} guild - The id of the guild
  179. * @param {String} type - The setting to change
  180. * @param {Object} settings - The new settings
  181. */
  182. function update_slash(res, userSettings, guild, type, settings) {
  183. if ( !slashCommands.some( slashCommand => slashCommand.id === type ) ) {
  184. return res(`/guild/${guild}/slash`, 'savefail');
  185. }
  186. if ( !settings.save_settings ) {
  187. return res(`/guild/${guild}/slash/${type}`, 'savefail');
  188. }
  189. let roles = userSettings.guilds.isMember.get(guild).roles;
  190. var permissions = Object.keys(settings).filter( perm => roles.some( role => ( 'permission-' + role.id === perm ) ) || perm === 'permission-' + guild ).map( perm => {
  191. return {
  192. id: perm.replace( 'permission-', '' ), type: 1,
  193. permission: ( settings[perm] === '1' ? true : false )
  194. };
  195. } );
  196. sendMsg( {
  197. type: 'getMember',
  198. member: userSettings.user.id,
  199. guild: guild
  200. } ).then( response => {
  201. if ( !response ) {
  202. userSettings.guilds.notMember.set(guild, userSettings.guilds.isMember.get(guild));
  203. userSettings.guilds.isMember.delete(guild);
  204. return res(`/guild/${guild}`, 'savefail');
  205. }
  206. if ( response === 'noMember' || !hasPerm(response.userPermissions, 'MANAGE_GUILD') ) {
  207. userSettings.guilds.isMember.delete(guild);
  208. return res('/', 'savefail');
  209. }
  210. var commandName = slashCommands.find( slashCommand => slashCommand.id === type )?.name;
  211. return got.get( `https://discord.com/api/v8/applications/${process.env.bot}/guilds/${guild}/commands/${type}/permissions`, {
  212. headers: {
  213. Authorization: `Bot ${process.env.token}`
  214. },
  215. timeout: {
  216. request: 10000
  217. }
  218. } ).then( response=> {
  219. if ( response.statusCode !== 200 || !response.body ) {
  220. if ( response.statusCode === 403 && response.body?.message === 'Missing Access' ) {
  221. res(`/guild/${guild}/slash/${type}`, 'noslash', guild);
  222. return Promise.reject();
  223. }
  224. else if ( response.statusCode !== 404 || response.body?.message !== 'Unknown application command permissions' ) {
  225. console.log( '- Dashboard: ' + response.statusCode + ': Error while getting the old slash command permissions: ' + response.body?.message );
  226. }
  227. else if ( commandName === 'verify' ) return db.query( 'SELECT 1 FROM verification WHERE guild = $1 LIMIT 1', [guild] ).then( ({rows}) => {
  228. if ( rows.length ) return [];
  229. res(`/guild/${guild}/verification/new`, 'noverify');
  230. return Promise.reject();
  231. }, dberror => {
  232. console.log( '- Dashboard: Error while checking for verifications: ' + dberror );
  233. res(`/guild/${guild}/verification/new`, 'noverify');
  234. return Promise.reject();
  235. } );
  236. return [];
  237. }
  238. return response.body.permissions;
  239. }, error => {
  240. console.log( '- Dashboard: Error while getting the old slash command permissions: ' + error );
  241. return [];
  242. } ).then( oldPermissions => {
  243. return got.put( `https://discord.com/api/v8/applications/${process.env.bot}/guilds/${guild}/commands/${type}/permissions`, {
  244. headers: {
  245. Authorization: `Bot ${process.env.token}`
  246. },
  247. json: {permissions},
  248. timeout: {
  249. request: 10000
  250. }
  251. } ).then( response=> {
  252. if ( response.statusCode !== 200 || !response.body ) {
  253. console.log( '- Dashboard: ' + response.statusCode + ': Error while saving the slash command permissions: ' + response.body?.message );
  254. return res(`/guild/${guild}/slash/${type}`, 'savefail');
  255. }
  256. res(`/guild/${guild}/slash/${type}`, 'save');
  257. var changes = [
  258. ...permissions.map( perm => {
  259. var oldPerm = oldPermissions.find( oldPerm => oldPerm.id === perm.id );
  260. if ( !oldPerm ) return {
  261. role: ( perm.id === guild ? '@everyone' : `<@&${perm.id}>` ),
  262. old: 'default',
  263. new: ( perm.permission ? 'allow' : 'deny' )
  264. };
  265. if ( perm.permission === oldPerm.permission ) return null;
  266. return {
  267. role: ( perm.id === guild ? '@everyone' : `<@&${perm.id}>` ),
  268. old: ( oldPerm.permission ? 'allow' : 'deny' ),
  269. new: ( perm.permission ? 'allow' : 'deny' )
  270. };
  271. } ).filter( change => change ),
  272. ...oldPermissions.filter( oldPerm => !permissions.some( perm => perm.id === oldPerm.id ) ).map( oldPerm => {
  273. return {
  274. role: ( oldPerm.id === guild ? '@everyone' : `<@&${oldPerm.id}>` ),
  275. old: ( oldPerm.permission ? 'allow' : 'deny' ),
  276. new: 'default'
  277. };
  278. } )
  279. ];
  280. if ( !changes.length ) return;
  281. return db.query( 'SELECT lang FROM discord WHERE guild = $1 AND channel IS NULL', [guild] ).then( ({rows:[channel]}) => {
  282. var lang = new Lang(channel?.lang);
  283. var text = lang.get('interaction.dashboard.updated', `<@${userSettings.user.id}>`, '/' + commandName);
  284. text += '\n' + changes.map( change => {
  285. return change.role + ': ~~`' + lang.get('interaction.dashboard.perm_' + change.old) + '`~~ → `' + lang.get('interaction.dashboard.perm_' + change.new) + '`';
  286. } ).join('\n');
  287. text += `\n<${new URL(`/guild/${guild}/slash/${type}`, process.env.dashboard).href}>`;
  288. sendMsg( {
  289. type: 'notifyGuild', guild, text
  290. } ).catch( error => {
  291. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  292. } );
  293. }, dberror => {
  294. console.log( '- Dashboard: Error while notifying the guild: ' + dberror );
  295. } );
  296. }, error => {
  297. console.log( '- Dashboard: Error while saving the slash command permissions: ' + error );
  298. return res(`/guild/${guild}/slash/${type}`, 'savefail');
  299. } );
  300. }, error => {
  301. if ( error ) {
  302. console.log( '- Dashboard: Error while getting the old slash command permissions: ' + error );
  303. return res(`/guild/${guild}/slash/${type}`, 'savefail');
  304. }
  305. } );
  306. }, error => {
  307. console.log( '- Dashboard: Error while getting the member: ' + error );
  308. return res(`/guild/${guild}/slash/${type}`, 'savefail');
  309. } );
  310. }
  311. export {
  312. dashboard_slash as get,
  313. update_slash as post
  314. };