settings.js 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554
  1. const cheerio = require('cheerio');
  2. const {defaultSettings} = require('../util/default.json');
  3. const Lang = require('../util/i18n.js');
  4. const allLangs = Lang.allLangs();
  5. const Wiki = require('../util/wiki.js');
  6. const {got, db, sendMsg, hasPerm} = require('./util.js');
  7. const fieldset = {
  8. channel: '<label for="wb-settings-channel">Channel:</label>'
  9. + '<select id="wb-settings-channel" name="channel" required></select>',
  10. wiki: '<label for="wb-settings-wiki">Default Wiki:</label>'
  11. + '<input type="url" id="wb-settings-wiki" name="wiki" required autocomplete="url">',
  12. //+ '<button type="button" id="wb-settings-wiki-search" class="collapsible">Search wiki</button>'
  13. //+ '<fieldset style="display: none;">'
  14. //+ '<legend>Wiki search</legend>'
  15. //+ '</fieldset>',
  16. lang: '<label for="wb-settings-lang">Language:</label>'
  17. + '<select id="wb-settings-lang" name="lang" required autocomplete="language">'
  18. + Object.keys(allLangs.names).map( lang => {
  19. return `<option id="wb-settings-lang-${lang}" value="${lang}">${allLangs.names[lang]}</option>`
  20. } ).join('\n')
  21. + '</select>',
  22. role: '<label for="wb-settings-role">Minimal Role:</label>'
  23. + '<select id="wb-settings-role" name="role"></select>',
  24. prefix: '<label for="wb-settings-prefix">Prefix:</label>'
  25. + '<input type="text" id="wb-settings-prefix" name="prefix" pattern="^\\s*[^\\s`\\\\]{1,100}\\s*$" minlength="1" maxlength="100" required autocomplete="on">'
  26. + '<br>'
  27. + '<label for="wb-settings-prefix-space">Prefix ends with space:</label>'
  28. + '<input type="checkbox" id="wb-settings-prefix-space" name="prefix_space">',
  29. inline: '<label for="wb-settings-inline">Inline commands:</label>'
  30. + '<input type="checkbox" id="wb-settings-inline" name="inline">',
  31. save: '<input type="submit" id="wb-settings-save" name="save_settings">',
  32. delete: '<input type="submit" id="wb-settings-delete" name="delete_settings">'
  33. };
  34. /**
  35. * Create a settings form
  36. * @param {import('cheerio')} $ - The response body
  37. * @param {String} header - The form header
  38. * @param {Object} settings - The current settings
  39. * @param {Boolean} settings.patreon
  40. * @param {String} settings.channel
  41. * @param {String} settings.wiki
  42. * @param {String} settings.lang
  43. * @param {String} settings.role
  44. * @param {Boolean} settings.inline
  45. * @param {String} settings.prefix
  46. * @param {Object[]} guildRoles - The guild roles
  47. * @param {String} guildRoles.id
  48. * @param {String} guildRoles.name
  49. * @param {Object[]} guildChannels - The guild channels
  50. * @param {String} guildChannels.id
  51. * @param {String} guildChannels.name
  52. * @param {Number} guildChannels.userPermissions
  53. */
  54. function createForm($, header, settings, guildRoles, guildChannels = []) {
  55. var readonly = ( process.env.READONLY ? true : false );
  56. if ( settings.channel && guildChannels.userPermissions === 0 && guildChannels.name === 'UNKNOWN' ) {
  57. readonly = true;
  58. }
  59. var fields = [];
  60. if ( settings.channel ) {
  61. let channel = $('<div>').append(fieldset.channel);
  62. channel.find('#wb-settings-channel').append(
  63. ...guildChannels.map( guildChannel => {
  64. return $(`<option id="wb-settings-channel-${guildChannel.id}">`).val(guildChannel.id).text(`${guildChannel.id} – #${guildChannel.name}`)
  65. } )
  66. );
  67. if ( guildChannels.length === 1 ) {
  68. channel.find(`#wb-settings-channel-${settings.channel}`).attr('selected', '');
  69. if ( !hasPerm(guildChannels[0].userPermissions, 'VIEW_CHANNEL', 'SEND_MESSAGES') ) {
  70. readonly = true;
  71. }
  72. }
  73. else channel.find('#wb-settings-channel').prepend(
  74. $(`<option id="wb-settings-channel-default" selected hidden>`).val('').text('-- Select a Channel --')
  75. );
  76. fields.push(channel);
  77. }
  78. let wiki = $('<div>').append(fieldset.wiki);
  79. wiki.find('#wb-settings-wiki').val(settings.wiki);
  80. fields.push(wiki);
  81. if ( !settings.channel || settings.patreon ) {
  82. let lang = $('<div>').append(fieldset.lang);
  83. lang.find(`#wb-settings-lang-${settings.lang}`).attr('selected', '');
  84. fields.push(lang);
  85. let role = $('<div>').append(fieldset.role);
  86. role.find('#wb-settings-role').append(
  87. $(`<option id="wb-settings-role-default">`).val('').text(`@everyone`),
  88. ...guildRoles.map( guildRole => {
  89. return $(`<option id="wb-settings-role-${guildRole.id}">`).val(guildRole.id).text(`${guildRole.id} – @${guildRole.name}`)
  90. } ),
  91. $(`<option id="wb-settings-role-everyone">`).val('').text(`@everyone`)
  92. );
  93. if ( settings.role ) role.find(`#wb-settings-role-${settings.role}`).attr('selected', '');
  94. else role.find(`#wb-settings-role-everyone`).attr('selected', '');
  95. fields.push(role);
  96. let inline = $('<div>').append(fieldset.inline);
  97. if ( !settings.inline ) inline.find('#wb-settings-inline').attr('checked', '');
  98. fields.push(inline);
  99. }
  100. if ( settings.patreon && !settings.channel ) {
  101. let prefix = $('<div>').append(fieldset.prefix);
  102. prefix.find('#wb-settings-prefix').val(settings.prefix.trim());
  103. if ( settings.prefix.endsWith( ' ' ) ) {
  104. prefix.find('#wb-settings-prefix-space').attr('checked', '');
  105. }
  106. fields.push(prefix);
  107. }
  108. fields.push($(fieldset.save).val('Save'));
  109. if ( settings.channel && settings.channel !== 'new' ) {
  110. fields.push($(fieldset.delete).val('Delete').attr('onclick', `return confirm('Are you sure?');`));
  111. }
  112. var form = $('<fieldset>').append(...fields);
  113. if ( readonly ) {
  114. form.find('input').attr('readonly', '');
  115. form.find('input[type="checkbox"], option').attr('disabled', '');
  116. form.find('input[type="submit"]').remove();
  117. }
  118. return $('<form id="wb-settings" method="post" enctype="application/x-www-form-urlencoded">').append(
  119. $('<h2>').text(header),
  120. form
  121. );
  122. }
  123. /**
  124. * Let a user change settings
  125. * @param {import('http').ServerResponse} res - The server response
  126. * @param {import('cheerio')} $ - The response body
  127. * @param {import('./util.js').Guild} guild - The current guild
  128. * @param {String[]} args - The url parts
  129. */
  130. function dashboard_settings(res, $, guild, args) {
  131. db.all( 'SELECT channel, wiki, lang, role, inline, prefix, patreon FROM discord WHERE guild = ? ORDER BY channel ASC', [guild.id], function(dberror, rows) {
  132. if ( dberror ) {
  133. console.log( '- Dashboard: Error while getting the settings: ' + dberror );
  134. $('#text .description').text('Failed to load the settings!');
  135. $('.channel#settings').addClass('selected');
  136. let body = $.html();
  137. res.writeHead(200, {'Content-Length': body.length});
  138. res.write( body );
  139. return res.end();
  140. }
  141. $('#text .description').text(`These are the settings for "${guild.name}":`);
  142. if ( !rows.length ) {
  143. $('.channel#settings').addClass('selected');
  144. createForm($, 'Server-wide Settings', Object.assign({
  145. prefix: process.env.prefix
  146. }, defaultSettings), guild.roles).attr('action', `/guild/${guild.id}/settings/default`).appendTo('#text');
  147. let body = $.html();
  148. res.writeHead(200, {'Content-Length': body.length});
  149. res.write( body );
  150. return res.end();
  151. }
  152. let isPatreon = rows.some( row => row.patreon );
  153. let channellist = rows.filter( row => row.channel ).map( row => {
  154. let channel = guild.channels.find( channel => channel.id === row.channel );
  155. return ( channel || {id: row.channel, name: 'UNKNOWN', userPermissions: 0} );
  156. } ).sort( (a, b) => {
  157. return guild.channels.indexOf(a) - guild.channels.indexOf(b);
  158. } );
  159. $('#channellist #settings').after(
  160. ...channellist.map( channel => {
  161. return $('<a class="channel">').attr('id', `channel-${channel.id}`).append(
  162. $('<img>').attr('src', '/src/channel.svg'),
  163. $('<div>').text(channel.name)
  164. ).attr('href', `/guild/${guild.id}/settings/${channel.id}`).attr('title', channel.id);
  165. } ),
  166. ( process.env.READONLY || !guild.channels.filter( channel => {
  167. return ( hasPerm(channel.userPermissions, 'VIEW_CHANNEL', 'SEND_MESSAGES') && !rows.some( row => row.channel === channel.id ) );
  168. } ).length ? '' :
  169. $('<a class="channel" id="channel-new">').append(
  170. $('<img>').attr('src', '/src/channel.svg'),
  171. $('<div>').text('New channel overwrite')
  172. ).attr('href', `/guild/${guild.id}/settings/new`) )
  173. );
  174. if ( args[4] === 'new' && !process.env.READONLY ) {
  175. $('.channel#channel-new').addClass('selected');
  176. createForm($, 'New Channel Overwrite', Object.assign({}, rows.find( row => !row.channel ), {
  177. patreon: isPatreon,
  178. channel: 'new'
  179. }), guild.roles, guild.channels.filter( channel => {
  180. return ( hasPerm(channel.userPermissions, 'VIEW_CHANNEL', 'SEND_MESSAGES') && !rows.some( row => row.channel === channel.id ) );
  181. } )).attr('action', `/guild/${guild.id}/settings/new`).appendTo('#text');
  182. }
  183. else if ( channellist.some( channel => channel.id === args[4] ) ) {
  184. let channel = channellist.find( channel => channel.id === args[4] );
  185. $(`.channel#channel-${channel.id}`).addClass('selected');
  186. createForm($, `#${channel.name} Settings`, Object.assign({}, rows.find( row => {
  187. return row.channel === channel.id;
  188. } ), {
  189. patreon: isPatreon
  190. }), guild.roles, [channel]).attr('action', `/guild/${guild.id}/settings/${channel.id}`).appendTo('#text');
  191. }
  192. else {
  193. $('.channel#settings').addClass('selected');
  194. createForm($, 'Server-wide Settings', rows.find( row => !row.channel ), guild.roles).attr('action', `/guild/${guild.id}/settings/default`).appendTo('#text');
  195. }
  196. let body = $.html();
  197. res.writeHead(200, {'Content-Length': body.length});
  198. res.write( body );
  199. return res.end();
  200. } );
  201. }
  202. /**
  203. * Change settings
  204. * @param {Function} res - The server response
  205. * @param {import('./util.js').Settings} userSettings - The settings of the user
  206. * @param {String} guild - The id of the guild
  207. * @param {String} type - The setting to change
  208. * @param {Object} settings - The new settings
  209. * @param {String} [settings.channel]
  210. * @param {String} settings.wiki
  211. * @param {String} [settings.lang]
  212. * @param {String} [settings.role]
  213. * @param {String} [settings.inline]
  214. * @param {String} [settings.prefix]
  215. * @param {String} [settings.prefix_space]
  216. * @param {String} [settings.save_settings]
  217. * @param {String} [settings.delete_settings]
  218. */
  219. function update_settings(res, userSettings, guild, type, settings) {
  220. if ( type !== 'default' && type !== 'new' && type !== settings.channel ) {
  221. return res(`/guild/${guild}/settings`, 'savefail');
  222. }
  223. if ( !settings.save_settings === !settings.delete_settings ) {
  224. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  225. }
  226. if ( settings.save_settings ) {
  227. if ( !settings.wiki || ( settings.lang && !( settings.lang in allLangs.names ) ) ) {
  228. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  229. }
  230. if ( settings.channel && !userSettings.guilds.isMember.get(guild).channels.some( channel => {
  231. return ( channel.id === settings.channel );
  232. } ) ) return res(`/guild/${guild}/settings/${type}`, 'savefail');
  233. if ( settings.role && !userSettings.guilds.isMember.get(guild).roles.some( role => {
  234. return ( role.id === settings.role );
  235. } ) ) return res(`/guild/${guild}/settings/${type}`, 'savefail');
  236. }
  237. if ( settings.delete_settings && ( type === 'default' || type === 'new' ) ) {
  238. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  239. }
  240. sendMsg( {
  241. type: 'getMember',
  242. member: userSettings.user.id,
  243. guild: guild,
  244. channel: ( type === settings.channel ? type : undefined )
  245. } ).then( response => {
  246. if ( !response ) {
  247. userSettings.guilds.notMember.set(guild, userSettings.guilds.isMember.get(guild));
  248. userSettings.guilds.isMember.delete(guild);
  249. return res(`/guild/${guild}`, 'savefail');
  250. }
  251. if ( response === 'noMember' || !hasPerm(response.userPermissions, 'MANAGE_GUILD') ) {
  252. userSettings.guilds.isMember.delete(guild);
  253. return res('/', 'savefail');
  254. }
  255. if ( response.message === 'noChannel' ) return db.run( 'DELETE FROM discord WHERE guild = ? AND channel = ?', [guild, type], function (delerror) {
  256. if ( delerror ) {
  257. console.log( '- Dashboard: Error while removing the settings: ' + delerror );
  258. return res(`/guild/${guild}/settings`, 'savefail');
  259. }
  260. console.log( `- Dashboard: Settings successfully removed: ${guild}#${type}` );
  261. if ( settings.delete_settings ) return res(`/guild/${guild}/settings`, 'save');
  262. else return res(`/guild/${guild}/settings`, 'savefail');
  263. } );
  264. if ( type === settings.channel && !hasPerm(response.userPermissions, 'VIEW_CHANNEL', 'SEND_MESSAGES') ) {
  265. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  266. }
  267. if ( settings.delete_settings ) return db.get( 'SELECT main.lang mainlang, main.patreon, main.lang mainwiki, main.role mainrole, main.inline maininline, old.wiki, old.lang, old.role, old.inline FROM discord main LEFT JOIN discord old ON main.guild = old.guild AND old.channel = ? WHERE main.guild = ? AND main.channel IS NULL', [type, guild], function(dberror, row) {
  268. db.run( 'DELETE FROM discord WHERE guild = ? AND channel = ?', [guild, type], function (delerror) {
  269. if ( delerror ) {
  270. console.log( '- Dashboard: Error while removing the settings: ' + delerror );
  271. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  272. }
  273. console.log( `- Dashboard: Settings successfully removed: ${guild}#${type}` );
  274. res(`/guild/${guild}/settings`, 'save');
  275. if ( dberror ) {
  276. console.log( '- Dashboard: Error while notifying the guild: ' + dberror );
  277. return;
  278. }
  279. if ( !row || row.wiki === null ) return;
  280. var lang = new Lang(row.mainlang);
  281. var text = lang.get('settings.dashboard.removed', `<@${userSettings.user.id}>`, `<#${type}>`);
  282. if ( row.wiki !== row.mainwiki ) text += `\n${lang.get('settings.currentwiki')} <${row.wiki}>`;
  283. if ( row.patreon ) {
  284. if ( row.lang !== row.mainlang ) text += `\n${lang.get('settings.currentlang')} \`${allLangs.names[row.lang]}\``;
  285. if ( row.role !== row.mainrole ) text += `\n${lang.get('settings.currentrole')} ` + ( row.role ? `<@&${row.role}>` : '@everyone' );
  286. if ( row.inline !== row.maininline ) text += `\n${lang.get('settings.currentinline')} ${( row.inline ? '~~' : '' )}\`[[${inlinepage}]]\`${( row.inline ? '~~' : '' )}`;
  287. }
  288. text += `\n<${new URL(`/guild/${guild}/settings`, process.env.dashboard).href}>`;
  289. sendMsg( {
  290. type: 'notifyGuild', guild, text
  291. } ).catch( error => {
  292. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  293. } );
  294. } );
  295. } );
  296. var wiki = Wiki.fromInput(settings.wiki);
  297. return got.get( wiki + 'api.php?&action=query&meta=siteinfo&siprop=general|extensions&format=json' ).then( fresponse => {
  298. if ( fresponse.statusCode === 404 && typeof fresponse.body === 'string' ) {
  299. let api = cheerio.load(fresponse.body)('head link[rel="EditURI"]').prop('href');
  300. if ( api ) {
  301. wiki = new Wiki(api.split('api.php?')[0], wiki);
  302. return got.get( wiki + 'api.php?action=query&meta=siteinfo&siprop=general|extensions&format=json' );
  303. }
  304. }
  305. return fresponse;
  306. } ).then( fresponse => {
  307. return new Promise( function (resolve, reject) {
  308. db.get( 'SELECT lang, wiki, role, inline, prefix FROM discord WHERE guild = ? AND channel IS NULL', [guild], function(error, row) {
  309. if ( error ) {
  310. console.log( '- Dashboard: Error while getting the settings: ' + error );
  311. reject();
  312. }
  313. var body = fresponse.body;
  314. if ( fresponse.statusCode !== 200 || !body?.query?.general || !body?.query?.extensions ) {
  315. console.log( '- Dashboard: ' + fresponse.statusCode + ': Error while testing the wiki: ' + body?.error?.info );
  316. if ( row?.wiki === wiki.href ) resolve(row);
  317. reject();
  318. }
  319. wiki.updateWiki(body.query.general);
  320. resolve(row, body.query);
  321. } );
  322. } );
  323. }, error => {
  324. console.log( '- Dashboard: Error while testing the wiki: ' + error );
  325. return Promise.reject();
  326. } ).then( (row, query) => {
  327. var lang = new Lang(( type === 'default' && settings.lang || row.lang ));
  328. var embed;
  329. if ( !wiki.isFandom() && query ) {
  330. let notice = [];
  331. if ( query.general.generator.replace( /^MediaWiki 1\.(\d\d).*$/, '$1' ) <= 30 ) {
  332. console.log( '- Dashboard: This wiki is using ' + query.general.generator + '.' );
  333. notice.push({
  334. name: 'MediaWiki',
  335. value: lang.get('test.MediaWiki', '[MediaWiki 1.30](https://www.mediawiki.org/wiki/MediaWiki_1.30)', query.general.generator)
  336. });
  337. }
  338. if ( !query.extensions.some( extension => extension.name === 'TextExtracts' ) ) {
  339. console.log( '- Dashboard: This wiki is missing Extension:TextExtracts.' );
  340. notice.push({
  341. name: 'TextExtracts',
  342. value: lang.get('test.TextExtracts', '[TextExtracts](https://www.mediawiki.org/wiki/Extension:TextExtracts)')
  343. });
  344. }
  345. if ( !query.extensions.some( extension => extension.name === 'PageImages' ) ) {
  346. console.log( '- Dashboard: This wiki is missing Extension:PageImages.' );
  347. notice.push({
  348. name: 'PageImages',
  349. value: lang.get('test.PageImages', '[PageImages](https://www.mediawiki.org/wiki/Extension:PageImages)')
  350. });
  351. }
  352. if ( notice.length ) {
  353. embed = {
  354. author: {name: query.general.sitename},
  355. title: lang.get('test.notice'),
  356. fields: notice
  357. }
  358. }
  359. }
  360. if ( type === 'default' ) {
  361. if ( settings.channel || !settings.lang || ( !response.patreon !== !settings.prefix ) ) {
  362. return res(`/guild/${guild}/settings`, 'savefail');
  363. }
  364. if ( settings.prefix ) {
  365. if ( !/^\s*[^\s`\\]{1,100}\s*$/.test(settings.prefix) ) {
  366. return res(`/guild/${guild}/settings`, 'savefail');
  367. }
  368. settings.prefix = settings.prefix.trim().toLowerCase();
  369. if ( settings.prefix_space ) settings.prefix += ' ';
  370. }
  371. if ( !row ) return db.run( 'INSERT INTO discord(wiki, lang, role, inline, prefix, guild, main) VALUES(?, ?, ?, ?, ?, ?, ?)', [wiki.href, settings.lang, ( settings.role || null ), ( settings.inline ? null : 1 ), ( settings.prefix || process.env.prefix ), guild, guild], function(dberror) {
  372. if ( dberror ) {
  373. console.log( '- Dashboard: Error while saving the settings: ' + dberror );
  374. return res(`/guild/${guild}/settings`, 'savefail');
  375. }
  376. console.log( '- Dashboard: Settings successfully saved: ' + guild );
  377. res(`/guild/${guild}/settings`, 'save');
  378. var text = lang.get('settings.dashboard.updated', `<@${userSettings.user.id}>`);
  379. text += '\n' + lang.get('settings.currentwiki') + ` <${wiki.href}>`;
  380. text += '\n' + lang.get('settings.currentlang') + ` \`${allLangs.names[settings.lang]}\``;
  381. text += '\n' + lang.get('settings.currentrole') + ( settings.role ? ` <@&${settings.role}>` : ' @everyone' );
  382. if ( response.patreon ) {
  383. text += '\n' + lang.get('settings.currentprefix') + ` \`${settings.prefix.replace( /\\/g, '\\$&' )}\``;
  384. }
  385. text += '\n' + lang.get('settings.currentinline') + ` ${( settings.inline ? '' : '~~' )}\`[[${( lang.localNames.page || 'page' )}]]\`${( settings.inline ? '' : '~~' )}`;
  386. text += `\n<${new URL(`/guild/${guild}/settings`, process.env.dashboard).href}>`;
  387. sendMsg( {
  388. type: 'notifyGuild', guild, text, embed
  389. } ).catch( error => {
  390. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  391. } );
  392. } );
  393. var diff = [];
  394. var updateGuild = false;
  395. var updateChannel = false;
  396. if ( row.wiki !== wiki.href ) {
  397. updateGuild = true;
  398. diff.push(lang.get('settings.currentwiki') + ` ~~<${row.wiki}>~~ → <${wiki.href}>`);
  399. }
  400. if ( row.lang !== settings.lang ) {
  401. updateChannel = true;
  402. diff.push(lang.get('settings.currentlang') + ` ~~\`${allLangs.names[row.lang]}\`~~ → \`${allLangs.names[settings.lang]}\``);
  403. }
  404. if ( response.patreon && row.prefix !== settings.prefix ) {
  405. updateChannel = true;
  406. diff.push(lang.get('settings.currentprefix') + ` ~~\`${row.prefix.replace( /\\/g, '\\$&' )}\`~~ → \`${settings.prefix.replace( /\\/g, '\\$&' )}\``);
  407. }
  408. if ( row.role !== ( settings.role || null ) ) {
  409. updateChannel = true;
  410. diff.push(lang.get('settings.currentrole') + ` ~~${( row.role ? `<@&${row.role}>` : '@everyone' )}~~ → ${( settings.role ? `<@&${settings.role}>` : '@everyone' )}`);
  411. }
  412. if ( row.inline !== ( settings.inline ? null : 1 ) ) {
  413. updateChannel = true;
  414. let inlinepage = ( lang.localNames.page || 'page' );
  415. diff.push(lang.get('settings.currentinline') + ` ${( row.inline ? '~~' : '' )}\`[[${inlinepage}]]\`${( row.inline ? '~~' : '' )} → ${( settings.inline ? '' : '~~' )}\`[[${inlinepage}]]\`${( settings.inline ? '' : '~~' )}`);
  416. }
  417. if ( diff.length ) {
  418. var dbupdate = [];
  419. if ( response.patreon ) {
  420. dbupdate.push([
  421. 'UPDATE discord SET wiki = ?, lang = ?, role = ?, inline = ?, prefix = ? WHERE guild = ? AND channel IS NULL',
  422. [wiki.href, settings.lang, ( settings.role || null ), ( settings.inline ? null : 1 ), ( settings.prefix || process.env.prefix ), guild]
  423. ]);
  424. }
  425. else {
  426. if ( updateGuild ) {
  427. dbupdate.push([
  428. 'UPDATE discord SET wiki = ? WHERE guild = ? AND channel IS NULL',
  429. [wiki.href, guild]
  430. ]);
  431. }
  432. if ( updateChannel ) {
  433. dbupdate.push([
  434. 'UPDATE discord SET lang = ?, role = ?, inline = ?, prefix = ? WHERE guild = ?',
  435. [settings.lang, ( settings.role || null ), ( settings.inline ? null : 1 ), ( settings.prefix || process.env.prefix ), guild]
  436. ]);
  437. }
  438. }
  439. return Promise.all([
  440. ...dbupdate.map( ([sql, sqlargs]) => {
  441. return new Promise( function (resolve, reject) {
  442. db.run( sql, sqlargs, function(error) {
  443. if (error) reject(error);
  444. else resolve(this);
  445. } );
  446. } );
  447. } )
  448. ]).then( () => {
  449. console.log( '- Dashboard: Settings successfully saved: ' + guild );
  450. res(`/guild/${guild}/settings`, 'save');
  451. var text = lang.get('settings.dashboard.updated', `<@${userSettings.user.id}>`);
  452. text += '\n' + diff.join('\n');
  453. text += `\n<${new URL(`/guild/${guild}/settings`, process.env.dashboard).href}>`;
  454. sendMsg( {
  455. type: 'notifyGuild', guild, text, embed,
  456. prefix: settings.prefix, voice: settings.lang
  457. } ).catch( error => {
  458. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  459. } );
  460. }, error => {
  461. console.log( '- Dashboard: Error while saving the settings: ' + error );
  462. return res(`/guild/${guild}/settings`, 'savefail');
  463. } );
  464. }
  465. return res(`/guild/${guild}/settings`, 'save');
  466. }
  467. if ( !row || !settings.channel || settings.prefix ||
  468. ( !response.patreon && ( settings.lang || settings.role || settings.inline ) ) ) {
  469. return res(`/guild/${guild}/settings`, 'savefail');
  470. }
  471. if ( row.wiki === wiki.href && ( !response.patreon ||
  472. ( row.lang === settings.lang && row.inline === ( settings.inline ? null : 1 ) && row.role === ( settings.role || null ) ) ) ) {
  473. if ( type === 'new' ) {
  474. return res(`/guild/${guild}/settings/${type}`, 'nochange');
  475. }
  476. return db.run( 'DELETE FROM discord WHERE guild = ? AND channel = ?', [guild, type], function (delerror) {
  477. if ( delerror ) {
  478. console.log( '- Dashboard: Error while removing the settings: ' + delerror );
  479. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  480. }
  481. console.log( `- Dashboard: Settings successfully removed: ${guild}#${type}` );
  482. res(`/guild/${guild}/settings`, 'save');
  483. var text = lang.get('settings.dashboard.removed', `<@${userSettings.user.id}>`, `<#${type}>`);
  484. text += `\n<${new URL(`/guild/${guild}/settings`, process.env.dashboard).href}>`;
  485. sendMsg( {
  486. type: 'notifyGuild', guild, text
  487. } ).catch( error => {
  488. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  489. } );
  490. } );
  491. }
  492. return db.get( 'SELECT lang, wiki, role, inline FROM discord WHERE guild = ? AND channel = ?', [guild, settings.channel], function(curerror, channel) {
  493. if ( curerror ) {
  494. console.log( '- Dashboard: Error while getting the channel settings: ' + curerror );
  495. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  496. }
  497. if ( !channel ) channel = row;
  498. var diff = [];
  499. if ( channel.wiki !== wiki.href ) {
  500. diff.push(lang.get('settings.currentwiki') + ` ~~<${channel.wiki}>~~ → <${wiki.href}>`);
  501. }
  502. if ( response.patreon && channel.lang !== settings.lang ) {
  503. diff.push(lang.get('settings.currentlang') + ` ~~\`${allLangs.names[channel.lang]}\`~~ → \`${allLangs.names[settings.lang]}\``);
  504. }
  505. if ( response.patreon && channel.role !== ( settings.role || null ) ) {
  506. diff.push(lang.get('settings.currentrole') + ` ~~${( channel.role ? `<@&${channel.role}>` : '@everyone' )}~~ → ${( settings.role ? `<@&${settings.role}>` : '@everyone' )}`);
  507. }
  508. if ( response.patreon && channel.inline !== ( settings.inline ? null : 1 ) ) {
  509. let inlinepage = ( lang.localNames.page || 'page' );
  510. diff.push(lang.get('settings.currentinline') + ` ${( channel.inline ? '~~' : '' )}\`[[${inlinepage}]]\`${( channel.inline ? '~~' : '' )} → ${( settings.inline ? '' : '~~' )}\`[[${inlinepage}]]\`${( settings.inline ? '' : '~~' )}`);
  511. }
  512. if ( !diff.length ) {
  513. return res(`/guild/${guild}/settings/${settings.channel}`, 'save');
  514. }
  515. let sql = 'UPDATE discord SET wiki = ?, lang = ?, role = ?, inline = ? WHERE guild = ? AND channel = ?';
  516. let sqlargs = [wiki.href, ( settings.lang || channel.lang ), ( response.patreon ? ( settings.role || null ) : channel.role ), ( response.patreon ? ( settings.inline ? null : 1 ) : channel.inline ), guild, settings.channel];
  517. if ( channel === row ) {
  518. sql = 'INSERT INTO discord(wiki, lang, role, inline, guild, channel, prefix) VALUES(?, ?, ?, ?, ?, ?, ?)';
  519. sqlargs.push(row.prefix);
  520. }
  521. return db.run( sql, sqlargs, function(dberror) {
  522. if ( dberror ) {
  523. console.log( '- Dashboard: Error while saving the settings: ' + dberror );
  524. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  525. }
  526. console.log( `- Dashboard: Settings successfully saved: ${guild}#${settings.channel}` );
  527. res(`/guild/${guild}/settings/${settings.channel}`, 'save');
  528. var text = lang.get('settings.dashboard.channel', `<@${userSettings.user.id}>`, `<#${settings.channel}>`);
  529. text += '\n' + diff.join('\n');
  530. text += `\n<${new URL(`/guild/${guild}/settings/${settings.channel}`, process.env.dashboard).href}>`;
  531. sendMsg( {
  532. type: 'notifyGuild', guild, text, embed
  533. } ).catch( error => {
  534. console.log( '- Dashboard: Error while notifying the guild: ' + error );
  535. } );
  536. } );
  537. } );
  538. }, () => {
  539. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  540. } );
  541. }, error => {
  542. console.log( '- Dashboard: Error while getting the member: ' + error );
  543. return res(`/guild/${guild}/settings/${type}`, 'savefail');
  544. } );
  545. }
  546. module.exports = {
  547. get: dashboard_settings,
  548. post: update_settings
  549. };