2017-02-16 03:27:16 +00:00
|
|
|
/* jslint node: true */
|
|
|
|
/* eslint-disable no-console */
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
// ENiGMA½
|
|
|
|
const resolvePath = require('../../core/misc_util.js').resolvePath;
|
2018-11-11 06:59:26 +00:00
|
|
|
const {
|
|
|
|
printUsageAndSetExitCode,
|
|
|
|
getConfigPath,
|
|
|
|
argv,
|
|
|
|
ExitCodes,
|
2018-11-25 03:02:19 +00:00
|
|
|
getAnswers,
|
|
|
|
writeConfig,
|
|
|
|
HJSONStringifyCommonOpts,
|
2018-11-11 06:59:26 +00:00
|
|
|
} = require('./oputil_common.js');
|
2017-02-16 03:27:16 +00:00
|
|
|
const getHelpFor = require('./oputil_help.js').getHelpFor;
|
|
|
|
|
|
|
|
// deps
|
2018-11-11 07:19:01 +00:00
|
|
|
const async = require('async');
|
|
|
|
const inq = require('inquirer');
|
|
|
|
const mkdirsSync = require('fs-extra').mkdirsSync;
|
|
|
|
const fs = require('graceful-fs');
|
|
|
|
const hjson = require('hjson');
|
|
|
|
const paths = require('path');
|
|
|
|
const _ = require('lodash');
|
|
|
|
const sanatizeFilename = require('sanitize-filename');
|
2017-02-16 03:27:16 +00:00
|
|
|
|
|
|
|
exports.handleConfigCommand = handleConfigCommand;
|
|
|
|
|
2018-11-08 03:24:05 +00:00
|
|
|
const ConfigIncludeKeys = [
|
|
|
|
'theme',
|
2018-11-11 08:00:42 +00:00
|
|
|
'users.preAuthIdleLogoutSeconds', 'users.idleLogoutSeconds',
|
2018-11-23 06:07:37 +00:00
|
|
|
'users.newUserNames', 'users.failedLogin', 'users.unlockAtEmailPwReset',
|
2018-11-11 08:00:42 +00:00
|
|
|
'paths.logs',
|
2018-11-11 03:00:25 +00:00
|
|
|
'loginServers',
|
|
|
|
'contentServers',
|
2018-11-11 03:26:23 +00:00
|
|
|
'fileBase.areaStoragePrefix',
|
2018-11-11 08:00:42 +00:00
|
|
|
'logging.rotatingFile',
|
2018-11-08 03:24:05 +00:00
|
|
|
];
|
|
|
|
|
2017-02-16 03:27:16 +00:00
|
|
|
const QUESTIONS = {
|
2018-06-22 05:15:04 +00:00
|
|
|
Intro : [
|
|
|
|
{
|
|
|
|
name : 'createNewConfig',
|
|
|
|
message : 'Create a new configuration?',
|
|
|
|
type : 'confirm',
|
|
|
|
default : false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name : 'configPath',
|
|
|
|
message : 'Configuration path:',
|
|
|
|
default : getConfigPath(),
|
|
|
|
when : answers => answers.createNewConfig
|
|
|
|
},
|
|
|
|
],
|
|
|
|
|
|
|
|
OverwriteConfig : [
|
|
|
|
{
|
|
|
|
name : 'overwriteConfig',
|
|
|
|
message : 'Config file exists. Overwrite?',
|
|
|
|
type : 'confirm',
|
|
|
|
default : false,
|
|
|
|
}
|
|
|
|
],
|
|
|
|
|
|
|
|
Basic : [
|
|
|
|
{
|
|
|
|
name : 'boardName',
|
|
|
|
message : 'BBS name:',
|
|
|
|
default : 'New ENiGMA½ BBS',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
|
|
|
|
Misc : [
|
|
|
|
{
|
|
|
|
name : 'loggingLevel',
|
|
|
|
message : 'Logging level:',
|
|
|
|
type : 'list',
|
|
|
|
choices : [ 'Error', 'Warn', 'Info', 'Debug', 'Trace' ],
|
|
|
|
default : 2,
|
|
|
|
filter : s => s.toLowerCase(),
|
|
|
|
},
|
|
|
|
],
|
|
|
|
|
|
|
|
MessageConfAndArea : [
|
|
|
|
{
|
|
|
|
name : 'msgConfName',
|
|
|
|
message : 'First message conference:',
|
|
|
|
default : 'Local',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name : 'msgConfDesc',
|
|
|
|
message : 'Conference description:',
|
|
|
|
default : 'Local Areas',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name : 'msgAreaName',
|
|
|
|
message : 'First area in message conference:',
|
|
|
|
default : 'General',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name : 'msgAreaDesc',
|
|
|
|
message : 'Area description:',
|
|
|
|
default : 'General chit-chat',
|
|
|
|
}
|
|
|
|
]
|
2017-02-16 03:27:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
function makeMsgConfAreaName(s) {
|
2018-06-22 05:15:04 +00:00
|
|
|
return s.toLowerCase().replace(/\s+/g, '_');
|
2017-02-16 03:27:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function askNewConfigQuestions(cb) {
|
2018-06-22 05:15:04 +00:00
|
|
|
|
|
|
|
const ui = new inq.ui.BottomBar();
|
|
|
|
|
|
|
|
let configPath;
|
|
|
|
let config;
|
|
|
|
|
|
|
|
async.waterfall(
|
|
|
|
[
|
|
|
|
function intro(callback) {
|
|
|
|
getAnswers(QUESTIONS.Intro, answers => {
|
|
|
|
if(!answers.createNewConfig) {
|
|
|
|
return callback('exit');
|
|
|
|
}
|
|
|
|
|
|
|
|
// adjust for ~ and the like
|
|
|
|
configPath = resolvePath(answers.configPath);
|
|
|
|
|
|
|
|
const configDir = paths.dirname(configPath);
|
|
|
|
mkdirsSync(configDir);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check if the file exists and can be written to
|
|
|
|
//
|
|
|
|
fs.access(configPath, fs.F_OK | fs.W_OK, err => {
|
|
|
|
if(err) {
|
|
|
|
if('EACCES' === err.code) {
|
|
|
|
ui.log.write(`${configPath} cannot be written to`);
|
|
|
|
callback('exit');
|
|
|
|
} else if('ENOENT' === err.code) {
|
|
|
|
callback(null, false);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
callback(null, true); // exists + writable
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function promptOverwrite(needPrompt, callback) {
|
|
|
|
if(needPrompt) {
|
|
|
|
getAnswers(QUESTIONS.OverwriteConfig, answers => {
|
2018-11-08 03:24:05 +00:00
|
|
|
return callback(answers.overwriteConfig ? null : 'exit');
|
2018-06-22 05:15:04 +00:00
|
|
|
});
|
|
|
|
} else {
|
2018-11-08 03:24:05 +00:00
|
|
|
return callback(null);
|
2018-06-22 05:15:04 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
function basic(callback) {
|
|
|
|
getAnswers(QUESTIONS.Basic, answers => {
|
2020-06-07 21:26:11 +00:00
|
|
|
const defaultConfig = require('../../core/config_default')();
|
2018-11-08 03:24:05 +00:00
|
|
|
|
|
|
|
// start by plopping in values we want directly from config.js
|
|
|
|
const template = hjson.rt.parse(fs.readFileSync(paths.join(__dirname, '../../misc/config_template.in.hjson'), 'utf8'));
|
2018-06-22 05:15:04 +00:00
|
|
|
|
2018-11-08 03:24:05 +00:00
|
|
|
const direct = {};
|
|
|
|
_.each(ConfigIncludeKeys, keyPath => {
|
|
|
|
_.set(direct, keyPath, _.get(defaultConfig, keyPath));
|
|
|
|
});
|
|
|
|
|
|
|
|
config = _.mergeWith(template, direct);
|
|
|
|
|
|
|
|
// we can override/add to it based on user input from this point on...
|
|
|
|
config.general.boardName = answers.boardName;
|
2018-11-10 02:02:07 +00:00
|
|
|
|
2018-11-08 03:24:05 +00:00
|
|
|
return callback(null);
|
2018-06-22 05:15:04 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
function msgConfAndArea(callback) {
|
|
|
|
getAnswers(QUESTIONS.MessageConfAndArea, answers => {
|
|
|
|
const confName = makeMsgConfAreaName(answers.msgConfName);
|
|
|
|
const areaName = makeMsgConfAreaName(answers.msgAreaName);
|
|
|
|
|
|
|
|
config.messageConferences[confName] = {
|
|
|
|
name : answers.msgConfName,
|
|
|
|
desc : answers.msgConfDesc,
|
|
|
|
sort : 1,
|
|
|
|
default : true,
|
|
|
|
};
|
|
|
|
|
|
|
|
config.messageConferences[confName].areas = {};
|
|
|
|
config.messageConferences[confName].areas[areaName] = {
|
|
|
|
name : answers.msgAreaName,
|
|
|
|
desc : answers.msgAreaDesc,
|
|
|
|
sort : 1,
|
|
|
|
default : true,
|
|
|
|
};
|
|
|
|
|
2018-11-08 03:24:05 +00:00
|
|
|
return callback(null);
|
2018-06-22 05:15:04 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
function misc(callback) {
|
|
|
|
getAnswers(QUESTIONS.Misc, answers => {
|
2018-11-08 03:24:05 +00:00
|
|
|
config.logging.rotatingFile.level = answers.loggingLevel;
|
2018-06-22 05:15:04 +00:00
|
|
|
|
2018-11-08 03:24:05 +00:00
|
|
|
return callback(null);
|
2018-06-22 05:15:04 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
],
|
|
|
|
err => {
|
2018-11-08 03:24:05 +00:00
|
|
|
return cb(err, configPath, config);
|
2018-06-22 05:15:04 +00:00
|
|
|
}
|
|
|
|
);
|
2017-02-16 03:27:16 +00:00
|
|
|
}
|
|
|
|
|
2018-11-11 07:19:01 +00:00
|
|
|
const copyFileSyncSilent = (to, from, flags) => {
|
|
|
|
try {
|
|
|
|
fs.copyFileSync(to, from, flags);
|
2018-11-25 03:02:19 +00:00
|
|
|
} catch(e) {
|
|
|
|
/* absorb! */
|
|
|
|
}
|
2018-11-11 07:19:01 +00:00
|
|
|
};
|
|
|
|
|
2017-02-20 18:31:24 +00:00
|
|
|
function buildNewConfig() {
|
2018-06-22 05:15:04 +00:00
|
|
|
askNewConfigQuestions( (err, configPath, config) => {
|
2018-11-25 03:02:19 +00:00
|
|
|
if(err) { return;
|
2018-06-22 05:15:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-05 20:06:34 +00:00
|
|
|
const boardName = sanatizeFilename(config.general.boardName)
|
2018-11-25 03:02:19 +00:00
|
|
|
.replace(/[^a-z0-9_-]/ig, '_')
|
2018-11-14 02:44:27 +00:00
|
|
|
.replace(/_+/g, '_')
|
|
|
|
.toLowerCase();
|
2020-07-05 20:06:34 +00:00
|
|
|
const menuFile = `menus/${boardName}-main.hjson`;
|
|
|
|
|
|
|
|
const mainTemplate = hjson.rt.parse(fs.readFileSync(paths.join(__dirname, '../../misc/menu_templates/main.in.hjson'), 'utf8'));
|
|
|
|
|
|
|
|
const includeFiles = [
|
|
|
|
'message_base.in.hjson',
|
|
|
|
'private_mail.in.hjson',
|
|
|
|
'login.in.hjson',
|
|
|
|
'new_user.in.hjson',
|
|
|
|
'doors.in.hjson',
|
|
|
|
'file_base.in.hjson',
|
|
|
|
];
|
|
|
|
|
|
|
|
includeFiles.forEach(incFile => {
|
|
|
|
const outName = `${boardName}-${incFile.replace('.in', '')}`;
|
|
|
|
copyFileSyncSilent(
|
|
|
|
paths.join(__dirname, '../../misc/menu_templates', incFile),
|
|
|
|
paths.join(__dirname, '../../config/menus', outName),
|
|
|
|
fs.constants.COPYFILE_EXCL
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
mainTemplate.includes = includeFiles.map(incFile => {
|
|
|
|
return `${boardName}-${incFile.replace('.in', '')}`;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (writeConfig(
|
|
|
|
mainTemplate,
|
|
|
|
paths.join(__dirname, '../../config/menus', `${boardName}-main.hjson`)))
|
|
|
|
{
|
|
|
|
config.general.menuFile = menuFile;
|
|
|
|
|
|
|
|
if(writeConfig(config, configPath)) {
|
|
|
|
console.info('Configuration generated');
|
|
|
|
} else {
|
|
|
|
console.error('Failed writing configuration');
|
|
|
|
}
|
2018-06-22 05:15:04 +00:00
|
|
|
} else {
|
|
|
|
console.error('Failed writing configuration');
|
|
|
|
}
|
|
|
|
});
|
2017-02-20 18:31:24 +00:00
|
|
|
}
|
|
|
|
|
2018-11-11 06:59:26 +00:00
|
|
|
function catCurrentConfig() {
|
|
|
|
try {
|
|
|
|
const config = hjson.rt.parse(fs.readFileSync(getConfigPath(), 'utf8'));
|
2018-11-25 03:02:19 +00:00
|
|
|
const hjsonOpts = Object.assign({}, HJSONStringifyCommonOpts, {
|
2018-11-11 06:59:26 +00:00
|
|
|
colors : false === argv.colors ? false : true,
|
|
|
|
keepWsc : false === argv.comments ? false : true,
|
|
|
|
});
|
|
|
|
|
|
|
|
console.log(hjson.stringify(config, hjsonOpts));
|
|
|
|
} catch(e) {
|
|
|
|
if('ENOENT' == e.code) {
|
|
|
|
console.error(`File not found: ${getConfigPath()}`);
|
|
|
|
} else {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-20 18:31:24 +00:00
|
|
|
function handleConfigCommand() {
|
2018-06-22 05:15:04 +00:00
|
|
|
if(true === argv.help) {
|
|
|
|
return printUsageAndSetExitCode(getHelpFor('Config'), ExitCodes.ERROR);
|
|
|
|
}
|
2017-02-20 18:31:24 +00:00
|
|
|
|
2018-06-22 05:15:04 +00:00
|
|
|
const action = argv._[1];
|
2017-02-20 18:31:24 +00:00
|
|
|
|
2018-06-22 05:15:04 +00:00
|
|
|
switch(action) {
|
2018-11-25 03:02:19 +00:00
|
|
|
case 'new' : return buildNewConfig();
|
|
|
|
case 'cat' : return catCurrentConfig();
|
2017-02-20 18:31:24 +00:00
|
|
|
|
2018-06-22 05:15:04 +00:00
|
|
|
default : return printUsageAndSetExitCode(getHelpFor('Config'), ExitCodes.ERROR);
|
|
|
|
}
|
2017-02-16 03:27:16 +00:00
|
|
|
}
|