|
|
|
const fs = require('fs');
|
|
|
|
const os = require('os');
|
|
|
|
const path = require('path');
|
|
|
|
const http = require('http');
|
|
|
|
const Koa = require('koa');
|
|
|
|
const Body = require('koa-body');
|
|
|
|
const Router = require('koa-router');
|
|
|
|
const cache = require('koa-static-cache');
|
|
|
|
const nunjucks = require('nunjucks');
|
|
|
|
const mongodb = require('mongodb');
|
|
|
|
|
|
|
|
class Loader {
|
|
|
|
getSource(name) { // eslint-disable-line class-methods-use-this
|
|
|
|
return {
|
|
|
|
src: `
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html data-page="setup" data-layout="immersive" class="layout--immersive page--setup nojs">
|
|
|
|
<head>
|
|
|
|
<meta charset="UTF-8">
|
|
|
|
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
|
|
|
|
<meta http-equiv="X-UA-Compatible" content="chrome=1"/>
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1" />
|
|
|
|
<meta name="msapplication-TileColor" content="#579e9a">
|
|
|
|
<meta name="theme-color" content="#56758f">
|
|
|
|
<link rel="stylesheet" media="all" href="/vj4.css">
|
|
|
|
<title>{{ _('Setup') }}</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<div class="slideout-panel" id="panel">
|
|
|
|
<div class="main">
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<div class="immersive--content immersive--center">
|
|
|
|
<h1>{{ _('Setup') }}</h1>
|
|
|
|
<form method="POST">
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<label class="inverse material textbox">
|
|
|
|
{{ _('Database Host') }}
|
|
|
|
<input name="host" type="text" value="127.0.0.1" autofocus>
|
|
|
|
</label>
|
|
|
|
</div></div>
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<label class="inverse material textbox">
|
|
|
|
{{ _('Database Port') }}
|
|
|
|
<input name="port" type="number" value="27017">
|
|
|
|
</label>
|
|
|
|
</div></div>
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<label class="inverse material textbox">
|
|
|
|
{{ _('Database Name') }}
|
|
|
|
<input name="name" type="text" value="hydro">
|
|
|
|
</label>
|
|
|
|
</div></div>
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<label class="inverse material textbox">
|
|
|
|
{{ _('Database Username') }}
|
|
|
|
<input name="username" type="text" placeholder="{{ _('Leave blank if none') }}">
|
|
|
|
</label>
|
|
|
|
</div></div>
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<label class="inverse material textbox">
|
|
|
|
{{ _('Database Password') }}
|
|
|
|
<input name="password" type="password" placeholder="{{ _('Leave blank if none') }}">
|
|
|
|
</label>
|
|
|
|
</div></div>
|
|
|
|
<div class="row"><div class="columns">
|
|
|
|
<div class="text-center">
|
|
|
|
<input type="submit" value="{{ _('Confirm') }}" class="inverse expanded rounded primary button">
|
|
|
|
</div>
|
|
|
|
</div></div>
|
|
|
|
</form>
|
|
|
|
</div>
|
|
|
|
</div></div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
`,
|
|
|
|
path: name,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const env = new nunjucks.Environment(new Loader(), { autoescape: true, trimBlocks: true });
|
|
|
|
|
|
|
|
function render(name) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
env.render(name, {
|
|
|
|
_: (str) => str,
|
|
|
|
}, (err, res) => {
|
|
|
|
if (err) reject(err);
|
|
|
|
else resolve(res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function setup() {
|
|
|
|
const app = new Koa();
|
|
|
|
const server = http.createServer(app.callback());
|
|
|
|
const router = new Router();
|
|
|
|
app.keys = ['Hydro'];
|
|
|
|
app.use(cache(path.join(os.tmpdir(), 'hydro', 'builtin'), {
|
|
|
|
maxAge: 365 * 24 * 60 * 60,
|
|
|
|
}));
|
|
|
|
app.use(Body());
|
|
|
|
router.get('/', async (ctx) => {
|
|
|
|
ctx.body = await render('setup.html');
|
|
|
|
ctx.response.type = 'text/html';
|
|
|
|
});
|
|
|
|
const p = new Promise((resolve) => {
|
|
|
|
router.post('/', async (ctx) => {
|
|
|
|
const {
|
|
|
|
host, port, name, username, password,
|
|
|
|
} = ctx.request.body;
|
|
|
|
let mongourl = 'mongodb://';
|
|
|
|
if (username) mongourl += `${username}:${password}@`;
|
|
|
|
mongourl += `${host}:${port}/${name}`;
|
|
|
|
try {
|
|
|
|
const Database = await mongodb.MongoClient.connect(mongourl, {
|
|
|
|
useNewUrlParser: true, useUnifiedTopology: true,
|
|
|
|
});
|
|
|
|
const db = Database.db(name);
|
|
|
|
await Promise.all([
|
|
|
|
db.collection('system').updateOne(
|
|
|
|
{ _id: 'server.host' },
|
|
|
|
{ $set: { value: ctx.request.host } },
|
|
|
|
{ upsert: true },
|
|
|
|
),
|
|
|
|
db.collection('system').updateOne(
|
|
|
|
{ _id: 'server.hostname' },
|
|
|
|
{ $set: { value: ctx.request.hostname } },
|
|
|
|
{ upsert: true },
|
|
|
|
),
|
|
|
|
db.collection('system').updateOne(
|
|
|
|
{ _id: 'server.url' },
|
|
|
|
{ $set: { value: ctx.request.href } },
|
|
|
|
{ upsert: true },
|
|
|
|
),
|
|
|
|
]);
|
|
|
|
fs.writeFileSync(path.resolve(process.cwd(), 'config.json'), JSON.stringify({
|
|
|
|
host, port, name, username, password,
|
|
|
|
}));
|
|
|
|
ctx.redirect('/');
|
|
|
|
resolve();
|
|
|
|
} catch (e) {
|
|
|
|
ctx.body = `Error connecting to database: ${e.message}\n${e.stack}`;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
app.use(router.routes()).use(router.allowedMethods());
|
|
|
|
server.listen(8888);
|
|
|
|
console.log('Server listening at: 8888');
|
|
|
|
await p;
|
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
server.close((err) => {
|
|
|
|
if (err) reject(err);
|
|
|
|
else resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
console.log('Done! Please restart.');
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = { setup };
|