2016-03-03 06:35:01 +08:00
|
|
|
<?php namespace BookStack\Repos;
|
|
|
|
|
|
|
|
|
|
|
|
use BookStack\Exceptions\PermissionsException;
|
|
|
|
use BookStack\Permission;
|
|
|
|
use BookStack\Role;
|
2016-04-24 23:54:20 +08:00
|
|
|
use BookStack\Services\RestrictionService;
|
2016-03-03 06:35:01 +08:00
|
|
|
use Setting;
|
|
|
|
|
|
|
|
class PermissionsRepo
|
|
|
|
{
|
|
|
|
|
|
|
|
protected $permission;
|
|
|
|
protected $role;
|
2016-04-24 23:54:20 +08:00
|
|
|
protected $restrictionService;
|
2016-03-03 06:35:01 +08:00
|
|
|
|
2016-05-02 02:36:53 +08:00
|
|
|
protected $systemRoles = ['admin', 'public'];
|
|
|
|
|
2016-03-03 06:35:01 +08:00
|
|
|
/**
|
|
|
|
* PermissionsRepo constructor.
|
2016-04-24 23:54:20 +08:00
|
|
|
* @param Permission $permission
|
|
|
|
* @param Role $role
|
|
|
|
* @param RestrictionService $restrictionService
|
2016-03-03 06:35:01 +08:00
|
|
|
*/
|
2016-04-24 23:54:20 +08:00
|
|
|
public function __construct(Permission $permission, Role $role, RestrictionService $restrictionService)
|
2016-03-03 06:35:01 +08:00
|
|
|
{
|
|
|
|
$this->permission = $permission;
|
|
|
|
$this->role = $role;
|
2016-04-24 23:54:20 +08:00
|
|
|
$this->restrictionService = $restrictionService;
|
2016-03-03 06:35:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all the user roles from the system.
|
|
|
|
* @return \Illuminate\Database\Eloquent\Collection|static[]
|
|
|
|
*/
|
|
|
|
public function getAllRoles()
|
|
|
|
{
|
2016-05-02 02:36:53 +08:00
|
|
|
return $this->role->where('hidden', '=', false)->get();
|
2016-03-03 06:35:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all the roles except for the provided one.
|
|
|
|
* @param Role $role
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function getAllRolesExcept(Role $role)
|
|
|
|
{
|
2016-05-02 02:36:53 +08:00
|
|
|
return $this->role->where('id', '!=', $role->id)->where('hidden', '=', false)->get();
|
2016-03-03 06:35:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a role via its ID.
|
|
|
|
* @param $id
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function getRoleById($id)
|
|
|
|
{
|
|
|
|
return $this->role->findOrFail($id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save a new role into the system.
|
|
|
|
* @param array $roleData
|
|
|
|
* @return Role
|
|
|
|
*/
|
|
|
|
public function saveNewRole($roleData)
|
|
|
|
{
|
|
|
|
$role = $this->role->newInstance($roleData);
|
|
|
|
$role->name = str_replace(' ', '-', strtolower($roleData['display_name']));
|
|
|
|
// Prevent duplicate names
|
|
|
|
while ($this->role->where('name', '=', $role->name)->count() > 0) {
|
|
|
|
$role->name .= strtolower(str_random(2));
|
|
|
|
}
|
|
|
|
$role->save();
|
|
|
|
|
|
|
|
$permissions = isset($roleData['permissions']) ? array_keys($roleData['permissions']) : [];
|
|
|
|
$this->assignRolePermissions($role, $permissions);
|
2016-04-24 23:54:20 +08:00
|
|
|
$this->restrictionService->buildEntityPermissionForRole($role);
|
2016-03-03 06:35:01 +08:00
|
|
|
return $role;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates an existing role.
|
|
|
|
* Ensure Admin role always has all permissions.
|
|
|
|
* @param $roleId
|
|
|
|
* @param $roleData
|
2016-05-02 02:36:53 +08:00
|
|
|
* @throws PermissionsException
|
2016-03-03 06:35:01 +08:00
|
|
|
*/
|
|
|
|
public function updateRole($roleId, $roleData)
|
|
|
|
{
|
|
|
|
$role = $this->role->findOrFail($roleId);
|
2016-05-02 02:36:53 +08:00
|
|
|
|
|
|
|
if ($role->hidden) throw new PermissionsException("Cannot update a hidden role");
|
|
|
|
|
2016-03-03 06:35:01 +08:00
|
|
|
$permissions = isset($roleData['permissions']) ? array_keys($roleData['permissions']) : [];
|
|
|
|
$this->assignRolePermissions($role, $permissions);
|
|
|
|
|
|
|
|
if ($role->name === 'admin') {
|
|
|
|
$permissions = $this->permission->all()->pluck('id')->toArray();
|
|
|
|
$role->permissions()->sync($permissions);
|
|
|
|
}
|
|
|
|
|
|
|
|
$role->fill($roleData);
|
|
|
|
$role->save();
|
2016-04-24 23:54:20 +08:00
|
|
|
$this->restrictionService->buildEntityPermissionForRole($role);
|
2016-03-03 06:35:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assign an list of permission names to an role.
|
|
|
|
* @param Role $role
|
|
|
|
* @param array $permissionNameArray
|
|
|
|
*/
|
|
|
|
public function assignRolePermissions(Role $role, $permissionNameArray = [])
|
|
|
|
{
|
|
|
|
$permissions = [];
|
2016-03-05 20:09:09 +08:00
|
|
|
$permissionNameArray = array_values($permissionNameArray);
|
2016-03-03 06:35:01 +08:00
|
|
|
if ($permissionNameArray && count($permissionNameArray) > 0) {
|
|
|
|
$permissions = $this->permission->whereIn('name', $permissionNameArray)->pluck('id')->toArray();
|
|
|
|
}
|
|
|
|
$role->permissions()->sync($permissions);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete a role from the system.
|
|
|
|
* Check it's not an admin role or set as default before deleting.
|
|
|
|
* If an migration Role ID is specified the users assign to the current role
|
|
|
|
* will be added to the role of the specified id.
|
|
|
|
* @param $roleId
|
|
|
|
* @param $migrateRoleId
|
|
|
|
* @throws PermissionsException
|
|
|
|
*/
|
|
|
|
public function deleteRole($roleId, $migrateRoleId)
|
|
|
|
{
|
|
|
|
$role = $this->role->findOrFail($roleId);
|
|
|
|
|
|
|
|
// Prevent deleting admin role or default registration role.
|
2016-05-02 02:36:53 +08:00
|
|
|
if ($role->system_name && in_array($role->system_name, $this->systemRoles)) {
|
|
|
|
throw new PermissionsException('This role is a system role and cannot be deleted');
|
2016-03-06 20:55:08 +08:00
|
|
|
} else if ($role->id == setting('registration-role')) {
|
2016-03-03 06:35:01 +08:00
|
|
|
throw new PermissionsException('This role cannot be deleted while set as the default registration role.');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($migrateRoleId) {
|
|
|
|
$newRole = $this->role->find($migrateRoleId);
|
|
|
|
if ($newRole) {
|
|
|
|
$users = $role->users->pluck('id')->toArray();
|
|
|
|
$newRole->users()->sync($users);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-24 23:54:20 +08:00
|
|
|
$this->restrictionService->deleteEntityPermissionsForRole($role);
|
2016-03-03 06:35:01 +08:00
|
|
|
$role->delete();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|