Files
cloudron-box/installer/src/server.js
T

209 lines
5.9 KiB
JavaScript
Raw Normal View History

2015-08-04 16:29:49 -07:00
#!/usr/bin/env node
/* jslint node: true */
'use strict';
2015-08-25 15:06:39 -07:00
var assert = require('assert'),
2015-08-04 16:29:49 -07:00
async = require('async'),
debug = require('debug')('installer:server'),
express = require('express'),
fs = require('fs'),
http = require('http'),
HttpError = require('connect-lastmile').HttpError,
HttpSuccess = require('connect-lastmile').HttpSuccess,
installer = require('./installer.js'),
json = require('body-parser').json,
lastMile = require('connect-lastmile'),
morgan = require('morgan'),
2016-10-21 15:53:12 +02:00
path = require('path'),
2016-10-06 14:14:48 +02:00
safe = require('safetydance');
2015-08-04 16:29:49 -07:00
exports = module.exports = {
start: start,
stop: stop
};
var PROVISION_CONFIG_FILE_JSON = '/root/userdata.json';
var PROVISION_CONFIG_FILE_JS = '/root/userdata.js';
2015-12-20 10:25:12 +01:00
var CLOUDRON_CONFIG_FILE = '/home/yellowtent/configs/cloudron.conf';
2016-10-21 11:49:56 +02:00
var BOX_VERSIONS_URL = 'https://s3.amazonaws.com/prod-cloudron-releases/versions.json';
2016-01-25 17:46:21 -08:00
var gHttpServer = null; // update server; used for updates
2016-10-21 15:53:12 +02:00
var gSetupServer = null; // setup server; only used for initial setup
2015-08-04 16:29:49 -07:00
2016-06-10 10:52:26 +02:00
function provision(callback) {
2016-08-19 17:33:55 -07:00
if (fs.existsSync(CLOUDRON_CONFIG_FILE)) {
debug('provision: already provisioned');
return callback(null); // already provisioned
}
2016-06-10 10:52:26 +02:00
var userData = null;
2016-10-06 14:14:48 +02:00
function retry(error) {
if (error) console.error(error);
2016-10-21 15:53:12 +02:00
if (!gSetupServer) startSetupServer(function () { debug('Setup Server started.'); });
setTimeout(provision.bind(null, callback), 5000);
}
// check for .json file then .js
if (fs.existsSync(PROVISION_CONFIG_FILE_JSON)) {
userData = safe.require(PROVISION_CONFIG_FILE_JSON);
if (!userData) return retry('Provisioning data invalid');
} else if (fs.existsSync(PROVISION_CONFIG_FILE_JS)) {
var tmp = safe.require(PROVISION_CONFIG_FILE_JS);
if (!tmp) return retry('Provisioning data invalid');
// translate to expected format
2016-10-21 11:49:56 +02:00
userData = { data: tmp };
}
2016-06-13 14:55:02 +02:00
if (!userData) return retry('No user data file found. Waiting for it...');
// validate the bare minimum
if (!userData.data || typeof userData.data !== 'object') return retry('user data misses "data" object');
if (!userData.data.fqdn || typeof userData.data.fqdn !== 'string') return retry('fqdn in user data has to be a non-empty string');
2016-10-21 11:49:56 +02:00
// set the fallback
if (!userData.data.boxVersionsUrl) userData.data.boxVersionsUrl = BOX_VERSIONS_URL;
if (typeof userData.data.boxVersionsUrl !== 'string') return retry('boxVersionsUrl in user data has to be a non-empty string');
2016-10-21 15:53:12 +02:00
stopSetupServer(function () { debug('Setup Server stopped.'); });
installer.provision(userData, callback);
2015-08-30 21:10:57 -07:00
}
2015-08-25 15:54:15 -07:00
function update(req, res, next) {
2015-08-04 16:29:49 -07:00
assert.strictEqual(typeof req.body, 'object');
if (!req.body.sourceTarballUrl || typeof req.body.sourceTarballUrl !== 'string') return next(new HttpError(400, 'No sourceTarballUrl provided'));
if (!req.body.data || typeof req.body.data !== 'object') return next(new HttpError(400, 'No data provided'));
2015-08-04 16:29:49 -07:00
2015-08-25 15:54:15 -07:00
debug('provision: received from box %j', req.body);
2015-08-04 16:29:49 -07:00
2015-08-27 18:49:52 -07:00
installer.provision(req.body, function (error) {
if (error) console.error(error);
2015-08-25 15:54:15 -07:00
});
2015-08-27 18:49:52 -07:00
next(new HttpSuccess(202, { }));
2015-08-04 16:29:49 -07:00
}
function startUpdateServer(callback) {
assert.strictEqual(typeof callback, 'function');
debug('Starting update server');
var app = express();
var router = new express.Router();
if (process.env.NODE_ENV !== 'test') app.use(morgan('dev', { immediate: false }));
app.use(json({ strict: true }))
.use(router)
.use(lastMile());
2015-08-25 15:54:15 -07:00
router.post('/api/v1/installer/update', update);
2015-08-04 16:29:49 -07:00
gHttpServer = http.createServer(app);
gHttpServer.on('error', console.error);
gHttpServer.listen(2020, '127.0.0.1', callback);
}
function stopUpdateServer(callback) {
assert.strictEqual(typeof callback, 'function');
debug('Stopping update server');
if (!gHttpServer) return callback(null);
gHttpServer.close(callback);
gHttpServer = null;
}
2016-10-21 15:53:12 +02:00
function setup(req, res, next) {
assert.strictEqual(typeof req.body, 'object');
if (!req.body.fqdn || typeof req.body.fqdn !== 'string') return next(new HttpError(400, 'No fqdn provided'));
debug('setup: received from box %j', req.body);
var data = {
fqdn: req.body.fqdn
};
if (req.body.boxVersionsUrl && typeof req.body.boxVersionsUrl === 'string') data.boxVersionsUrl = req.body.boxVersionsUrl;
fs.writeFile(PROVISION_CONFIG_FILE_JSON, JSON.stringify({ data: data }), function (error) {
if (error) return next(new HttpError(500, error));
next(new HttpSuccess(201, {}));
});
}
function startSetupServer(callback) {
assert.strictEqual(typeof callback, 'function');
debug('Starting setup server');
var app = express();
var router = new express.Router();
if (process.env.NODE_ENV !== 'test') app.use(morgan('dev', { immediate: false }));
app.use(express.static(path.join(__dirname, '../www/')))
.use(json({ strict: true }))
.use(router)
.use(lastMile());
router.post('/api/v1/setup', setup);
gSetupServer = http.createServer(app);
gSetupServer.on('error', console.error);
2016-10-21 16:04:08 -07:00
gSetupServer.listen(80, '0.0.0.0', callback);
2016-10-21 15:53:12 +02:00
}
function stopSetupServer(callback) {
assert.strictEqual(typeof callback, 'function');
debug('Stopping setup server');
if (!gSetupServer) return callback(null);
gSetupServer.close(callback);
gSetupServer = null;
}
2015-08-04 16:29:49 -07:00
function start(callback) {
assert.strictEqual(typeof callback, 'function');
2016-10-05 14:35:40 +02:00
debug('Starting Installer');
2016-10-05 14:35:40 +02:00
var actions = [
startUpdateServer,
provision
];
async.series(actions, callback);
2015-08-04 16:29:49 -07:00
}
function stop(callback) {
assert.strictEqual(typeof callback, 'function');
async.series([
2016-01-25 17:46:21 -08:00
stopUpdateServer
2015-08-04 16:29:49 -07:00
], callback);
}
if (require.main === module) {
start(function (error) {
if (error) console.error(error);
});
}