settings.js 31 KB

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