2017-04-23 20:02:26 +02:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
let async = require('async');
|
|
|
|
let bodyParser = require('body-parser');
|
|
|
|
let Cookies = require('cookies');
|
|
|
|
let express = require('express');
|
|
|
|
let fs = require('fs');
|
|
|
|
let Handlebars = require('handlebars');
|
|
|
|
let renderReadme = require('render-readme');
|
|
|
|
let Search = require('./search');
|
|
|
|
let Middleware = require('./middleware');
|
2017-05-17 00:20:33 +02:00
|
|
|
let Utils = require('./utils');
|
2017-04-23 20:02:26 +02:00
|
|
|
let match = Middleware.match;
|
|
|
|
let validate_name = Middleware.validate_name;
|
|
|
|
let validate_pkg = Middleware.validate_package;
|
2014-11-04 17:47:03 +03:00
|
|
|
|
2015-04-08 23:54:59 +03:00
|
|
|
module.exports = function(config, auth, storage) {
|
2017-04-23 20:02:26 +02:00
|
|
|
let app = express.Router();
|
|
|
|
let can = Middleware.allow(auth);
|
2014-11-13 18:52:13 +03:00
|
|
|
|
2014-11-13 20:13:37 +03:00
|
|
|
// validate all of these params as a package name
|
|
|
|
// this might be too harsh, so ask if it causes trouble
|
2017-04-23 20:02:26 +02:00
|
|
|
app.param('package', validate_pkg);
|
|
|
|
app.param('filename', validate_name);
|
|
|
|
app.param('version', validate_name);
|
|
|
|
app.param('anything', match(/.*/));
|
|
|
|
|
|
|
|
app.use(Cookies.express());
|
|
|
|
app.use(bodyParser.urlencoded({extended: false}));
|
|
|
|
app.use(auth.cookie_middleware());
|
2014-11-12 14:14:37 +03:00
|
|
|
app.use(function(req, res, next) {
|
|
|
|
// disable loading in frames (clickjacking, etc.)
|
2017-04-23 20:02:26 +02:00
|
|
|
res.header('X-Frame-Options', 'deny');
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
|
|
|
|
Search.configureStorage(storage);
|
|
|
|
|
|
|
|
Handlebars.registerPartial('entry', fs.readFileSync(require.resolve('./GUI/entry.hbs'), 'utf8'));
|
|
|
|
let template;
|
|
|
|
if (config.web && config.web.template) {
|
|
|
|
template = Handlebars.compile(fs.readFileSync(config.web.template, 'utf8'));
|
|
|
|
} else {
|
|
|
|
template = Handlebars.compile(fs.readFileSync(require.resolve('./GUI/index.hbs'), 'utf8'));
|
2015-02-25 19:15:36 -03:00
|
|
|
}
|
2014-11-12 14:14:37 +03:00
|
|
|
app.get('/', function(req, res, next) {
|
2017-05-17 00:20:33 +02:00
|
|
|
let base = Utils.get_base_url(req.protocol, req.get('host'), config.url_prefix);
|
2017-04-23 20:02:26 +02:00
|
|
|
res.setHeader('Content-Type', 'text/html');
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
storage.get_local(function(err, packages) {
|
2017-04-23 20:02:26 +02:00
|
|
|
if (err) throw err; // that function shouldn't produce any
|
|
|
|
async.filterSeries(packages, function(pkg, cb) {
|
|
|
|
auth.allow_access(pkg.name, req.remote_user, function(err, allowed) {
|
|
|
|
setImmediate(function() {
|
2016-09-04 13:03:23 -04:00
|
|
|
if (err) {
|
|
|
|
cb(null, false);
|
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
cb(err, allowed);
|
2016-09-04 13:03:23 -04:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
|
|
|
});
|
2016-04-21 21:13:49 +01:00
|
|
|
}, function(err, packages) {
|
2017-04-23 20:02:26 +02:00
|
|
|
if (err) throw err;
|
2015-09-24 11:24:23 +03:00
|
|
|
packages.sort(function(p1, p2) {
|
|
|
|
if (p1.name < p2.name) {
|
|
|
|
return -1;
|
2017-04-23 20:02:26 +02:00
|
|
|
} else {
|
2015-09-24 11:24:23 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-02-24 22:21:57 -05:00
|
|
|
next(template({
|
2017-04-23 20:02:26 +02:00
|
|
|
name: config.web && config.web.title ? config.web.title : 'Verdaccio',
|
|
|
|
tagline: config.web && config.web.tagline ? config.web.tagline : '',
|
|
|
|
packages: packages,
|
|
|
|
baseUrl: base,
|
|
|
|
username: req.remote_user.name,
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
// Static
|
|
|
|
app.get('/-/static/:filename', function(req, res, next) {
|
2017-04-23 20:02:26 +02:00
|
|
|
let file = __dirname + '/static/' + req.params.filename;
|
2014-11-13 19:15:50 +03:00
|
|
|
res.sendFile(file, function(err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
if (!err) return;
|
2014-11-12 19:25:33 +03:00
|
|
|
if (err.status === 404) {
|
2017-04-23 20:02:26 +02:00
|
|
|
next();
|
2014-11-12 14:14:37 +03:00
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
next(err);
|
2014-11-12 14:14:37 +03:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
app.get('/-/logo', function(req, res, next) {
|
2015-03-29 22:59:08 +03:00
|
|
|
res.sendFile( config.web && config.web.logo
|
|
|
|
? config.web.logo
|
2017-04-23 20:02:26 +02:00
|
|
|
: __dirname + '/static/logo-sm.png' );
|
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
app.post('/-/login', function(req, res, next) {
|
2014-11-16 15:37:50 +03:00
|
|
|
auth.authenticate(req.body.user, req.body.pass, function(err, user) {
|
|
|
|
if (!err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
req.remote_user = user;
|
|
|
|
// res.cookies.set('token', auth.issue_token(req.remote_user))
|
2014-11-24 22:46:37 +03:00
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
let str = req.body.user + ':' + req.body.pass;
|
|
|
|
res.cookies.set('token', auth.aes_encrypt(str).toString('base64'));
|
2014-11-16 15:37:50 +03:00
|
|
|
}
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
let base = config.url_prefix
|
2015-03-29 22:59:08 +03:00
|
|
|
? config.url_prefix.replace(/\/$/, '')
|
2017-04-23 20:02:26 +02:00
|
|
|
: req.protocol + '://' + req.get('host');
|
|
|
|
res.redirect(base);
|
|
|
|
});
|
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
app.post('/-/logout', function(req, res, next) {
|
2017-04-23 20:02:26 +02:00
|
|
|
let base = config.url_prefix
|
2015-03-29 22:59:08 +03:00
|
|
|
? config.url_prefix.replace(/\/$/, '')
|
2017-04-23 20:02:26 +02:00
|
|
|
: req.protocol + '://' + req.get('host');
|
|
|
|
res.cookies.set('token', '');
|
|
|
|
res.redirect(base);
|
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
// Search
|
|
|
|
app.get('/-/search/:anything', function(req, res, next) {
|
2017-04-23 20:02:26 +02:00
|
|
|
const results = Search.query(req.params.anything);
|
|
|
|
const packages = [];
|
2014-11-12 14:14:37 +03:00
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
const getData = function(i) {
|
2014-11-12 14:14:37 +03:00
|
|
|
storage.get_package(results[i].ref, function(err, entry) {
|
|
|
|
if (!err && entry) {
|
2017-04-20 16:50:48 +08:00
|
|
|
auth.allow_access(entry.name, req.remote_user, function(err, allowed) { // TODO: This may cause performance issue?
|
2017-04-23 20:02:26 +02:00
|
|
|
if (err || !allowed) return;
|
2017-04-20 16:50:48 +08:00
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
packages.push(entry.versions[entry['dist-tags'].latest]);
|
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (i >= results.length - 1) {
|
2017-04-23 20:02:26 +02:00
|
|
|
next(packages);
|
2014-11-12 14:14:37 +03:00
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
getData(i + 1);
|
2014-11-12 14:14:37 +03:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
|
|
|
};
|
2014-11-12 14:14:37 +03:00
|
|
|
|
|
|
|
if (results.length) {
|
2017-04-23 20:02:26 +02:00
|
|
|
getData(0);
|
2014-11-12 14:14:37 +03:00
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
next([]);
|
2014-11-12 14:14:37 +03:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
2014-11-12 14:14:37 +03:00
|
|
|
|
2017-03-15 08:57:24 -05:00
|
|
|
app.get('/-/readme(/@:scope?)?/:package/:version?', can('access'), function(req, res, next) {
|
2017-04-23 20:02:26 +02:00
|
|
|
let packageName = req.params.package;
|
|
|
|
if (req.params.scope) packageName = '@'+ req.params.scope + '/' + packageName;
|
2017-03-15 08:57:24 -05:00
|
|
|
storage.get_package(packageName, {req: req}, function(err, info) {
|
2017-04-23 20:02:26 +02:00
|
|
|
if (err) return next(err);
|
|
|
|
next( renderReadme(info.readme || 'ERROR: No README data found!') );
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return app;
|
|
|
|
};
|