harisvsulaiman / node_acl-1

Access control lists for node applications

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

NODE ACL - Access Control Lists for Node

This module provides a minimalistic ACL implementation inspired by Zend_ACL.

When you develop a website or application you will soon notice that sessions are not enough to protect all the available resources. Avoiding that malicious users access other users content proves a much more complicated task than anticipated. ACL can solve this problem in a flexible and elegant way.

Create roles and assign roles to users. Sometimes it may even be useful to create one role per user, to get the finest granularity possible, while in other situations you will give the asterisk permission for admin kind of functionality.

A Redis, MongoDB and In-Memory based backends are provided built-in in the module. There are other third party backends such as knex based, firebase and elasticsearch. There is also an alternative memory backend that supports regexps.

Forked, improved and renamed from acl to acl2

Breaking changes comparing to the original acl

  • The backend constructors take options object instead of multiple argument.

Original acl:

new ACL.mongodbBackend(db, prefix, useSingle, useRawCollectionNames)
new ACL.redisBackend(redis, prefix)

New acl2:

new ACL.mongodbBackend({ client, db, prefix = "acl_", useSingle, useRawCollectionNames })
new ACL.redisBackend({ redis, prefix = "acl_" })
  • The new default "acl_" prefix for both Redis and MongoDB.

  • The mongodb dependency upgraded from v2 to the latest v3.

  • Both mongodb and redis dependencies moved to devDependencies. You have to install them to your project separately.

  • The minimal supported nodejs version was 0.10, but became the current LTS 10. But, acl2 runs fine on node 8.

  • The first published version of acl2 is 1.0 to be more SemVer compliant.

Other notable changes comparing to the original acl

  • ES6
  • ESLint
  • Prettier
  • Internally use more promises, fewer callbacks for better stack traces
  • Upgraded all possible dependencies
  • Made unit test debuggable, split them by backend type
  • MongoDB backend accepts either client or db objects
  • Removed all possible warnings
  • Run CI tests using multiple MongoDB versions.

Features

  • Users
  • Roles
  • Hierarchies
  • Resources
  • Express middleware for protecting resources.
  • Robust implementation with good unit test coverage.

Installation

Using npm:

npm install acl2

Optionally:

npm install mongodb@3

npm install redis@2

Documentation

Examples

Create your acl module by requiring it and instantiating it with a valid backend instance:

var ACL = require("acl2");

// Using Redis backend
acl = new ACL(new ACL.redisBackend({ client: redisClient }));

// Or Using the memory backend
acl = new ACL(new ACL.memoryBackend());

// Or Using the MongoDB backend
acl = new ACL(new ACL.mongodbBackend({ client: mongoClient }));

See below for full list of backend constructor arguments.

All the following functions return a promise or optionally take a callback with an err parameter as last parameter. We omit them in the examples for simplicity.

Create roles implicitly by giving them permissions:

// guest is allowed to view blogs
acl.allow("guest", "blogs", "view");

// allow function accepts arrays as any parameter
acl.allow("member", "blogs", ["edit", "view", "delete"]);

Users are likewise created implicitly by assigning them roles:

acl.addUserRoles("joed", "guest");

Hierarchies of roles can be created by assigning parents to roles:

acl.addRoleParents("baz", ["foo", "bar"]);

Note that the order in which you call all the functions is irrelevant (you can add parents first and assign permissions to roles later)

acl.allow("foo", ["blogs", "forums", "news"], ["view", "delete"]);

Use the wildcard to give all permissions:

acl.allow("admin", ["blogs", "forums"], "*");

Sometimes is necessary to set permissions on many different roles and resources. This would lead to unnecessary nested callbacks for handling errors. Instead use the following:

acl.allow([
  {
    roles: ["guest", "member"],
    allows: [
      { resources: "blogs", permissions: "get" },
      { resources: ["forums", "news"], permissions: ["get", "put", "delete"] },
    ],
  },
  {
    roles: ["gold", "silver"],
    allows: [
      { resources: "cash", permissions: ["sell", "exchange"] },
      { resources: ["account", "deposit"], permissions: ["put", "delete"] },
    ],
  },
]);

You can check if a user has permissions to access a given resource with isAllowed:

acl.isAllowed("joed", "blogs", "view", function (err, res) {
  if (res) {
    console.log("User joed is allowed to view blogs");
  }
});

Of course arrays are also accepted in this function:

acl.isAllowed("jsmith", "blogs", ["edit", "view", "delete"]);

Note that all permissions must be fulfilled in order to get true.

Sometimes is necessary to know what permissions a given user has over certain resources:

acl.allowedPermissions("james", ["blogs", "forums"], function (
  err,
  permissions
) {
  console.log(permissions);
});

It will return an array of resource:[permissions] like this:

[{ blogs: ["get", "delete"] }, { forums: ["get", "put"] }];

Finally, we provide a middleware for Express for easy protection of resources.

acl.middleware();

We can protect a resource like this:

app.put('/blogs/:id', acl.middleware(), function(req, res, next){}

The middleware will protect the resource named by req.url, pick the user from req.session.userId and check the permission for req.method, so the above would be equivalent to something like this:

acl.isAllowed(req.session.userId, "/blogs/12345", "put");

The middleware accepts 3 optional arguments, that are useful in some situations. For example, sometimes we cannot consider the whole url as the resource:

app.put('/blogs/:id/comments/:commentId', acl.middleware(3), function(req, res, next){}

In this case the resource will be just the three first components of the url (without the ending slash).

It is also possible to add a custom userId or check for other permissions than the method:

app.put('/blogs/:id/comments/:commentId', acl.middleware(3, 'joed', 'post'), function(req, res, next){}

Methods

Adds roles to a given user id.

Arguments

    userId   {String|Number} User id.
    roles    {String|Array} Role(s) to add to the user id.
    callback {Function} Callback called when finished.

Remove roles from a given user.

Arguments

    userId   {String|Number} User id.
    roles    {String|Array} Role(s) to remove to the user id.
    callback {Function} Callback called when finished.

Return all the roles from a given user.

Arguments

    userId   {String|Number} User id.
    callback {Function} Callback called when finished.

Return all users who has a given role.

Arguments

    rolename   {String|Number} User id.
    callback {Function} Callback called when finished.

Return boolean whether user has the role

Arguments

    userId   {String|Number} User id.
    rolename {String|Number} role name.
    callback {Function} Callback called when finished.

Adds a parent or parent list to role.

Arguments

    role     {String} Child role.
    parents  {String|Array} Parent role(s) to be added.
    callback {Function} Callback called when finished.

Removes a parent or parent list from role.

If parents is not specified, removes all parents.

Arguments

    role     {String} Child role.
    parents  {String|Array} Parent role(s) to be removed [optional].
    callback {Function} Callback called when finished [optional].

Removes a role from the system.

Arguments

    role     {String} Role to be removed
    callback {Function} Callback called when finished.

### removeResource( resource, function(err) )

Removes a resource from the system

Arguments

    resource {String} Resource to be removed
    callback {Function} Callback called when finished.

Adds the given permissions to the given roles over the given resources.

Arguments

    roles       {String|Array} role(s) to add permissions to.
    resources   {String|Array} resource(s) to add permisisons to.
    permissions {String|Array} permission(s) to add to the roles over the resources.
    callback    {Function} Callback called when finished.

allow( permissionsArray, function(err) )

Arguments

    permissionsArray {Array} Array with objects expressing what permissions to give.
       [{roles:{String|Array}, allows:[{resources:{String|Array}, permissions:{String|Array}]]

    callback         {Function} Callback called when finished.

Remove permissions from the given roles owned by the given role.

Note: we loose atomicity when removing empty role_resources.

Arguments

    role        {String}
    resources   {String|Array}
    permissions {String|Array}
    callback    {Function}

Returns all the allowable permissions a given user have to access the given resources.

It returns an array of objects where every object maps a resource name to a list of permissions for that resource.

Arguments

    userId    {String|Number} User id.
    resources {String|Array} resource(s) to ask permissions for.
    callback  {Function} Callback called when finished.

Checks if the given user is allowed to access the resource for the given permissions (note: it must fulfill all the permissions).

Arguments

    userId      {String|Number} User id.
    resource    {String} resource to ask permissions for.
    permissions {String|Array} asked permissions.
    callback    {Function} Callback called with the result.

Returns true if any of the given roles have the right permissions.

Arguments

    roles       {String|Array} Role(s) to check the permissions for.
    resource    {String} resource to ask permissions for.
    permissions {String|Array} asked permissions.
    callback    {Function} Callback called with the result.

Returns what resources a given role has permissions over.

Arguments

    role        {String|Array} Roles
    callback    {Function} Callback called with the result.

whatResources(role, permissions, function(err, resources) )

Returns what resources a role has the given permissions over.

Arguments

    role        {String|Array} Roles
    permissions {String|Array} Permissions
    callback    {Function} Callback called with the result.

Middleware for express.

To create a custom getter for userId, pass a function(req, res) which returns the userId when called (must not be async).

Arguments

    numPathComponents {Number} number of components in the url to be considered part of the resource name.
    userId            {String|Number|Function} the user id for the acl system (defaults to req.session.userId)
    permissions       {String|Array} the permission(s) to check for (defaults to req.method.toLowerCase())

Creates a MongoDB backend instance.

Arguments

    client    {Object} MongoClient instance. If missing, the `db` will be used. 
    db        {Object} Database instance. If missing, the `client` will be used.
    prefix    {String} Optional collection prefix. Default is "acl_".
    useSingle {Boolean} Create one collection for all resources (defaults to false)

Example:

const client = await require("mongodb").connect("mongodb://127.0.0.1:27017/acl_test");
const ACL = require("acl2");
const acl = new ACL(new ACL.mongodbBackend({ client, useSingle: true }));

redisBackend

Creates a Redis backend instance.

Arguments

    client    {Object} Redis client instance. 
    prefix    {String} Optional prefix. Default is "acl_".

Example:

var client = require("redis").createClient(6379, "127.0.0.1", { no_ready_check: true });
const ACL = require("acl2");
const acl = new ACL(new acl.redisBackend({ client, prefix: "my_acl_prefix_" }));

Tests

Run tests with npm. Requires both local databases running - MongoDB and Redis.

 npm test

You can run tests for Memory, Redis, or MongoDB only like this:

npm run test_memory
npm run test_redis
npm run test_mongo
npm run test_mongo_single

About

Access control lists for node applications

License:MIT License


Languages

Language:JavaScript 100.0%