settings.js 31 KB

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