2017-07-26 16:52:55 +08:00
|
|
|
'use strict';
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
const Joi = require('../joi');
|
2017-11-27 05:54:53 +08:00
|
|
|
const MongoPaging = require('mongo-cursor-pagination-node6');
|
2017-07-26 16:52:55 +08:00
|
|
|
const ObjectID = require('mongodb').ObjectID;
|
|
|
|
const tools = require('../tools');
|
|
|
|
|
|
|
|
module.exports = (db, server) => {
|
2017-11-28 19:32:59 +08:00
|
|
|
/**
|
|
|
|
* @api {get} /addresses List registered Addresses
|
|
|
|
* @apiName GetAddresses
|
|
|
|
* @apiGroup Addresses
|
|
|
|
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
|
|
|
|
* @apiHeaderExample {json} Header-Example:
|
|
|
|
* {
|
|
|
|
* "X-Access-Token": "59fc66a03e54454869460e45"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiParam {String} [query] Partial match of an address
|
|
|
|
* @apiParam {Number} [limit=20] How many records to return
|
|
|
|
* @apiParam {Number} [page=1] Current page number. Informational only, page numbers start from 1
|
|
|
|
* @apiParam {Number} [next] Cursor value for next page, retrieved from <code>nextCursor</code> response value
|
|
|
|
* @apiParam {Number} [previous] Cursor value for previous page, retrieved from <code>previousCursor</code> response value
|
|
|
|
*
|
|
|
|
* @apiSuccess {Boolean} success Indicates successful response
|
|
|
|
* @apiSuccess {Number} total How many results were found
|
|
|
|
* @apiSuccess {Number} page Current page number. Derived from <code>page</code> query argument
|
|
|
|
* @apiSuccess {String} previousCursor Either a cursor string or false if there are not any previous results
|
|
|
|
* @apiSuccess {String} nextCursor Either a cursor string or false if there are not any next results
|
|
|
|
* @apiSuccess {Object[]} results Address listing
|
|
|
|
* @apiSuccess {String} results.id ID of the Address
|
|
|
|
* @apiSuccess {String} results.address E-mail address string
|
|
|
|
* @apiSuccess {String} results.user User ID this address belongs to
|
|
|
|
*
|
|
|
|
* @apiError error Description of the error
|
|
|
|
*
|
|
|
|
* @apiExample {curl} Example usage:
|
|
|
|
* curl -i http://localhost:8080/addresses
|
|
|
|
*
|
|
|
|
* @apiSuccessExample {json} Success-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "success": true,
|
|
|
|
* "total": 1,
|
|
|
|
* "page": 1,
|
|
|
|
* "previousCursor": false,
|
|
|
|
* "nextCursor": false,
|
|
|
|
* "results": [
|
|
|
|
* {
|
|
|
|
* "id": "59ef21aef255ed1d9d790e81",
|
|
|
|
* "address": "user@example.com",
|
|
|
|
* "user": "59ef21aef255ed1d9d790e7a"
|
|
|
|
* }
|
|
|
|
* ]
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiErrorExample {json} Error-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "error": "Database error"
|
|
|
|
* }
|
|
|
|
*/
|
2017-07-26 16:52:55 +08:00
|
|
|
server.get({ name: 'addresses', path: '/addresses' }, (req, res, next) => {
|
|
|
|
res.charSet('utf-8');
|
|
|
|
|
|
|
|
const schema = Joi.object().keys({
|
2017-09-01 19:50:53 +08:00
|
|
|
query: Joi.string()
|
|
|
|
.trim()
|
|
|
|
.empty('')
|
|
|
|
.max(255),
|
|
|
|
limit: Joi.number()
|
|
|
|
.default(20)
|
|
|
|
.min(1)
|
|
|
|
.max(250),
|
|
|
|
next: Joi.string()
|
2017-09-04 19:49:04 +08:00
|
|
|
.empty('')
|
2017-11-27 20:20:57 +08:00
|
|
|
.mongoCursor()
|
2017-09-04 19:49:04 +08:00
|
|
|
.max(1024),
|
2017-09-04 20:04:43 +08:00
|
|
|
previous: Joi.string()
|
2017-09-04 19:49:04 +08:00
|
|
|
.empty('')
|
2017-11-27 20:20:57 +08:00
|
|
|
.mongoCursor()
|
2017-09-04 19:49:04 +08:00
|
|
|
.max(1024),
|
2017-07-26 16:52:55 +08:00
|
|
|
page: Joi.number().default(1)
|
|
|
|
});
|
|
|
|
|
|
|
|
const result = Joi.validate(req.query, schema, {
|
|
|
|
abortEarly: false,
|
|
|
|
convert: true,
|
|
|
|
allowUnknown: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (result.error) {
|
|
|
|
res.json({
|
|
|
|
error: result.error.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let query = result.value.query;
|
|
|
|
let limit = result.value.limit;
|
|
|
|
let page = result.value.page;
|
|
|
|
let pageNext = result.value.next;
|
2017-09-04 20:04:43 +08:00
|
|
|
let pagePrevious = result.value.previous;
|
2017-07-26 16:52:55 +08:00
|
|
|
|
|
|
|
let filter = query
|
|
|
|
? {
|
|
|
|
address: {
|
2017-07-31 15:59:18 +08:00
|
|
|
// cannot use dotless version as this would break domain search
|
2017-07-26 16:52:55 +08:00
|
|
|
$regex: query.replace(/[-/\\^$*+?.()|[\]{}]/g, '\\$&'),
|
|
|
|
$options: ''
|
|
|
|
}
|
|
|
|
}
|
|
|
|
: {};
|
|
|
|
|
|
|
|
db.users.collection('addresses').count(filter, (err, total) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let opts = {
|
|
|
|
limit,
|
|
|
|
query: filter,
|
|
|
|
fields: {
|
|
|
|
_id: true,
|
|
|
|
address: true,
|
|
|
|
user: true
|
|
|
|
},
|
2017-09-04 14:49:02 +08:00
|
|
|
paginatedField: 'addrview',
|
2017-07-26 16:52:55 +08:00
|
|
|
sortAscending: true
|
|
|
|
};
|
|
|
|
|
|
|
|
if (pageNext) {
|
|
|
|
opts.next = pageNext;
|
2017-09-04 20:04:43 +08:00
|
|
|
} else if (pagePrevious) {
|
|
|
|
opts.previous = pagePrevious;
|
2017-07-26 16:52:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
MongoPaging.find(db.users.collection('addresses'), opts, (err, result) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!result.hasPrevious) {
|
|
|
|
page = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
let response = {
|
|
|
|
success: true,
|
|
|
|
query,
|
|
|
|
total,
|
|
|
|
page,
|
2017-09-01 19:50:53 +08:00
|
|
|
previousCursor: result.hasPrevious ? result.previous : false,
|
|
|
|
nextCursor: result.hasNext ? result.next : false,
|
2017-07-26 16:52:55 +08:00
|
|
|
results: (result.results || []).map(addressData => ({
|
|
|
|
id: addressData._id.toString(),
|
|
|
|
address: addressData.address,
|
|
|
|
user: addressData.user.toString()
|
|
|
|
}))
|
|
|
|
};
|
|
|
|
|
|
|
|
res.json(response);
|
|
|
|
return next();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2017-11-28 19:32:59 +08:00
|
|
|
/**
|
|
|
|
* @api {post} /users/:user/addresses Create new Address
|
|
|
|
* @apiName PostUserAddress
|
|
|
|
* @apiGroup Addresses
|
|
|
|
* @apiDescription Add a new email address for an User. Addresses can contain unicode characters.
|
|
|
|
* Dots in usernames are normalized so no need to create both "firstlast@example.com" and "first.last@example.com"
|
|
|
|
*
|
2017-12-01 21:04:32 +08:00
|
|
|
* Special addresses <code>*@example.com</code> and <code>username@*</code> catches all emails to these domains or users without a registered destination (requires <code>allowWildcard</code> argument)
|
2017-11-28 19:32:59 +08:00
|
|
|
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
|
|
|
|
* @apiHeaderExample {json} Header-Example:
|
|
|
|
* {
|
|
|
|
* "X-Access-Token": "59fc66a03e54454869460e45"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiParam {String} user ID of the User
|
|
|
|
* @apiParam {String} address E-mail Address
|
|
|
|
* @apiParam {Boolean} [main=false] Indicates if this is the default address for the User
|
|
|
|
* @apiParam {Boolean} [allowWildcard=false] If <code>true</code> then address value can be in the form of <code>*@example.com</code>, otherwise using * is not allowed
|
|
|
|
*
|
|
|
|
* @apiSuccess {Boolean} success Indicates successful response
|
|
|
|
* @apiSuccess {String} id ID of the Address
|
|
|
|
*
|
|
|
|
* @apiError error Description of the error
|
|
|
|
*
|
|
|
|
* @apiExample {curl} Example usage:
|
2017-12-01 21:04:32 +08:00
|
|
|
* curl -i -XPOST http://localhost:8080/users/59fc66a03e54454869460e45/addresses \
|
2017-11-28 19:32:59 +08:00
|
|
|
* -H 'Content-type: application/json' \
|
|
|
|
* -d '{
|
|
|
|
* "address": "my.new.address@example.com"
|
|
|
|
* }'
|
|
|
|
*
|
|
|
|
* @apiSuccessExample {json} Success-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
2017-12-01 21:04:32 +08:00
|
|
|
* "success": true,
|
|
|
|
* "id": "59ef21aef255ed1d9d790e81"
|
2017-11-28 19:32:59 +08:00
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiErrorExample {json} Error-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "error": "This user does not exist"
|
|
|
|
* }
|
|
|
|
*/
|
2017-07-26 16:52:55 +08:00
|
|
|
server.post('/users/:user/addresses', (req, res, next) => {
|
|
|
|
res.charSet('utf-8');
|
|
|
|
|
|
|
|
const schema = Joi.object().keys({
|
2017-09-01 19:50:53 +08:00
|
|
|
user: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required(),
|
2017-12-01 21:21:44 +08:00
|
|
|
address: [
|
|
|
|
Joi.string()
|
|
|
|
.email()
|
|
|
|
.required(),
|
|
|
|
Joi.string().regex(/^\w+@\*$/, 'special address')
|
|
|
|
],
|
2017-11-28 19:32:59 +08:00
|
|
|
main: Joi.boolean().truthy(['Y', 'true', 'yes', 1]),
|
|
|
|
allowWildcard: Joi.boolean().truthy(['Y', 'true', 'yes', 1])
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
let address = tools.normalizeAddress(req.params.address);
|
|
|
|
|
|
|
|
const result = Joi.validate(req.params, schema, {
|
|
|
|
abortEarly: false,
|
|
|
|
convert: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (result.error) {
|
|
|
|
res.json({
|
|
|
|
error: result.error.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let user = new ObjectID(result.value.user);
|
|
|
|
let main = result.value.main;
|
|
|
|
|
|
|
|
if (address.indexOf('+') >= 0) {
|
|
|
|
res.json({
|
|
|
|
error: 'Address can not contain +'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-12-01 21:04:32 +08:00
|
|
|
let wcpos = req.params.address.indexOf('*');
|
2017-11-28 19:32:59 +08:00
|
|
|
|
2017-12-01 21:04:32 +08:00
|
|
|
if (wcpos >= 0) {
|
|
|
|
if (!result.value.allowWildcard) {
|
|
|
|
res.json({
|
|
|
|
error: 'Address can not contain *'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (/[^@]\*|\*[^@]/.test(result.value) || wcpos !== req.params.address.lastIndexOf('*')) {
|
|
|
|
res.json({
|
|
|
|
error: 'Invalid wildcard address, use "*@domain" or "user@*"'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (main) {
|
|
|
|
res.json({
|
|
|
|
error: 'Main address can not contain *'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-11-28 19:57:38 +08:00
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('users').findOne(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: {
|
|
|
|
address: true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(err, userData) => {
|
2017-07-26 16:52:55 +08:00
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
if (!userData) {
|
2017-07-26 16:52:55 +08:00
|
|
|
res.json({
|
2017-11-27 20:20:57 +08:00
|
|
|
error: 'This user does not exist'
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('addresses').findOne(
|
|
|
|
{
|
|
|
|
addrview: address.substr(0, address.indexOf('@')).replace(/\./g, '') + address.substr(address.indexOf('@'))
|
|
|
|
},
|
|
|
|
(err, addressData) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
if (addressData) {
|
|
|
|
res.json({
|
|
|
|
error: 'This email address already exists'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
// insert alias address to email address registry
|
|
|
|
db.users.collection('addresses').insertOne(
|
2017-07-26 16:52:55 +08:00
|
|
|
{
|
2017-11-27 20:20:57 +08:00
|
|
|
user,
|
|
|
|
address,
|
|
|
|
addrview: address.substr(0, address.indexOf('@')).replace(/\./g, '') + address.substr(address.indexOf('@')),
|
|
|
|
created: new Date()
|
2017-07-26 16:52:55 +08:00
|
|
|
},
|
2017-11-27 20:20:57 +08:00
|
|
|
(err, r) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
2017-07-26 16:52:55 +08:00
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
|
|
|
|
let insertId = r.insertedId;
|
|
|
|
|
|
|
|
let done = () => {
|
|
|
|
// ignore potential user update error
|
|
|
|
res.json({
|
|
|
|
success: !!insertId,
|
|
|
|
id: insertId
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!userData.address || main) {
|
|
|
|
// register this address as the default address for that user
|
|
|
|
return db.users.collection('users').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set: {
|
|
|
|
address
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{},
|
|
|
|
done
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
done();
|
|
|
|
}
|
2017-07-26 16:52:55 +08:00
|
|
|
);
|
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
2017-11-28 19:32:59 +08:00
|
|
|
/**
|
|
|
|
* @api {get} /users/:user/addresses List registered Addresses for an User
|
|
|
|
* @apiName GetUserAddresses
|
|
|
|
* @apiGroup Addresses
|
|
|
|
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
|
|
|
|
* @apiHeaderExample {json} Header-Example:
|
|
|
|
* {
|
|
|
|
* "X-Access-Token": "59fc66a03e54454869460e45"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiParam {String} user ID of the User
|
|
|
|
*
|
|
|
|
* @apiSuccess {Boolean} success Indicates successful response
|
|
|
|
* @apiSuccess {Object[]} results Address listing
|
|
|
|
* @apiSuccess {String} results.id ID of the Address
|
|
|
|
* @apiSuccess {String} results.address E-mail address string
|
|
|
|
* @apiSuccess {Boolean} results.main Indicates if this is the default address for the User
|
|
|
|
* @apiSuccess {String} results.created Datestring of the time the address was created
|
|
|
|
*
|
|
|
|
* @apiError error Description of the error
|
|
|
|
*
|
|
|
|
* @apiExample {curl} Example usage:
|
|
|
|
* curl -i http://localhost:8080/users/59ef21aef255ed1d9d790e7a/addresses
|
|
|
|
*
|
|
|
|
* @apiSuccessExample {json} Success-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "success": true,
|
|
|
|
* "total": 1,
|
|
|
|
* "page": 1,
|
|
|
|
* "previousCursor": false,
|
|
|
|
* "nextCursor": false,
|
|
|
|
* "results": [
|
|
|
|
* {
|
|
|
|
* "id": "59ef21aef255ed1d9d790e81",
|
|
|
|
* "address": "user@example.com",
|
|
|
|
* "main": true,
|
|
|
|
* "created": "2017-10-24T11:19:10.911Z"
|
|
|
|
* }
|
|
|
|
* ]
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiErrorExample {json} Error-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "error": "This user does not exist"
|
|
|
|
* }
|
|
|
|
*/
|
2017-07-26 16:52:55 +08:00
|
|
|
server.get('/users/:user/addresses', (req, res, next) => {
|
|
|
|
res.charSet('utf-8');
|
|
|
|
|
|
|
|
const schema = Joi.object().keys({
|
2017-09-01 19:50:53 +08:00
|
|
|
user: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required()
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const result = Joi.validate(req.params, schema, {
|
|
|
|
abortEarly: false,
|
|
|
|
convert: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (result.error) {
|
|
|
|
res.json({
|
|
|
|
error: result.error.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let user = new ObjectID(result.value.user);
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('users').findOne(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: {
|
|
|
|
address: true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(err, userData) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
if (!userData) {
|
|
|
|
res.json({
|
|
|
|
error: 'This user does not exist'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
db.users
|
|
|
|
.collection('addresses')
|
|
|
|
.find({
|
|
|
|
user
|
|
|
|
})
|
|
|
|
.sort({
|
|
|
|
addrview: 1
|
|
|
|
})
|
|
|
|
.toArray((err, addresses) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!addresses) {
|
|
|
|
addresses = [];
|
|
|
|
}
|
2017-07-26 16:52:55 +08:00
|
|
|
|
|
|
|
res.json({
|
2017-11-27 20:20:57 +08:00
|
|
|
success: true,
|
|
|
|
|
|
|
|
results: addresses.map(address => ({
|
|
|
|
id: address._id,
|
|
|
|
address: address.address,
|
|
|
|
main: address.address === userData.address,
|
|
|
|
created: address.created
|
|
|
|
}))
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
return next();
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
2017-11-27 20:20:57 +08:00
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
2017-11-28 19:32:59 +08:00
|
|
|
/**
|
|
|
|
* @api {get} /users/:user/addresses/:address Request Addresses information
|
|
|
|
* @apiName GetUserAddress
|
|
|
|
* @apiGroup Addresses
|
|
|
|
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
|
|
|
|
* @apiHeaderExample {json} Header-Example:
|
|
|
|
* {
|
|
|
|
* "X-Access-Token": "59fc66a03e54454869460e45"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiParam {String} user ID of the User
|
|
|
|
* @apiParam {String} address ID of the Address
|
|
|
|
*
|
|
|
|
* @apiSuccess {Boolean} success Indicates successful response
|
|
|
|
* @apiSuccess {String} id ID of the Address
|
|
|
|
* @apiSuccess {String} address E-mail address string
|
|
|
|
* @apiSuccess {Boolean} main Indicates if this is the default address for the User
|
|
|
|
* @apiSuccess {String} created Datestring of the time the address was created
|
|
|
|
*
|
|
|
|
* @apiError error Description of the error
|
|
|
|
*
|
|
|
|
* @apiExample {curl} Example usage:
|
|
|
|
* curl -i http://localhost:8080/users/59ef21aef255ed1d9d790e7a/addresses/59ef21aef255ed1d9d790e81
|
|
|
|
*
|
|
|
|
* @apiSuccessExample {json} Success-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "success": true,
|
|
|
|
* "id": "59ef21aef255ed1d9d790e81",
|
|
|
|
* "address": "user@example.com",
|
|
|
|
* "main": true,
|
|
|
|
* "created": "2017-10-24T11:19:10.911Z"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiErrorExample {json} Error-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "error": "This user does not exist"
|
|
|
|
* }
|
|
|
|
*/
|
2017-07-26 16:52:55 +08:00
|
|
|
server.get('/users/:user/addresses/:address', (req, res, next) => {
|
|
|
|
res.charSet('utf-8');
|
|
|
|
|
|
|
|
const schema = Joi.object().keys({
|
2017-09-01 19:50:53 +08:00
|
|
|
user: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required(),
|
|
|
|
address: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required()
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const result = Joi.validate(req.params, schema, {
|
|
|
|
abortEarly: false,
|
|
|
|
convert: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (result.error) {
|
|
|
|
res.json({
|
|
|
|
error: result.error.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let user = new ObjectID(result.value.user);
|
|
|
|
let address = new ObjectID(result.value.address);
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('users').findOne(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: {
|
|
|
|
address: true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(err, userData) => {
|
2017-07-26 16:52:55 +08:00
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
if (!userData) {
|
2017-07-26 16:52:55 +08:00
|
|
|
res.json({
|
2017-11-27 20:20:57 +08:00
|
|
|
error: 'This user does not exist'
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('addresses').findOne(
|
|
|
|
{
|
|
|
|
_id: address,
|
|
|
|
user
|
|
|
|
},
|
|
|
|
(err, addressData) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
if (!addressData) {
|
|
|
|
res.status(404);
|
|
|
|
res.json({
|
|
|
|
error: 'Invalid or unknown address'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-07-26 16:52:55 +08:00
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
res.json({
|
|
|
|
success: true,
|
|
|
|
id: addressData._id,
|
|
|
|
address: addressData.address,
|
|
|
|
main: addressData.address === userData.address,
|
|
|
|
created: addressData.created
|
|
|
|
});
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
2017-11-28 19:32:59 +08:00
|
|
|
/**
|
|
|
|
* @api {put} /users/:user/addresses/:address Update Address information
|
|
|
|
* @apiName PutUserAddress
|
|
|
|
* @apiGroup Addresses
|
|
|
|
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
|
|
|
|
* @apiHeaderExample {json} Header-Example:
|
|
|
|
* {
|
|
|
|
* "X-Access-Token": "59fc66a03e54454869460e45"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiParam {String} user ID of the User
|
|
|
|
* @apiParam {String} user ID of the Address
|
|
|
|
* @apiParam {Boolean} main Indicates if this is the default address for the User
|
|
|
|
*
|
|
|
|
* @apiSuccess {Boolean} success Indicates successful response
|
|
|
|
*
|
|
|
|
* @apiError error Description of the error
|
|
|
|
*
|
|
|
|
* @apiExample {curl} Example usage:
|
|
|
|
* curl -i -XPUT http://localhost:8080/users/59fc66a03e54454869460e45/addresses/5a1d4541153888cdcd62a71b \
|
|
|
|
* -H 'Content-type: application/json' \
|
|
|
|
* -d '{
|
|
|
|
* "main": true
|
|
|
|
* }'
|
|
|
|
*
|
|
|
|
* @apiSuccessExample {json} Success-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "success": true
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiErrorExample {json} Error-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "error": "This user does not exist"
|
|
|
|
* }
|
|
|
|
*/
|
2017-07-26 16:52:55 +08:00
|
|
|
server.put('/users/:user/addresses/:address', (req, res, next) => {
|
|
|
|
res.charSet('utf-8');
|
|
|
|
|
|
|
|
const schema = Joi.object().keys({
|
2017-09-01 19:50:53 +08:00
|
|
|
user: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required(),
|
|
|
|
address: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required(),
|
|
|
|
main: Joi.boolean()
|
|
|
|
.truthy(['Y', 'true', 'yes', 1])
|
|
|
|
.required()
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const result = Joi.validate(req.params, schema, {
|
|
|
|
abortEarly: false,
|
|
|
|
convert: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (result.error) {
|
|
|
|
res.json({
|
|
|
|
error: result.error.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let user = new ObjectID(result.value.user);
|
|
|
|
let address = new ObjectID(result.value.address);
|
|
|
|
let main = result.value.main;
|
|
|
|
|
|
|
|
if (!main) {
|
|
|
|
res.json({
|
|
|
|
error: 'Cannot unset main status'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('users').findOne(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: {
|
|
|
|
address: true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(err, userData) => {
|
2017-07-26 16:52:55 +08:00
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
if (!userData) {
|
2017-07-26 16:52:55 +08:00
|
|
|
res.json({
|
2017-11-27 20:20:57 +08:00
|
|
|
error: 'This user does not exist'
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('addresses').findOne(
|
|
|
|
{
|
|
|
|
_id: address
|
|
|
|
},
|
|
|
|
(err, addressData) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!addressData || addressData.user.toString() !== user.toString()) {
|
|
|
|
res.status(404);
|
|
|
|
res.json({
|
|
|
|
error: 'Invalid or unknown email address identifier'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (addressData.address === userData.address) {
|
|
|
|
res.json({
|
|
|
|
error: 'Selected address is already the main email address for the user'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-28 19:57:38 +08:00
|
|
|
if (addressData.address.indexOf('*') >= 0 && main) {
|
|
|
|
res.json({
|
|
|
|
error: 'Can not set wildcard address as default'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
// insert alias address to email address registry
|
|
|
|
db.users.collection('users').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set: {
|
|
|
|
address: addressData.address
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
returnOriginal: false
|
|
|
|
},
|
|
|
|
(err, r) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-07-26 16:52:55 +08:00
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
res.json({
|
|
|
|
success: !!r.value
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
2017-11-28 19:32:59 +08:00
|
|
|
/**
|
|
|
|
* @api {delete} /users/:user/addresses/:address Delete an Address
|
|
|
|
* @apiName DeleteUserAddress
|
|
|
|
* @apiGroup Addresses
|
|
|
|
* @apiHeader {String} X-Access-Token Optional access token if authentication is enabled
|
|
|
|
* @apiHeaderExample {json} Header-Example:
|
|
|
|
* {
|
|
|
|
* "X-Access-Token": "59fc66a03e54454869460e45"
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiParam {String} user ID of the User
|
|
|
|
* @apiParam {String} address ID of the Address
|
|
|
|
*
|
|
|
|
* @apiSuccess {Boolean} success Indicates successful response
|
|
|
|
*
|
|
|
|
* @apiError error Description of the error
|
|
|
|
*
|
|
|
|
* @apiExample {curl} Example usage:
|
|
|
|
* curl -i -XDELETE http://localhost:8080/users/59ef21aef255ed1d9d790e7a/addresses/59ef21aef255ed1d9d790e81
|
|
|
|
*
|
|
|
|
* @apiSuccessExample {json} Success-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "success": true
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @apiErrorExample {json} Error-Response:
|
|
|
|
* HTTP/1.1 200 OK
|
|
|
|
* {
|
|
|
|
* "error": "Trying to delete main address. Set a new main address first"
|
|
|
|
* }
|
|
|
|
*/
|
2017-07-26 16:52:55 +08:00
|
|
|
server.del('/users/:user/addresses/:address', (req, res, next) => {
|
|
|
|
res.charSet('utf-8');
|
|
|
|
|
|
|
|
const schema = Joi.object().keys({
|
2017-09-01 19:50:53 +08:00
|
|
|
user: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required(),
|
|
|
|
address: Joi.string()
|
|
|
|
.hex()
|
|
|
|
.lowercase()
|
|
|
|
.length(24)
|
|
|
|
.required()
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const result = Joi.validate(req.params, schema, {
|
|
|
|
abortEarly: false,
|
|
|
|
convert: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (result.error) {
|
|
|
|
res.json({
|
|
|
|
error: result.error.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let user = new ObjectID(result.value.user);
|
|
|
|
let address = new ObjectID(result.value.address);
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('users').findOne(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: {
|
|
|
|
address: true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(err, userData) => {
|
2017-07-26 16:52:55 +08:00
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
if (!userData) {
|
2017-07-26 16:52:55 +08:00
|
|
|
res.json({
|
2017-11-27 20:20:57 +08:00
|
|
|
error: 'This user does not exist'
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
db.users.collection('addresses').findOne(
|
|
|
|
{
|
|
|
|
_id: address
|
|
|
|
},
|
|
|
|
(err, addressData) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!addressData || addressData.user.toString() !== user.toString()) {
|
|
|
|
res.status(404);
|
|
|
|
res.json({
|
|
|
|
error: 'Invalid or unknown email address identifier'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (addressData.address === userData.address) {
|
|
|
|
res.json({
|
|
|
|
error: 'Trying to delete main address. Set a new main address first'
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
// delete address from email address registry
|
|
|
|
db.users.collection('addresses').deleteOne(
|
|
|
|
{
|
|
|
|
_id: address
|
|
|
|
},
|
|
|
|
(err, r) => {
|
|
|
|
if (err) {
|
|
|
|
res.json({
|
|
|
|
error: 'MongoDB Error: ' + err.message
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
2017-07-26 16:52:55 +08:00
|
|
|
|
2017-11-27 20:20:57 +08:00
|
|
|
res.json({
|
|
|
|
success: !!r.deletedCount
|
|
|
|
});
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
}
|
2017-11-27 20:20:57 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-26 16:52:55 +08:00
|
|
|
});
|
|
|
|
};
|