275 lines
11 KiB
JavaScript
275 lines
11 KiB
JavaScript
'use strict';
|
|
|
|
exports = module.exports = {
|
|
setup: setup,
|
|
restore: restore,
|
|
activate: activate,
|
|
getStatus: getStatus,
|
|
|
|
autoRegister: autoRegister
|
|
};
|
|
|
|
var appstore = require('./appstore.js'),
|
|
assert = require('assert'),
|
|
async = require('async'),
|
|
backups = require('./backups.js'),
|
|
BoxError = require('./boxerror.js'),
|
|
constants = require('./constants.js'),
|
|
clients = require('./clients.js'),
|
|
cloudron = require('./cloudron.js'),
|
|
debug = require('debug')('box:provision'),
|
|
domains = require('./domains.js'),
|
|
eventlog = require('./eventlog.js'),
|
|
fs = require('fs'),
|
|
mail = require('./mail.js'),
|
|
paths = require('./paths.js'),
|
|
safe = require('safetydance'),
|
|
semver = require('semver'),
|
|
settings = require('./settings.js'),
|
|
superagent = require('superagent'),
|
|
sysinfo = require('./sysinfo.js'),
|
|
users = require('./users.js'),
|
|
tld = require('tldjs'),
|
|
_ = require('underscore');
|
|
|
|
const NOOP_CALLBACK = function (error) { if (error) debug(error); };
|
|
|
|
// we cannot use tasks since the tasks table gets overwritten when db is imported
|
|
let gProvisionStatus = {
|
|
setup: {
|
|
active: false,
|
|
message: '',
|
|
errorMessage: null
|
|
},
|
|
restore: {
|
|
active: false,
|
|
message: '',
|
|
errorMessage: null
|
|
}
|
|
};
|
|
|
|
function setProgress(task, message, callback) {
|
|
debug(`setProgress: ${task} - ${message}`);
|
|
gProvisionStatus[task].message = message;
|
|
callback();
|
|
}
|
|
|
|
function autoRegister(domain, callback) {
|
|
assert.strictEqual(typeof domain, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
if (!fs.existsSync(paths.LICENSE_FILE)) return callback();
|
|
|
|
const license = safe.fs.readFileSync(paths.LICENSE_FILE, 'utf8');
|
|
if (!license) return callback(new BoxError(BoxError.LICENSE_ERROR, 'Cannot read license'));
|
|
|
|
debug('Auto-registering cloudron');
|
|
|
|
appstore.registerWithLicense(license.trim(), domain, function (error) {
|
|
if (error && error.reason !== BoxError.CONFLICT) { // not already registered
|
|
debug('Failed to auto-register cloudron', error);
|
|
return callback(new BoxError(BoxError.LICENSE_ERROR, 'Failed to auto-register Cloudron with license. Please contact support@cloudron.io'));
|
|
}
|
|
|
|
callback();
|
|
});
|
|
}
|
|
|
|
function unprovision(callback) {
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
debug('unprovision');
|
|
|
|
// TODO: also cancel any existing configureWebadmin task
|
|
async.series([
|
|
settings.setAdmin.bind(null, '', ''),
|
|
mail.clearDomains,
|
|
domains.clear
|
|
], callback);
|
|
}
|
|
|
|
|
|
function setup(dnsConfig, backupConfig, auditSource, callback) {
|
|
assert.strictEqual(typeof dnsConfig, 'object');
|
|
assert.strictEqual(typeof backupConfig, 'object');
|
|
assert.strictEqual(typeof auditSource, 'object');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
if (gProvisionStatus.setup.active || gProvisionStatus.restore.active) return callback(new BoxError(BoxError.BAD_STATE, 'Already setting up or restoring'));
|
|
|
|
gProvisionStatus.setup = { active: true, errorMessage: '', message: 'Adding domain' };
|
|
|
|
function done(error) {
|
|
gProvisionStatus.setup.active = false;
|
|
gProvisionStatus.setup.errorMessage = error ? error.message : '';
|
|
callback(error);
|
|
}
|
|
|
|
users.isActivated(function (error, activated) {
|
|
if (error) return done(error);
|
|
if (activated) return done(new BoxError(BoxError.CONFLICT, 'Already activated', { activate: true }));
|
|
|
|
unprovision(function (error) {
|
|
if (error) return done(error);
|
|
|
|
const domain = dnsConfig.domain.toLowerCase();
|
|
const zoneName = dnsConfig.zoneName ? dnsConfig.zoneName : (tld.getDomain(domain) || domain);
|
|
|
|
debug(`provision: Setting up Cloudron with domain ${domain} and zone ${zoneName}`);
|
|
|
|
let data = {
|
|
zoneName: zoneName,
|
|
provider: dnsConfig.provider,
|
|
config: dnsConfig.config,
|
|
fallbackCertificate: dnsConfig.fallbackCertificate || null,
|
|
tlsConfig: dnsConfig.tlsConfig || { provider: 'letsencrypt-prod' }
|
|
};
|
|
|
|
domains.add(domain, data, auditSource, function (error) {
|
|
if (error) return done(error);
|
|
|
|
callback(); // now that args are validated run the task in the background
|
|
|
|
async.series([
|
|
autoRegister.bind(null, domain),
|
|
domains.prepareDashboardDomain.bind(null, domain, auditSource, (progress) => setProgress('setup', progress.message, NOOP_CALLBACK)),
|
|
cloudron.setDashboardDomain.bind(null, domain, auditSource),
|
|
mail.addDomain.bind(null, domain), // this relies on settings.mailFqdn() and settings.adminDomain()
|
|
(next) => { if (!backupConfig) return next(); settings.setBackupConfig(backupConfig, next); },
|
|
setProgress.bind(null, 'setup', 'Done'),
|
|
eventlog.add.bind(null, eventlog.ACTION_PROVISION, auditSource, { })
|
|
], function (error) {
|
|
gProvisionStatus.setup.active = false;
|
|
gProvisionStatus.setup.errorMessage = error ? error.message : '';
|
|
});
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
function setTimeZone(ip, callback) {
|
|
assert.strictEqual(typeof ip, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
debug('setTimeZone ip:%s', ip);
|
|
|
|
superagent.get('https://geolocation.cloudron.io/json').query({ ip: ip }).timeout(10 * 1000).end(function (error, result) {
|
|
if ((error && !error.response) || result.statusCode !== 200) {
|
|
debug('Failed to get geo location: %s', error.message);
|
|
return callback(null);
|
|
}
|
|
|
|
var timezone = safe.query(result.body, 'location.time_zone');
|
|
|
|
if (!timezone || typeof timezone !== 'string') {
|
|
debug('No timezone in geoip response : %j', result.body);
|
|
return callback(null);
|
|
}
|
|
|
|
debug('Setting timezone to ', timezone);
|
|
|
|
settings.setTimeZone(timezone, callback);
|
|
});
|
|
}
|
|
|
|
function activate(username, password, email, displayName, ip, auditSource, callback) {
|
|
assert.strictEqual(typeof username, 'string');
|
|
assert.strictEqual(typeof password, 'string');
|
|
assert.strictEqual(typeof email, 'string');
|
|
assert.strictEqual(typeof displayName, 'string');
|
|
assert.strictEqual(typeof ip, 'string');
|
|
assert.strictEqual(typeof auditSource, 'object');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
debug('activating user:%s email:%s', username, email);
|
|
|
|
setTimeZone(ip, function () { }); // TODO: get this from user. note that timezone is detected based on the browser location and not the cloudron region
|
|
|
|
users.createOwner(username, password, email, displayName, auditSource, function (error, userObject) {
|
|
if (error && error.reason === BoxError.ALREADY_EXISTS) return callback(new BoxError(BoxError.CONFLICT, 'Already activated'));
|
|
if (error) return callback(error);
|
|
|
|
clients.addTokenByUserId('cid-webadmin', userObject.id, Date.now() + constants.DEFAULT_TOKEN_EXPIRATION, {}, function (error, result) {
|
|
if (error) return callback(error);
|
|
|
|
eventlog.add(eventlog.ACTION_ACTIVATE, auditSource, { });
|
|
|
|
callback(null, {
|
|
userId: userObject.id,
|
|
token: result.accessToken,
|
|
expires: result.expires
|
|
});
|
|
|
|
setImmediate(cloudron.onActivated.bind(null, NOOP_CALLBACK)); // hack for now to not block the above http response
|
|
});
|
|
});
|
|
}
|
|
|
|
function restore(backupConfig, backupId, version, auditSource, callback) {
|
|
assert.strictEqual(typeof backupConfig, 'object');
|
|
assert.strictEqual(typeof backupId, 'string');
|
|
assert.strictEqual(typeof version, 'string');
|
|
assert.strictEqual(typeof auditSource, 'object');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
if (!semver.valid(version)) return callback(new BoxError(BoxError.BAD_FIELD, 'version is not a valid semver', { field: 'version' }));
|
|
if (semver.major(constants.VERSION) !== semver.major(version) || semver.minor(constants.VERSION) !== semver.minor(version)) return callback(new BoxError(BoxError.BAD_STATE, `Run cloudron-setup with --version ${version} to restore from this backup`));
|
|
|
|
if (gProvisionStatus.setup.active || gProvisionStatus.restore.active) return callback(new BoxError(BoxError.BAD_STATE, 'Already setting up or restoring'));
|
|
|
|
gProvisionStatus.restore = { active: true, errorMessage: '', message: 'Testing backup config' };
|
|
|
|
function done(error) {
|
|
gProvisionStatus.restore.active = false;
|
|
gProvisionStatus.restore.errorMessage = error ? error.message : '';
|
|
callback(error);
|
|
}
|
|
|
|
users.isActivated(function (error, activated) {
|
|
if (error) return done(error);
|
|
if (activated) return done(new BoxError(BoxError.CONFLICT, 'Already activated. Restore with a fresh Cloudron installation.'));
|
|
|
|
backups.testConfig(backupConfig, function (error) {
|
|
if (error) return done(error);
|
|
|
|
debug(`restore: restoring from ${backupId} from provider ${backupConfig.provider} with format ${backupConfig.format}`);
|
|
|
|
callback(); // now that the fields are validated, continue task in the background
|
|
|
|
async.series([
|
|
setProgress.bind(null, 'restore', 'Downloading backup'),
|
|
backups.restore.bind(null, backupConfig, backupId, (progress) => setProgress('restore', progress.message, NOOP_CALLBACK)),
|
|
cloudron.setupDashboard.bind(null, auditSource, (progress) => setProgress('restore', progress.message, NOOP_CALLBACK)),
|
|
settings.setBackupConfig.bind(null, backupConfig), // update with the latest backupConfig
|
|
eventlog.add.bind(null, eventlog.ACTION_RESTORE, auditSource, { backupId }),
|
|
], function (error) {
|
|
gProvisionStatus.restore.active = false;
|
|
gProvisionStatus.restore.errorMessage = error ? error.message : '';
|
|
|
|
if (!error) cloudron.onActivated(NOOP_CALLBACK);
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
function getStatus(callback) {
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
users.isActivated(function (error, activated) {
|
|
if (error) return callback(error);
|
|
|
|
settings.getCloudronName(function (error, cloudronName) {
|
|
if (error) return callback(error);
|
|
|
|
callback(null, _.extend({
|
|
version: constants.VERSION,
|
|
apiServerOrigin: settings.apiServerOrigin(), // used by CaaS tool
|
|
provider: sysinfo.provider(),
|
|
cloudronName: cloudronName,
|
|
adminFqdn: settings.adminDomain() ? settings.adminFqdn() : null,
|
|
activated: activated,
|
|
}, gProvisionStatus));
|
|
});
|
|
});
|
|
}
|