The EloyekunlePermissionsBundle provides support for a database-backed permissions system in Symfony2. It provides a flexible framework for permissions management that aims to handle common tasks such as flexible Permissions Definitions, Roles Creation and Authorization Checking (using Symfony Voters).
Features include:
- Roles can be stored via Doctrine ORM.
- Flexible & Modular permissions definitions in YAML files. From few permissions to hundreds, this bundle has your back.
- Symfony Voter for Authorization Checking.
- Unit tested.
Installation is a quick (I promise!) 5 step process:
- Download EloyekunlePermissionsBundle using composer
- Enable the Bundle
- Create your Role class
- Configure your User class
- Configure the bundle
- Update your database schema
Open a command console, enter your project directory and execute the following command to download the latest stable version of this bundle:
$ composer require "eloyekunle/permissions-bundle"
This command requires you to have Composer installed globally, as explained in the installation chapter of the Composer documentation.
Then, enable the bundle by adding the following line in the config/bundles.php
file of your project, e.g. (Symfony >=4):
// config/bundles.php
return [
// ...
Eloyekunle\PermissionsBundle\EloyekunlePermissionsBundle::class => ['all' => true],
];
The goal of this bundle is to persist some Role
class to a database.
Your first job, then, is to create the Role
class
for your application. This class can look and act however you want: add any
properties or methods you find useful. This is your Role
class.
The bundle provides base classes which are already mapped for most fields to make it easier to create your entity. Here is how you use it:
- Extend the base
Role
class (from theModel
folder if you are using any of the doctrine variants) - Map the
id
field. It must be protected as it is inherited from the parent class. - When you extend from the mapped superclass provided by the bundle, don't redefine the mapping for the other fields as it is provided by the bundle.
- If you override the __construct() method in your Role class, be sure to call parent::__construct(), as the base Role class depends on this to initialize some fields.
If you're persisting your roles via the Doctrine ORM, then your Role
class
should live in the Entity
namespace of your bundle and look like this to
start:
<?php
// src/App/Entity/Role.php
namespace App\Entity;
use Eloyekunle\PermissionsBundle\Model\Role as BaseRole;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Entity
* @ORM\Table(name="role")
*/
class Role extends BaseRole
{
/**
* @ORM\Id
* @ORM\Column(type="integer")
* @ORM\GeneratedValue(strategy="AUTO")
*/
protected $id;
public function __construct()
{
parent::__construct();
// your own logic
}
}
The bundle currently ships with a User
class which you can extend from your own User
entity.
<?php
// src/App/Entity/User.php
namespace App\Entity;
use Eloyekunle\PermissionsBundle\Model\User as BaseUser;
class User extends BaseUser
{
public function __construct()
{
parent::__construct();
// your own logic
}
}
If you already extend another base user class in your User
entity (e.g. from the excellent FOSUserBundle),
you will need to implement Eloyekunle\PermissionsBundle\Model\UserInterface
in your entity and implement the methods (especially UserInterface::hasPermission
). You can view the User
Entity for sample implementation details.
To configure the bundle, add your custom Role
class that was created above, and also a path where your permissions
will be defined. E.g.
# config/packages/eloyekunle_permissions.yaml
eloyekunle_permissions:
role_class: App\Entity\Role
module:
definitions_path: '%kernel.project_dir%/config/modules'
Now that the bundle is configured, the last thing you need to do is update your
database schema because you have added a new entity, the Role
class which you
created in Step 3.
Run the following command.
$ php bin/console doctrine:schema:update --force
Or to create and execute a migration:
$ php bin/console doctrine:migrations:diff
$ php bin/console doctrine:migrations:migrate
To start using the bundle:
This bundle currently supports a modular form of defining permissions, since a lot of projects usually have various
components e.g. Content Management, User Management, Comments, Files etc. You can easily separate your permissions
into various YAML files in your config/modules
directory (or any directory as long as you specify it in the
config as described above).
For example:
# config/modules/content.yaml
# This shows the basic expected structure of the permissions definitions.
name: 'Content'
permissions:
# The key is the important stuff here. You can add your own fields too...
edit content:
# E.g. Add a human-readable name for the permission.
title: 'Edit Content'
description: 'Grants permission to edit content.'
dependencies:
- view content
view content:
title: 'View Content'
description: 'Grants permission to view content.'
The bundle ships with a PermissionsHandler
that is available as a service and can be injected into your Controllers/Services. You can use it to get all available
permissions.
<?php
// src/Controller/PermissionsController.php
namespace App\Controller;
use Eloyekunle\PermissionsBundle\Util\PermissionsHandler;
class PermissionsController {
public function getPermissions(PermissionsHandler $permissionsHandler)
{
/*
* @var array
*
* array (
* 'edit content' =>
* array (
* 'title' => 'Edit Content',
* 'description' => 'Grants permission to edit content.',
* 'dependencies' =>
* array (
* 0 => 'view content',
* ),
* 'provider' => 'content',
* ),
* 'view content' =>
* array (
* 'title' => 'View Content',
* 'description' => 'Grants permission to view content.',
* 'provider' => 'content',
* ),
* )
*/
$permissions = $permissionsHandler->getPermissions();
// ........................
}
}
The bundle ships with a RoleManager
which is available as a service and can be injected into your Controllers/Services. It contains useful utility methods
to manager roles. You can also get
it from the container as eloyekunle_permissions.role_manager
.
<?php
// src/App/Controller/RoleController.php
namespace App\Controller;
use Eloyekunle\PermissionsBundle\Doctrine\RoleManager;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;
class RoleController extends Controller
{
public function create(RoleManager $roleManager) {
$role = $roleManager->createRole();
$role->setRole('Content Admin');
$roleManager->updateRole($role);
// ......
}
}
This creates and persists a Role
entity to your database.
The bundle ships with a voter, PermissionsVoter
.
You can check for user permissions by using the isGranted()
method on Symfony's authorization checker or call
denyAccessUnlessGranted()
in a controller.
<?php
// src/App/Controller/ContentController.php
namespace App\Controller;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;
use Symfony\Component\Routing\Annotation\Route;
class ContentController extends Controller
{
/**
* @Route("/content/{id}/edit", name="content_edit")
*
*
*/
public function edit($id)
{
$this->denyAccessUnlessGranted('edit content');
// Get a Content object - e.g. query for it.
// $content = ......;
}
/**
* @Route("/content/{id}", name="content_show")
*/
public function show($id)
{
$this->denyAccessUnlessGranted('view content');
// Get a Content object - e.g. query for it.
// $content = ......;
}
}
Contributions of any kind are welcome: code, documentation, ideas etc. Issues and feature requests are tracked in the Github issue tracker.
If you need any support related to this bundle, you can contact me on the Symfony Slack group (eloyekunle), or send me an email (eloyekunle@gmail.com).
- Bundle inspired by the Drupal Permissions System!
- Implementation inspired by some excellent Symfony bundles, especially FOSUserBundle.
- Elijah Oyekunle - LinkedIn - Twitter - Github