# Access Control (ACL)

In this page you will understand usage of ACL in our template and how you can use it.

FAQ: Authentication/ACL gives "Unexpected token u in JSON at position 0"

FAQ: ACL - I have role based system for access control instead ability based

# Overview

Unlike previous version of Vuexy, new Vuexy uses CASL package for providing access control. Which is future oriented (Supports Vue 3) and is more detailed on Access Control.

You can find docs for CASL here (opens new window).

CASL may look complex at first so please make sure you first read their docs carefully and understand base logic of Access Control to proceed further.

INFO

We used JSON objects (opens new window) for ability. We are getting user ability in response upon login and updating it once user get logged in.

# ACL Flow

You can find CASL configuration in src/libs/acl folder. Let's explore what each file contains:

  • ability.js: This file exports ACL ability
  • config.js: ACL configuration. Currently, contains initial ability of user.
  • index.js: Main ACL file for injecting ACL in Vue.
  • routeProtection.js: Route protection files.

WARNING

Files might get updated/moved/removed in future updates to make structure more minimal and clutter free.

WARNING

Our current implementation of CASL/ACL stores the user ability in localStorage to keep it persist between browser refreshes/close. Ability array is nested under userData in localStorage. You can check API response of login in our article.

# ability.js

It exports current user's ability. If user is not logged in or user's ability isn't retrieved from localStorage then it uses initialAbility defined in config.js file.

Once user logs in ability is updated using this.$ability.update(newAbility). Sample newAbility array:

newAbility: [
  {
    action: 'read',
    subject: 'User',
  },
]

Also, this same ability is stored in user's object userData in localStorage under ability property.

// User data stored in localStorage
// Key Name: userData

{
  "id": 1,
  "fullName": "John Doe",
  "username": "johndoe",
  "avatar": "/img/13-small.d796bffd.png",
  "email": "admin@demo.com",
  "ability": [
    {
      "action": "manage",
      "subject": "all"
    }
  ],
  // more...
}

TIP

We are getting user ability on refresh from userData which is stored in localStorage. We have stored ability in userData.ability. If you have different approach for storing and retrieving ability then please update src/libs/acl/ability.js file.

# Route Protection

You can protect your route from being visited by other users through as well. We already configured router.beforeEach hook to redirect user to not authorized page if user don't have ability for that route. All you have to do is use route meta to define user ability.






 
 



{
  path: '/access-control',
  name: 'access-control',
  component: () => import('@/views/extensions/acl/AccessControl.vue'),
  meta: {
    resource: 'ACL',
    action: 'read',
  },
},

Here, resource refers to subject in CASL docs (opens new window).

Once you define ability for route using meta, if user don't have defined ability for visiting route then it will get redirected to not authorized page.

If user is not logged in and try to visit the route with ability which user don't have then user will get redirected to login page instead of not authorized page.

Read detailed article on route protection here.

# Omitting Defining resource and action for route

If you do not define resource and action then only user with below ability will be able to visit the route:

{
  action: 'manage',
  subject: 'all',
}

TIP

manage and all are special keywords in CASL. manage represents any action and all represents any subject.

So, in your project if you don't have any user with above mentioned ability and you don't define action and resource for route then no one will be able to visit that route ever.

NOTE

We have user (Admin user) with above mentioned ability in our demo. That's why we don't have to write action and resource on each route.

# Show/Hide Navigation Items

Besides route protection you can also show/hide navigation items based on user ability.

Besides existing options provided by navigation items there's two more option you can specify if you use ACL.

  • resource: This refers to subject in CASL docs
  • action: This is action for ability.

Add these two options in your navigation item to show/hide navigation items based on user ability.

# Omitting defining resource and action for groups

You can optionally define resource & action on navigation item type which have children:

  • Vertical Navigation Menu Group
  • Horizontal Navigation Menu Group
  • Horizontal Navigation Menu Header Group

e.g. If you have vertical navigation menu group and your provided both options for every group child and you want to hide group if all it's child don't have ability then don't worry about adding both option in menu group options. We already handled hiding group if none of it's child is visible. Means, We will hide group if there's no child to render due to lack of ability.

export default [
  {
    title: 'Invoice',
    icon: 'FileIcon',

    /** Both group child have `resource` and `action` defined
     * So, If I omit defining `resource` and `action` for group
     *    => then group will be hidden if both child are hidden
     *
     * Conclusion: You can omit defining `resource` and `action` for group if you want this kind of behavior
     */
    children: [
      {
        title: 'List',
        route: 'invoice-list',
        resource: 'Invoice',
        action: 'read',
      },
      {
        title: 'Trash',
        route: 'invoice-trash',
        resource: 'Invoice',
        action: 'read',
      },
    ],
  },
]

# Group & ACL scenarios

NOTE

"can be viewed" means ACL ability is resolved to true and user can view it. "hidden" means ACL ability is resolved to false and user can't view it.

  1. If group can be viewed can any of it's child can also be viewed then group will be visible.
  2. If Group can be viewed and all it's children are hidden then group also got hidden.
  3. If Group can't be viewed then in all cases group and it's children will be hidden. It will be hidden even if any of or all of it's children can be viewed.

Please note this scenarios apply to all three group types of navigation item:

  • Vertical Navigation Menu Group
  • Horizontal Navigation Menu Group
  • Horizontal Navigation Menu Header Group

CAVEAT

As header in vertical navigation menu is independent of it's child items if you want to hide header if all it's below items are hidden then you need to specify resource & action as well with other following items.

Last Updated: 11/16/2022, 12:36:59 PM