update DKIM handling

This commit is contained in:
Andris Reinman 2018-09-11 11:13:53 +03:00
parent c828893c9c
commit 1f9ae407a9
8 changed files with 263 additions and 114 deletions

View file

@ -31,6 +31,13 @@
"read:any": ["*"], "read:any": ["*"],
"update:any": ["*"], "update:any": ["*"],
"delete:any": ["*"] "delete:any": ["*"]
},
"dkim": {
"create:any": ["*"],
"read:any": ["*"],
"update:any": ["*"],
"delete:any": ["*"]
} }
}, },

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View file

@ -1 +1 @@
define({ "name": "wildduck", "version": "1.0.0", "description": "WildDuck API docs", "title": "WildDuck API", "url": "https://api.wildduck.email", "sampleUrl": false, "defaultVersion": "0.0.0", "apidoc": "0.3.0", "generator": { "name": "apidoc", "time": "2018-09-10T11:13:03.088Z", "url": "http://apidocjs.com", "version": "0.17.6" } }); define({ "name": "wildduck", "version": "1.0.0", "description": "WildDuck API docs", "title": "WildDuck API", "url": "https://api.wildduck.email", "sampleUrl": false, "defaultVersion": "0.0.0", "apidoc": "0.3.0", "generator": { "name": "apidoc", "time": "2018-09-11T08:09:57.476Z", "url": "http://apidocjs.com", "version": "0.17.6" } });

View file

@ -1 +1 @@
{ "name": "wildduck", "version": "1.0.0", "description": "WildDuck API docs", "title": "WildDuck API", "url": "https://api.wildduck.email", "sampleUrl": false, "defaultVersion": "0.0.0", "apidoc": "0.3.0", "generator": { "name": "apidoc", "time": "2018-09-10T11:13:03.088Z", "url": "http://apidocjs.com", "version": "0.17.6" } } { "name": "wildduck", "version": "1.0.0", "description": "WildDuck API docs", "title": "WildDuck API", "url": "https://api.wildduck.email", "sampleUrl": false, "defaultVersion": "0.0.0", "apidoc": "0.3.0", "generator": { "name": "apidoc", "time": "2018-09-11T08:09:57.476Z", "url": "http://apidocjs.com", "version": "0.17.6" } }

View file

@ -6,6 +6,8 @@ const MongoPaging = require('mongo-cursor-pagination');
const ObjectID = require('mongodb').ObjectID; const ObjectID = require('mongodb').ObjectID;
const DkimHandler = require('../dkim-handler'); const DkimHandler = require('../dkim-handler');
const tools = require('../tools'); const tools = require('../tools');
const util = require('util');
const roles = require('../roles');
module.exports = (db, server) => { module.exports = (db, server) => {
const dkimHandler = new DkimHandler({ const dkimHandler = new DkimHandler({
@ -16,6 +18,10 @@ module.exports = (db, server) => {
database: db.database database: db.database
}); });
const setDkim = util.promisify(dkimHandler.set.bind(dkimHandler));
const getDkim = util.promisify(dkimHandler.get.bind(dkimHandler));
const delDkim = util.promisify(dkimHandler.del.bind(dkimHandler));
/** /**
* @api {get} /dkim List registered DKIM keys * @api {get} /dkim List registered DKIM keys
* @apiName GetDkim * @apiName GetDkim
@ -120,6 +126,9 @@ module.exports = (db, server) => {
return next(); return next();
} }
// permissions check
req.validate(roles.can(req.role).readAny('dkim'));
let query = result.value.query; let query = result.value.query;
let limit = result.value.limit; let limit = result.value.limit;
let page = result.value.page; let page = result.value.page;
@ -187,6 +196,110 @@ module.exports = (db, server) => {
}) })
); );
/**
* @api {get} /dkim/resolve/:domain Resolve ID for a DKIM domain
* @apiName ResolveDKIM
* @apiGroup DKIM
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
* @apiHeaderExample {json} Header-Example:
* {
* "X-Access-Token": "59fc66a03e54454869460e45"
* }
*
* @apiParam {String} domain DKIM domain
*
* @apiSuccess {Boolean} success Indicates successful response
* @apiSuccess {String} id DKIM unique ID (24 byte hex)
*
* @apiError error Description of the error
*
* @apiExample {curl} Example usage:
* curl -i http://localhost:8080/dkim/resolve/example.com
*
* @apiSuccessExample {json} Success-Response:
* HTTP/1.1 200 OK
* {
* "success": true,
* "id": "59fc66a03e54454869460e45"
* }
*
* @apiErrorExample {json} Error-Response:
* HTTP/1.1 200 OK
* {
* "error": "This domain does not exist"
* }
*/
server.get(
'/dkim/resolve/:domain',
tools.asyncifyJson(async (req, res, next) => {
res.charSet('utf-8');
const schema = Joi.object().keys({
domain: Joi.string()
.max(255)
//.hostname()
.required(),
sess: Joi.string().max(255),
ip: Joi.string().ip({
version: ['ipv4', 'ipv6'],
cidr: 'forbidden'
})
});
const result = Joi.validate(req.params, schema, {
abortEarly: false,
convert: true
});
if (result.error) {
res.json({
error: result.error.message,
code: 'InputValidationError'
});
return next();
}
// permissions check
req.validate(roles.can(req.role).readAny('dkim'));
let domain = tools.normalizeDomain(result.value.domain);
let dkimData;
try {
dkimData = await db.database.collection('dkim').findOne(
{
domain
},
{
projection: { _id: 1 }
}
);
} catch (err) {
res.json({
error: 'MongoDB Error: ' + err.message,
code: 'InternalDatabaseError'
});
return next();
}
if (!dkimData) {
res.json({
error: 'This domain does not exist',
code: 'DkimNotFound'
});
return next();
}
res.json({
success: true,
id: dkimData._id
});
return next();
})
);
/** /**
* @api {post} /dkim Create or update DKIM key for domain * @api {post} /dkim Create or update DKIM key for domain
* @apiName PostDkim * @apiName PostDkim
@ -248,62 +361,72 @@ module.exports = (db, server) => {
* "error": "This user does not exist" * "error": "This user does not exist"
* } * }
*/ */
server.post('/dkim', (req, res, next) => { server.post(
res.charSet('utf-8'); '/dkim',
tools.asyncifyJson(async (req, res, next) => {
res.charSet('utf-8');
const schema = Joi.object().keys({ const schema = Joi.object().keys({
domain: Joi.string() domain: Joi.string()
.max(255) .max(255)
//.hostname() //.hostname()
.required(), .required(),
selector: Joi.string() selector: Joi.string()
.max(255) .max(255)
//.hostname() //.hostname()
.trim() .trim()
.required(), .required(),
privateKey: Joi.string() privateKey: Joi.string()
.empty('') .empty('')
.trim() .trim()
.regex(/^-----BEGIN (RSA )?PRIVATE KEY-----/, 'DKIM key format'), .regex(/^-----BEGIN (RSA )?PRIVATE KEY-----/, 'DKIM key format'),
description: Joi.string() description: Joi.string()
.max(255) .max(255)
//.hostname() //.hostname()
.trim(), .trim(),
sess: Joi.string().max(255), sess: Joi.string().max(255),
ip: Joi.string().ip({ ip: Joi.string().ip({
version: ['ipv4', 'ipv6'], version: ['ipv4', 'ipv6'],
cidr: 'forbidden' cidr: 'forbidden'
}) })
});
const result = Joi.validate(req.params, schema, {
abortEarly: false,
convert: true
});
if (result.error) {
res.json({
error: result.error.message,
code: 'InputValidationError'
}); });
return next();
}
dkimHandler.set(result.value, (err, response) => { const result = Joi.validate(req.params, schema, {
if (err) { abortEarly: false,
convert: true
});
if (result.error) {
res.json({
error: result.error.message,
code: 'InputValidationError'
});
return next();
}
// permissions check
req.validate(roles.can(req.role).createAny('dkim'));
let response;
try {
response = await setDkim(result.value);
} catch (err) {
res.json({ res.json({
error: err.message, error: err.message,
code: err.code code: err.code
}); });
return next(); return next();
} }
if (response) { if (response) {
response.success = true; response.success = true;
} }
res.json(response); res.json(response);
return next(); return next();
}); })
}); );
/** /**
* @api {get} /dkim/:dkim Request DKIM information * @api {get} /dkim/:dkim Request DKIM information
@ -357,52 +480,61 @@ module.exports = (db, server) => {
* "error": "This Alias does not exist" * "error": "This Alias does not exist"
* } * }
*/ */
server.get('/dkim/:dkim', (req, res, next) => { server.get(
res.charSet('utf-8'); '/dkim/:dkim',
tools.asyncifyJson(async (req, res, next) => {
res.charSet('utf-8');
const schema = Joi.object().keys({ const schema = Joi.object().keys({
dkim: Joi.string() dkim: Joi.string()
.hex() .hex()
.lowercase() .lowercase()
.length(24) .length(24)
.required(), .required(),
sess: Joi.string().max(255), sess: Joi.string().max(255),
ip: Joi.string().ip({ ip: Joi.string().ip({
version: ['ipv4', 'ipv6'], version: ['ipv4', 'ipv6'],
cidr: 'forbidden' cidr: 'forbidden'
}) })
});
const result = Joi.validate(req.params, schema, {
abortEarly: false,
convert: true
});
if (result.error) {
res.json({
error: result.error.message,
code: 'InputValidationError'
}); });
return next();
}
let dkim = new ObjectID(result.value.dkim); const result = Joi.validate(req.params, schema, {
abortEarly: false,
convert: true
});
dkimHandler.get({ _id: dkim }, false, (err, response) => { if (result.error) {
if (err) { res.json({
error: result.error.message,
code: 'InputValidationError'
});
return next();
}
// permissions check
req.validate(roles.can(req.role).readAny('dkim'));
let dkim = new ObjectID(result.value.dkim);
let response;
try {
response = await getDkim({ _id: dkim }, false);
} catch (err) {
res.json({ res.json({
error: err.message, error: err.message,
code: err.code code: err.code
}); });
return next(); return next();
} }
if (response) { if (response) {
response.success = true; response.success = true;
} }
res.json(response); res.json(response);
return next(); return next();
}); })
}); );
/** /**
* @api {delete} /dkim/:dkim Delete a DKIM key * @api {delete} /dkim/:dkim Delete a DKIM key
@ -435,49 +567,59 @@ module.exports = (db, server) => {
* "error": "Database error" * "error": "Database error"
* } * }
*/ */
server.del('/dkim/:dkim', (req, res, next) => { server.del(
res.charSet('utf-8'); '/dkim/:dkim',
tools.asyncifyJson(async (req, res, next) => {
res.charSet('utf-8');
const schema = Joi.object().keys({ const schema = Joi.object().keys({
dkim: Joi.string() dkim: Joi.string()
.hex() .hex()
.lowercase() .lowercase()
.length(24) .length(24)
.required(), .required(),
sess: Joi.string().max(255), sess: Joi.string().max(255),
ip: Joi.string().ip({ ip: Joi.string().ip({
version: ['ipv4', 'ipv6'], version: ['ipv4', 'ipv6'],
cidr: 'forbidden' cidr: 'forbidden'
}) })
});
const result = Joi.validate(req.params, schema, {
abortEarly: false,
convert: true
});
if (result.error) {
res.json({
error: result.error.message,
code: 'InputValidationError'
}); });
return next();
}
let dkim = new ObjectID(result.value.dkim); const result = Joi.validate(req.params, schema, {
abortEarly: false,
convert: true
});
dkimHandler.del({ _id: dkim }, (err, response) => { if (result.error) {
if (err) { res.json({
error: result.error.message,
code: 'InputValidationError'
});
return next();
}
// permissions check
req.validate(roles.can(req.role).deleteAny('dkim'));
let dkim = new ObjectID(result.value.dkim);
let response;
try {
response = await delDkim({ _id: dkim });
} catch (err) {
res.json({ res.json({
error: err.message, error: err.message,
code: err.code code: err.code
}); });
return next(); return next();
} }
res.json({ res.json({
success: response success: response
}); });
return next(); return next();
}); })
}); );
}; };

View file

@ -329,7 +329,7 @@ module.exports = (db, server) => {
* @apiParam {String} alias Alias domain * @apiParam {String} alias Alias domain
* *
* @apiSuccess {Boolean} success Indicates successful response * @apiSuccess {Boolean} success Indicates successful response
* @apiSuccess {String} id Users unique ID (24 byte hex) * @apiSuccess {String} id Alias unique ID (24 byte hex)
* *
* @apiError error Description of the error * @apiError error Description of the error
* *
@ -377,7 +377,7 @@ module.exports = (db, server) => {
return next(); return next();
} }
let alias = tools.normalizeDomain(req.params.alias); let alias = tools.normalizeDomain(result.value.alias);
db.users.collection('domainaliases').findOne( db.users.collection('domainaliases').findOne(
{ {

View file

@ -200,7 +200,7 @@ class DkimHandler {
query._id = new ObjectID(options._id); query._id = new ObjectID(options._id);
} else { } else {
let err = new Error('Invalid or unknown DKIM key'); let err = new Error('Invalid or unknown DKIM key');
err.code = 'KeyNotFound'; err.code = 'DkimNotFound';
return setImmediate(() => callback(err)); return setImmediate(() => callback(err));
} }
@ -211,7 +211,7 @@ class DkimHandler {
} }
if (!dkimData) { if (!dkimData) {
let err = new Error('Invalid or unknown DKIM key'); let err = new Error('Invalid or unknown DKIM key');
err.code = 'KeyNotFound'; err.code = 'DkimNotFound';
return callback(err); return callback(err);
} }
@ -263,7 +263,7 @@ class DkimHandler {
query._id = new ObjectID(options._id); query._id = new ObjectID(options._id);
} else { } else {
let err = new Error('Invalid or unknown DKIM key'); let err = new Error('Invalid or unknown DKIM key');
err.code = 'KeyNotFound'; err.code = 'DkimNotFound';
return setImmediate(() => callback(err)); return setImmediate(() => callback(err));
} }
@ -276,7 +276,7 @@ class DkimHandler {
if (!r.deletedCount) { if (!r.deletedCount) {
let err = new Error('Invalid or unknown DKIM key'); let err = new Error('Invalid or unknown DKIM key');
err.code = 'KeyNotFound'; err.code = 'DkimNotFound';
return callback(err); return callback(err);
} }