2017-03-06 22:13:40 +08:00
|
|
|
'use strict';
|
|
|
|
|
2017-07-16 19:37:33 +08:00
|
|
|
const config = require('wild-config');
|
2017-03-06 22:13:40 +08:00
|
|
|
const restify = require('restify');
|
|
|
|
const log = require('npmlog');
|
2017-04-21 04:32:19 +08:00
|
|
|
const UserHandler = require('./lib/user-handler');
|
2017-07-21 02:33:41 +08:00
|
|
|
const MailboxHandler = require('./lib/mailbox-handler');
|
2017-07-21 21:29:57 +08:00
|
|
|
const MessageHandler = require('./lib/message-handler');
|
2017-07-18 22:38:05 +08:00
|
|
|
const ImapNotifier = require('./lib/imap-notifier');
|
2017-03-27 15:36:45 +08:00
|
|
|
const db = require('./lib/db');
|
2017-10-05 20:14:43 +08:00
|
|
|
const certs = require('./lib/certs');
|
2017-07-26 16:52:55 +08:00
|
|
|
|
|
|
|
const usersRoutes = require('./lib/api/users');
|
|
|
|
const addressesRoutes = require('./lib/api/addresses');
|
|
|
|
const mailboxesRoutes = require('./lib/api/mailboxes');
|
|
|
|
const messagesRoutes = require('./lib/api/messages');
|
|
|
|
const filtersRoutes = require('./lib/api/filters');
|
|
|
|
const aspsRoutes = require('./lib/api/asps');
|
2017-10-10 16:19:10 +08:00
|
|
|
const totpRoutes = require('./lib/api/2fa/totp');
|
|
|
|
const u2fRoutes = require('./lib/api/2fa/u2f');
|
2017-07-26 16:52:55 +08:00
|
|
|
const updatesRoutes = require('./lib/api/updates');
|
|
|
|
const authRoutes = require('./lib/api/auth');
|
2017-07-30 23:07:35 +08:00
|
|
|
const autoreplyRoutes = require('./lib/api/autoreply');
|
2017-03-06 22:13:40 +08:00
|
|
|
|
2017-07-17 21:32:31 +08:00
|
|
|
const serverOptions = {
|
2017-07-19 16:06:47 +08:00
|
|
|
name: 'Wild Duck API',
|
2017-07-20 18:10:43 +08:00
|
|
|
strictRouting: true,
|
|
|
|
formatters: {
|
|
|
|
'application/json; q=0.4': (req, res, body) => {
|
2017-07-20 21:10:36 +08:00
|
|
|
let data = body ? JSON.stringify(body, false, 2) + '\n' : 'null';
|
2017-07-20 18:10:43 +08:00
|
|
|
res.setHeader('Content-Length', Buffer.byteLength(data));
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
}
|
2017-07-17 21:32:31 +08:00
|
|
|
};
|
|
|
|
|
2017-10-05 20:14:43 +08:00
|
|
|
let certOptions = {};
|
|
|
|
certs.loadTLSOptions(certOptions, 'api');
|
|
|
|
|
|
|
|
if (config.api.secure && certOptions.key) {
|
|
|
|
serverOptions.key = certOptions.key;
|
|
|
|
if (certOptions.ca) {
|
|
|
|
serverOptions.ca = certOptions.ca;
|
2017-07-17 21:32:31 +08:00
|
|
|
}
|
2017-10-05 20:14:43 +08:00
|
|
|
serverOptions.certificate = certOptions.cert;
|
2017-07-17 21:32:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const server = restify.createServer(serverOptions);
|
2017-03-06 22:13:40 +08:00
|
|
|
|
2017-04-21 04:32:19 +08:00
|
|
|
let userHandler;
|
2017-07-21 02:33:41 +08:00
|
|
|
let mailboxHandler;
|
2017-07-21 21:29:57 +08:00
|
|
|
let messageHandler;
|
2017-07-18 22:38:05 +08:00
|
|
|
let notifier;
|
2017-03-06 22:13:40 +08:00
|
|
|
|
2017-07-20 18:10:43 +08:00
|
|
|
// disable compression for EventSource response
|
|
|
|
// this needs to be called before gzipResponse
|
|
|
|
server.use((req, res, next) => {
|
|
|
|
if (req.route.path === '/users/:user/updates') {
|
|
|
|
req.headers['accept-encoding'] = '';
|
|
|
|
}
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
server.use(restify.plugins.gzipResponse());
|
|
|
|
|
2017-07-11 03:15:16 +08:00
|
|
|
server.use(restify.plugins.queryParser());
|
2017-06-03 14:51:58 +08:00
|
|
|
server.use(
|
2017-07-11 03:15:16 +08:00
|
|
|
restify.plugins.bodyParser({
|
2017-06-03 14:51:58 +08:00
|
|
|
maxBodySize: 0,
|
|
|
|
mapParams: true,
|
|
|
|
mapFiles: false,
|
|
|
|
overrideParams: false
|
|
|
|
})
|
|
|
|
);
|
2017-07-19 16:06:47 +08:00
|
|
|
server.get(
|
|
|
|
/\/public\/?.*/,
|
|
|
|
restify.plugins.serveStatic({
|
|
|
|
directory: __dirname,
|
|
|
|
default: 'index.html'
|
|
|
|
})
|
|
|
|
);
|
2017-03-06 22:13:40 +08:00
|
|
|
|
2017-07-25 20:50:16 +08:00
|
|
|
server.use((req, res, next) => {
|
2017-07-28 21:34:22 +08:00
|
|
|
if (config.api.accessToken && ![req.query.accessToken, req.headers['x-access-token']].includes(config.api.accessToken)) {
|
2017-07-25 20:50:16 +08:00
|
|
|
res.status(403);
|
|
|
|
res.charSet('utf-8');
|
|
|
|
return res.json({
|
|
|
|
error: 'Invalid accessToken value'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (req.query.accessToken) {
|
|
|
|
delete req.query.accessToken;
|
|
|
|
}
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
module.exports = done => {
|
2017-04-13 16:35:39 +08:00
|
|
|
if (!config.imap.enabled) {
|
|
|
|
return setImmediate(() => done(null, false));
|
|
|
|
}
|
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
let started = false;
|
|
|
|
|
2017-07-18 22:38:05 +08:00
|
|
|
notifier = new ImapNotifier({
|
|
|
|
database: db.database,
|
|
|
|
redis: db.redis
|
|
|
|
});
|
2017-08-07 02:25:10 +08:00
|
|
|
|
2017-08-04 19:07:17 +08:00
|
|
|
messageHandler = new MessageHandler({
|
|
|
|
database: db.database,
|
|
|
|
users: db.users,
|
2017-08-07 16:29:29 +08:00
|
|
|
redis: db.redis,
|
2017-08-04 19:07:17 +08:00
|
|
|
gridfs: db.gridfs,
|
2017-08-07 16:29:29 +08:00
|
|
|
attachments: config.attachments
|
2017-08-04 19:07:17 +08:00
|
|
|
});
|
2017-08-07 02:25:10 +08:00
|
|
|
|
2017-08-04 19:07:17 +08:00
|
|
|
userHandler = new UserHandler({
|
|
|
|
database: db.database,
|
|
|
|
users: db.users,
|
|
|
|
redis: db.redis,
|
2017-08-08 19:35:18 +08:00
|
|
|
messageHandler,
|
|
|
|
authlogExpireDays: config.log.authlogExpireDays
|
2017-08-04 19:07:17 +08:00
|
|
|
});
|
2017-08-07 02:25:10 +08:00
|
|
|
|
2017-08-04 19:07:17 +08:00
|
|
|
mailboxHandler = new MailboxHandler({
|
|
|
|
database: db.database,
|
|
|
|
users: db.users,
|
|
|
|
redis: db.redis,
|
|
|
|
notifier
|
|
|
|
});
|
2017-03-27 15:36:45 +08:00
|
|
|
|
2017-07-26 16:52:55 +08:00
|
|
|
usersRoutes(db, server, userHandler);
|
|
|
|
addressesRoutes(db, server);
|
|
|
|
mailboxesRoutes(db, server, mailboxHandler);
|
|
|
|
messagesRoutes(db, server, messageHandler);
|
|
|
|
filtersRoutes(db, server);
|
|
|
|
aspsRoutes(db, server, userHandler);
|
2017-10-10 16:19:10 +08:00
|
|
|
totpRoutes(db, server, userHandler);
|
|
|
|
u2fRoutes(db, server, userHandler);
|
2017-07-26 16:52:55 +08:00
|
|
|
updatesRoutes(db, server, notifier);
|
|
|
|
authRoutes(db, server, userHandler);
|
2017-07-30 23:07:35 +08:00
|
|
|
autoreplyRoutes(db, server);
|
2017-07-26 16:52:55 +08:00
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
server.on('error', err => {
|
|
|
|
if (!started) {
|
|
|
|
started = true;
|
|
|
|
return done(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
log.error('API', err);
|
|
|
|
});
|
|
|
|
|
|
|
|
server.listen(config.api.port, config.api.host, () => {
|
|
|
|
if (started) {
|
|
|
|
return server.close();
|
|
|
|
}
|
|
|
|
started = true;
|
|
|
|
log.info('API', 'Server listening on %s:%s', config.api.host || '0.0.0.0', config.api.port);
|
|
|
|
done(null, server);
|
2017-03-06 22:13:40 +08:00
|
|
|
});
|
|
|
|
};
|