npm WARN deprecated ejs-cli@1.2.0: This has breaking change. (in ejs package) use <= 2.0.0. npm WARN deprecated node-uuid@1.4.8: Use uuid module instead npm WARN deprecated minimatch@0.3.0: Please update to minimatch 3.0.2 or higher to avoid a RegExp DoS issue npm WARN deprecated minimatch@2.0.10: Please update to minimatch 3.0.2 or higher to avoid a RegExp DoS issue npm WARN deprecated minimatch@0.2.14: Please update to minimatch 3.0.2 or higher to avoid a RegExp DoS issue npm WARN deprecated graceful-fs@1.2.3: graceful-fs v3.0.0 and before will fail on node releases >= v7.0. Please update to graceful-fs@^4.0.0 as soon as possible. Use 'npm ls graceful-fs' to find it in the tree.
242 lines
8.4 KiB
JavaScript
242 lines
8.4 KiB
JavaScript
'use strict';
|
|
|
|
exports = module.exports = {
|
|
GroupError: GroupError,
|
|
|
|
create: create,
|
|
remove: remove,
|
|
get: get,
|
|
getWithMembers: getWithMembers,
|
|
getAll: getAll,
|
|
getAllWithMembers: getAllWithMembers,
|
|
|
|
getMembers: getMembers,
|
|
addMember: addMember,
|
|
setMembers: setMembers,
|
|
removeMember: removeMember,
|
|
isMember: isMember,
|
|
|
|
getGroups: getGroups,
|
|
setGroups: setGroups
|
|
};
|
|
|
|
var assert = require('assert'),
|
|
constants = require('./constants.js'),
|
|
DatabaseError = require('./databaseerror.js'),
|
|
groupdb = require('./groupdb.js'),
|
|
util = require('util'),
|
|
uuid = require('uuid');
|
|
|
|
// http://dustinsenos.com/articles/customErrorsInNode
|
|
// http://code.google.com/p/v8/wiki/JavaScriptStackTraceApi
|
|
function GroupError(reason, errorOrMessage) {
|
|
assert.strictEqual(typeof reason, 'string');
|
|
assert(errorOrMessage instanceof Error || typeof errorOrMessage === 'string' || typeof errorOrMessage === 'undefined');
|
|
|
|
Error.call(this);
|
|
Error.captureStackTrace(this, this.constructor);
|
|
|
|
this.name = this.constructor.name;
|
|
this.reason = reason;
|
|
if (typeof errorOrMessage === 'undefined') {
|
|
this.message = reason;
|
|
} else if (typeof errorOrMessage === 'string') {
|
|
this.message = errorOrMessage;
|
|
} else {
|
|
this.message = 'Internal error';
|
|
this.nestedError = errorOrMessage;
|
|
}
|
|
}
|
|
util.inherits(GroupError, Error);
|
|
GroupError.INTERNAL_ERROR = 'Internal Error';
|
|
GroupError.ALREADY_EXISTS = 'Already Exists';
|
|
GroupError.NOT_FOUND = 'Not Found';
|
|
GroupError.BAD_FIELD = 'Field error';
|
|
GroupError.NOT_EMPTY = 'Not Empty';
|
|
GroupError.NOT_ALLOWED = 'Not Allowed';
|
|
|
|
// keep this in sync with validateUsername
|
|
function validateGroupname(name) {
|
|
assert.strictEqual(typeof name, 'string');
|
|
|
|
if (name.length < 1) return new GroupError(GroupError.BAD_FIELD, 'name must be atleast 1 char');
|
|
if (name.length >= 200) return new GroupError(GroupError.BAD_FIELD, 'name too long');
|
|
|
|
if (constants.RESERVED_NAMES.indexOf(name) !== -1) return new GroupError(GroupError.BAD_FIELD, 'name is reserved');
|
|
|
|
// +/- can be tricky in emails. also need to consider valid LDAP characters here (e.g '+' is reserved)
|
|
if (/[^a-zA-Z0-9.]/.test(name)) return new GroupError(GroupError.BAD_FIELD, 'name can only contain alphanumerals and dot');
|
|
|
|
// app emails are sent using the .app suffix
|
|
if (name.indexOf('.app') !== -1) return new GroupError(GroupError.BAD_FIELD, 'name pattern is reserved for apps');
|
|
|
|
return null;
|
|
}
|
|
|
|
function create(name, callback) {
|
|
assert.strictEqual(typeof name, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
// we store names in lowercase
|
|
name = name.toLowerCase();
|
|
|
|
var error = validateGroupname(name);
|
|
if (error) return callback(error);
|
|
|
|
var id = 'gid-' + uuid.v4();
|
|
groupdb.add(id, name, function (error) {
|
|
if (error && error.reason === DatabaseError.ALREADY_EXISTS) return callback(new GroupError(GroupError.ALREADY_EXISTS));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
callback(null, { id: id, name: name });
|
|
});
|
|
}
|
|
|
|
function remove(id, callback) {
|
|
assert.strictEqual(typeof id, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
// never allow admin group to be deleted
|
|
if (id === constants.ADMIN_GROUP_ID) return callback(new GroupError(GroupError.NOT_ALLOWED));
|
|
|
|
groupdb.del(id, function (error) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
callback(null);
|
|
});
|
|
}
|
|
|
|
function get(id, callback) {
|
|
assert.strictEqual(typeof id, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.get(id, function (error, result) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|
|
|
|
function getWithMembers(id, callback) {
|
|
assert.strictEqual(typeof id, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.getWithMembers(id, function (error, result) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|
|
|
|
function getAll(callback) {
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.getAll(function (error, result) {
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|
|
|
|
function getAllWithMembers(callback) {
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.getAllWithMembers(function (error, result) {
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|
|
|
|
function getMembers(groupId, callback) {
|
|
assert.strictEqual(typeof groupId, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.getMembers(groupId, function (error, result) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|
|
|
|
function getGroups(userId, callback) {
|
|
assert.strictEqual(typeof userId, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.getGroups(userId, function (error, result) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|
|
|
|
function setGroups(userId, groupIds, callback) {
|
|
assert.strictEqual(typeof userId, 'string');
|
|
assert(Array.isArray(groupIds));
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.setGroups(userId, groupIds, function (error) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null);
|
|
});
|
|
}
|
|
|
|
function addMember(groupId, userId, callback) {
|
|
assert.strictEqual(typeof groupId, 'string');
|
|
assert.strictEqual(typeof userId, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.addMember(groupId, userId, function (error) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null);
|
|
});
|
|
}
|
|
|
|
function setMembers(groupId, userIds, callback) {
|
|
assert.strictEqual(typeof groupId, 'string');
|
|
assert(Array.isArray(userIds));
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.setMembers(groupId, userIds, function (error) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND, 'Invalid group or user id'));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null);
|
|
});
|
|
}
|
|
|
|
function removeMember(groupId, userId, callback) {
|
|
assert.strictEqual(typeof groupId, 'string');
|
|
assert.strictEqual(typeof userId, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.removeMember(groupId, userId, function (error) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null);
|
|
});
|
|
}
|
|
|
|
function isMember(groupId, userId, callback) {
|
|
assert.strictEqual(typeof groupId, 'string');
|
|
assert.strictEqual(typeof userId, 'string');
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
groupdb.isMember(groupId, userId, function (error, result) {
|
|
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new GroupError(GroupError.NOT_FOUND));
|
|
if (error) return callback(new GroupError(GroupError.INTERNAL_ERROR, error));
|
|
|
|
return callback(null, result);
|
|
});
|
|
}
|