2016-01-04 00:02:08 +00:00
|
|
|
#!/usr/bin/env node
|
|
|
|
|
2016-01-03 22:50:56 +00:00
|
|
|
/* jslint node: true */
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
// ENiGMA½
|
2016-04-12 04:13:16 +00:00
|
|
|
const config = require('./core/config.js');
|
|
|
|
const db = require('./core/database.js');
|
|
|
|
const resolvePath = require('./core/misc_util.js').resolvePath;
|
2016-01-03 22:50:56 +00:00
|
|
|
|
2016-04-12 04:13:16 +00:00
|
|
|
// deps
|
|
|
|
const _ = require('lodash');
|
|
|
|
const async = require('async');
|
|
|
|
const assert = require('assert');
|
|
|
|
const inq = require('inquirer');
|
|
|
|
const mkdirsSync = require('fs-extra').mkdirsSync;
|
|
|
|
const fs = require('fs');
|
|
|
|
const hjson = require('hjson');
|
|
|
|
const paths = require('path');
|
2016-01-03 22:50:56 +00:00
|
|
|
|
|
|
|
var argv = require('minimist')(process.argv.slice(2));
|
|
|
|
|
2016-02-03 04:35:59 +00:00
|
|
|
const ExitCodes = {
|
2016-01-03 22:50:56 +00:00
|
|
|
SUCCESS : 0,
|
|
|
|
ERROR : -1,
|
|
|
|
BAD_COMMAND : -2,
|
|
|
|
BAD_ARGS : -3,
|
|
|
|
}
|
|
|
|
|
2016-07-02 20:05:38 +00:00
|
|
|
const USAGE_HELP = {
|
|
|
|
General :
|
|
|
|
`usage: optutil.js [--version] [--help]
|
|
|
|
<command> [<args>]
|
|
|
|
|
|
|
|
global args:
|
|
|
|
--config PATH : specify config path (${getDefaultConfigPath()})
|
|
|
|
|
|
|
|
commands:
|
|
|
|
user : user utilities
|
|
|
|
config : config file management
|
|
|
|
|
|
|
|
`,
|
|
|
|
User :
|
|
|
|
`usage: optutil.js user --user USERNAME <args>
|
|
|
|
|
|
|
|
valid args:
|
|
|
|
--user USERNAME : specify username
|
|
|
|
-- password PASS : specify password (to reset)
|
|
|
|
`,
|
2016-01-03 22:50:56 +00:00
|
|
|
|
2016-07-02 20:15:28 +00:00
|
|
|
Config :
|
|
|
|
`usage: optutil.js config <args>
|
2016-01-03 22:50:56 +00:00
|
|
|
|
2016-07-02 20:15:28 +00:00
|
|
|
valid args:
|
|
|
|
--new : generate a new/initial configuration
|
|
|
|
`
|
|
|
|
}
|
2016-01-03 22:50:56 +00:00
|
|
|
|
2016-07-02 20:15:28 +00:00
|
|
|
function printUsage(command) {
|
|
|
|
console.error(USAGE_HELP[command]);
|
2016-01-03 22:50:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function initConfig(cb) {
|
2016-02-03 04:35:59 +00:00
|
|
|
const configPath = argv.config ? argv.config : config.getDefaultPath();
|
2016-01-03 22:50:56 +00:00
|
|
|
|
|
|
|
config.init(configPath, cb);
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleUserCommand() {
|
|
|
|
if(true === argv.help || !_.isString(argv.user) || 0 === argv.user.length) {
|
|
|
|
process.exitCode = ExitCodes.ERROR;
|
2016-07-02 20:15:28 +00:00
|
|
|
return printUsage('User');
|
2016-01-03 22:50:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(_.isString(argv.password)) {
|
|
|
|
if(0 === argv.password.length) {
|
|
|
|
process.exitCode = ExitCodes.BAD_ARGS;
|
|
|
|
return console.error('Invalid password');
|
|
|
|
}
|
|
|
|
|
|
|
|
var user;
|
|
|
|
async.waterfall(
|
|
|
|
[
|
|
|
|
function init(callback) {
|
|
|
|
initConfig(callback);
|
|
|
|
},
|
|
|
|
function initDb(callback) {
|
|
|
|
db.initializeDatabases(callback);
|
|
|
|
},
|
|
|
|
function getUser(callback) {
|
|
|
|
user = require('./core/user.js');
|
2016-01-03 23:04:37 +00:00
|
|
|
user.getUserIdAndName(argv.user, function userNameAndId(err, userId) {
|
2016-01-03 22:50:56 +00:00
|
|
|
if(err) {
|
|
|
|
process.exitCode = ExitCodes.BAD_ARGS;
|
|
|
|
callback(new Error('Failed to retrieve user'));
|
|
|
|
} else {
|
|
|
|
callback(null, userId);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function setNewPass(userId, callback) {
|
2016-01-03 23:04:37 +00:00
|
|
|
assert(_.isNumber(userId));
|
|
|
|
assert(userId > 0);
|
2016-01-04 00:02:08 +00:00
|
|
|
|
2016-02-03 04:35:59 +00:00
|
|
|
let u = new user.User();
|
2016-01-03 22:50:56 +00:00
|
|
|
u.userId = userId;
|
2016-01-03 23:04:37 +00:00
|
|
|
|
2016-01-03 22:50:56 +00:00
|
|
|
u.setNewAuthCredentials(argv.password, function credsSet(err) {
|
|
|
|
if(err) {
|
|
|
|
process.exitCode = ExitCodes.ERROR;
|
|
|
|
callback(new Error('Failed setting password'));
|
|
|
|
} else {
|
|
|
|
callback(null);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
],
|
|
|
|
function complete(err) {
|
|
|
|
if(err) {
|
|
|
|
console.error(err.message);
|
|
|
|
} else {
|
|
|
|
console.info('Password set');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-12 04:13:16 +00:00
|
|
|
function getAnswers(questions, cb) {
|
|
|
|
inq.prompt(questions, cb);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getDefaultConfigPath() {
|
2016-04-12 04:17:43 +00:00
|
|
|
return resolvePath('~/.config/enigma-bbs/config.hjson');
|
2016-04-12 04:13:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const QUESTIONS = {
|
|
|
|
Intro : [
|
|
|
|
{
|
|
|
|
name : 'createNewConfig',
|
|
|
|
message : 'Create a new configuration?',
|
|
|
|
type : 'confirm',
|
|
|
|
default : false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name : 'configPath',
|
|
|
|
message : 'Configuration path:',
|
2016-07-02 20:05:38 +00:00
|
|
|
default : argv.config ? argv.config : getDefaultConfigPath(),
|
2016-04-12 04:13:16 +00:00
|
|
|
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(),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name : 'sevenZipExe',
|
|
|
|
message : '7-Zip executable:',
|
|
|
|
type : 'list',
|
|
|
|
choices : [ '7z', '7za', 'None' ]
|
|
|
|
}
|
|
|
|
],
|
|
|
|
|
|
|
|
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',
|
|
|
|
}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
|
|
|
|
function makeMsgConfAreaName(s) {
|
|
|
|
return s.toLowerCase().replace(/\s+/g, '_');
|
|
|
|
}
|
|
|
|
|
|
|
|
function askQuestions(cb) {
|
|
|
|
|
|
|
|
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 => {
|
|
|
|
callback(answers.overwriteConfig ? null : 'exit');
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
callback(null);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function basic(callback) {
|
|
|
|
getAnswers(QUESTIONS.Basic, answers => {
|
|
|
|
config = {
|
|
|
|
general : {
|
|
|
|
boardName : answers.boardName,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
callback(null);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function msgConfAndArea(callback) {
|
|
|
|
getAnswers(QUESTIONS.MessageConfAndArea, answers => {
|
|
|
|
config.messageConferences = {};
|
|
|
|
|
|
|
|
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.another_sample_conf = {
|
|
|
|
name : 'Another Sample Conference',
|
|
|
|
desc : 'Another conference example. Change me!',
|
|
|
|
sort : 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
config.messageConferences[confName].areas = {};
|
|
|
|
config.messageConferences[confName].areas[areaName] = {
|
|
|
|
name : answers.msgAreaName,
|
|
|
|
desc : answers.msgAreaDesc,
|
|
|
|
sort : 1,
|
|
|
|
default : true,
|
|
|
|
};
|
|
|
|
|
|
|
|
config.messageConferences.another_sample_conf = {
|
|
|
|
areas : {
|
|
|
|
another_sample_area : {
|
|
|
|
name : 'Another Sample Area',
|
|
|
|
desc : 'Another area example. Change me!',
|
|
|
|
sort : 2
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
callback(null);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function misc(callback) {
|
|
|
|
getAnswers(QUESTIONS.Misc, answers => {
|
|
|
|
if('None' !== answers.sevenZipExe) {
|
|
|
|
config.archivers = {
|
|
|
|
zip : {
|
|
|
|
compressCmd : answers.sevenZipExe,
|
|
|
|
decompressCmd : answers.sevenZipExe,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
config.logging = {
|
|
|
|
level : answers.loggingLevel,
|
|
|
|
};
|
|
|
|
|
|
|
|
callback(null);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
],
|
|
|
|
err => {
|
|
|
|
cb(err, configPath, config);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleConfigCommand() {
|
2016-07-02 20:15:28 +00:00
|
|
|
if(true === argv.help) {
|
|
|
|
process.exitCode = ExitCodes.ERROR;
|
|
|
|
return printUsage('Config');
|
|
|
|
}
|
|
|
|
|
|
|
|
if(argv.new) {
|
|
|
|
askQuestions( (err, configPath, config) => {
|
|
|
|
if(err) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
config = hjson.stringify(config, { bracesSameLine : true, spaces : '\t' } );
|
|
|
|
|
|
|
|
try {
|
|
|
|
fs.writeFileSync(configPath, config, 'utf8');
|
|
|
|
console.info('Configuration generated');
|
|
|
|
} catch(e) {
|
|
|
|
console.error('Exception attempting to create config: ' + e.toString());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
process.exitCode = ExitCodes.ERROR;
|
|
|
|
return printUsage('Config');
|
|
|
|
}
|
2016-04-12 04:13:16 +00:00
|
|
|
}
|
|
|
|
|
2016-01-03 22:50:56 +00:00
|
|
|
function main() {
|
|
|
|
|
|
|
|
process.exitCode = ExitCodes.SUCCESS;
|
|
|
|
|
2016-01-04 00:57:57 +00:00
|
|
|
if(true === argv.version) {
|
|
|
|
return console.info(require('./package.json').version);
|
|
|
|
}
|
|
|
|
|
2016-01-03 22:50:56 +00:00
|
|
|
if(0 === argv._.length ||
|
|
|
|
'help' === argv._[0])
|
|
|
|
{
|
2016-07-02 20:15:28 +00:00
|
|
|
printUsage('General');
|
2016-01-03 22:50:56 +00:00
|
|
|
process.exit(ExitCodes.SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(argv._[0]) {
|
|
|
|
case 'user' :
|
|
|
|
handleUserCommand();
|
|
|
|
break;
|
2016-04-12 04:13:16 +00:00
|
|
|
|
|
|
|
case 'config' :
|
|
|
|
handleConfigCommand();
|
|
|
|
break;
|
2016-01-03 22:50:56 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
printUsage('');
|
|
|
|
process.exitCode = ExitCodes.BAD_COMMAND;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
main();
|