From 10dde1fd3fbce059ba380153152543c5d1bbb8f0 Mon Sep 17 00:00:00 2001 From: Ali Kazmi Date: Fri, 14 Oct 2022 17:21:53 +0500 Subject: [PATCH] Adds feature to manage direct permissions to user (#685) --- .../Actions/AttachPermissionsToUserAction.php | 30 +++++ .../DetachPermissionsFromUserAction.php | 28 +++++ .../Actions/GetRolePermissionsAction.php | 26 +++++ .../Actions/GetUserPermissionsAction.php | 22 ++++ .../Actions/GetUserRolesAction.php | 22 ++++ .../AuthorizationPermissionsSeeder_1.php | 1 + .../Tasks/AttachPermissionsToUserTask.php | 21 ++++ .../Tasks/DetachPermissionsFromUserTask.php | 23 ++++ .../AttachPermissionsToUserController.php | 22 ++++ .../DetachPermissionsFromUserController.php | 21 ++++ .../GetRolePermissionsController.php | 23 ++++ .../GetUserPermissionsController.php | 23 ++++ .../Controllers/GetUserRolesController.php | 18 +++ .../AttachPermissionsToUserRequest.php | 51 ++++++++ .../DetachPermissionsFromUserRequest.php | 51 ++++++++ .../Requests/GetRolePermissionsRequest.php | 51 ++++++++ .../Requests/GetUserPermissionsRequest.php | 51 ++++++++ .../UI/API/Requests/GetUserRolesRequest.php | 51 ++++++++ .../AttachPermissionsToUser.v1.private.php | 27 +++++ .../DetachPermissionsFromUser.v1.private.php | 27 +++++ .../Routes/GetRolePermissions.v1.private.php | 25 ++++ .../Routes/GetUserPermissions.v1.private.php | 25 ++++ .../UI/API/Routes/GetUserRoles.v1.private.php | 25 ++++ .../API/Routes/_userPermissions.v1.public.php | 48 ++++++++ .../Functional/AttachPermissionToUserTest.php | 105 +++++++++++++++++ .../DetachPermissionFromUserTest.php | 109 ++++++++++++++++++ .../Functional/GetRolePermissionsTest.php | 39 +++++++ .../Functional/GetUserPermissionsTest.php | 40 +++++++ .../API/Tests/Functional/GetUserRolesTest.php | 40 +++++++ .../UI/API/Transformers/UserTransformer.php | 7 ++ 30 files changed, 1052 insertions(+) create mode 100644 app/Containers/AppSection/Authorization/Actions/AttachPermissionsToUserAction.php create mode 100644 app/Containers/AppSection/Authorization/Actions/DetachPermissionsFromUserAction.php create mode 100644 app/Containers/AppSection/Authorization/Actions/GetRolePermissionsAction.php create mode 100644 app/Containers/AppSection/Authorization/Actions/GetUserPermissionsAction.php create mode 100644 app/Containers/AppSection/Authorization/Actions/GetUserRolesAction.php create mode 100644 app/Containers/AppSection/Authorization/Tasks/AttachPermissionsToUserTask.php create mode 100644 app/Containers/AppSection/Authorization/Tasks/DetachPermissionsFromUserTask.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Controllers/AttachPermissionsToUserController.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Controllers/DetachPermissionsFromUserController.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Controllers/GetRolePermissionsController.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserPermissionsController.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserRolesController.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Requests/AttachPermissionsToUserRequest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Requests/DetachPermissionsFromUserRequest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Requests/GetRolePermissionsRequest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Requests/GetUserPermissionsRequest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Requests/GetUserRolesRequest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Routes/AttachPermissionsToUser.v1.private.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Routes/DetachPermissionsFromUser.v1.private.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Routes/GetRolePermissions.v1.private.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Routes/GetUserPermissions.v1.private.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Routes/GetUserRoles.v1.private.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Routes/_userPermissions.v1.public.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Tests/Functional/AttachPermissionToUserTest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Tests/Functional/DetachPermissionFromUserTest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetRolePermissionsTest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserPermissionsTest.php create mode 100644 app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserRolesTest.php diff --git a/app/Containers/AppSection/Authorization/Actions/AttachPermissionsToUserAction.php b/app/Containers/AppSection/Authorization/Actions/AttachPermissionsToUserAction.php new file mode 100644 index 000000000..aace13b92 --- /dev/null +++ b/app/Containers/AppSection/Authorization/Actions/AttachPermissionsToUserAction.php @@ -0,0 +1,30 @@ +run($request->id); + + $permissionIds = (array)$request->permissions_ids; + + $permissions = array_map(static function ($permissionId) { + return app(FindPermissionTask::class)->run($permissionId); + }, $permissionIds); + + return app(AttachPermissionsToUserTask::class)->run($user, $permissions); + } +} diff --git a/app/Containers/AppSection/Authorization/Actions/DetachPermissionsFromUserAction.php b/app/Containers/AppSection/Authorization/Actions/DetachPermissionsFromUserAction.php new file mode 100644 index 000000000..3ab634549 --- /dev/null +++ b/app/Containers/AppSection/Authorization/Actions/DetachPermissionsFromUserAction.php @@ -0,0 +1,28 @@ +run($request->id); + + $permissions = array_map(static function ($permissionId) { + return app(FindPermissionTask::class)->run($permissionId); + }, $request->permissions_ids); + + return app(DetachPermissionsFromUserTask::class)->run($role, $permissions); + } +} diff --git a/app/Containers/AppSection/Authorization/Actions/GetRolePermissionsAction.php b/app/Containers/AppSection/Authorization/Actions/GetRolePermissionsAction.php new file mode 100644 index 000000000..ba4019597 --- /dev/null +++ b/app/Containers/AppSection/Authorization/Actions/GetRolePermissionsAction.php @@ -0,0 +1,26 @@ +run($request->id); + return $role->permissions; + } +} diff --git a/app/Containers/AppSection/Authorization/Actions/GetUserPermissionsAction.php b/app/Containers/AppSection/Authorization/Actions/GetUserPermissionsAction.php new file mode 100644 index 000000000..35f6c914a --- /dev/null +++ b/app/Containers/AppSection/Authorization/Actions/GetUserPermissionsAction.php @@ -0,0 +1,22 @@ +run($request->id); + return $user->permissions; + } +} diff --git a/app/Containers/AppSection/Authorization/Actions/GetUserRolesAction.php b/app/Containers/AppSection/Authorization/Actions/GetUserRolesAction.php new file mode 100644 index 000000000..1027f3add --- /dev/null +++ b/app/Containers/AppSection/Authorization/Actions/GetUserRolesAction.php @@ -0,0 +1,22 @@ +run($request->id); + return $user->roles; + } +} diff --git a/app/Containers/AppSection/Authorization/Data/Seeders/AuthorizationPermissionsSeeder_1.php b/app/Containers/AppSection/Authorization/Data/Seeders/AuthorizationPermissionsSeeder_1.php index f91672a5b..ec0d164bd 100644 --- a/app/Containers/AppSection/Authorization/Data/Seeders/AuthorizationPermissionsSeeder_1.php +++ b/app/Containers/AppSection/Authorization/Data/Seeders/AuthorizationPermissionsSeeder_1.php @@ -17,6 +17,7 @@ public function run(): void $createPermissionTask = app(CreatePermissionTask::class); foreach (array_keys(config('auth.guards')) as $guardName) { $createPermissionTask->run('manage-roles', 'Create, Update, Delete, Get All, Attach/detach permissions to Roles and Get All Permissions.', guardName: $guardName); + $createPermissionTask->run('manage-permissions', 'Create, Update, Delete, Get All, Attach/detach permissions to User.', guardName: $guardName); $createPermissionTask->run('create-admins', 'Create new Users (Admins) from the dashboard.', guardName: $guardName); $createPermissionTask->run('manage-admins-access', 'Assign users to Roles.', guardName: $guardName); $createPermissionTask->run('access-dashboard', 'Access the admins dashboard.', guardName: $guardName); diff --git a/app/Containers/AppSection/Authorization/Tasks/AttachPermissionsToUserTask.php b/app/Containers/AppSection/Authorization/Tasks/AttachPermissionsToUserTask.php new file mode 100644 index 000000000..20cbb96f5 --- /dev/null +++ b/app/Containers/AppSection/Authorization/Tasks/AttachPermissionsToUserTask.php @@ -0,0 +1,21 @@ +givePermissionTo($permissions);; + } +} diff --git a/app/Containers/AppSection/Authorization/Tasks/DetachPermissionsFromUserTask.php b/app/Containers/AppSection/Authorization/Tasks/DetachPermissionsFromUserTask.php new file mode 100644 index 000000000..7f9fb01bd --- /dev/null +++ b/app/Containers/AppSection/Authorization/Tasks/DetachPermissionsFromUserTask.php @@ -0,0 +1,23 @@ +revokePermissionTo($permission); + }, $permissions); + + return $user; + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Controllers/AttachPermissionsToUserController.php b/app/Containers/AppSection/Authorization/UI/API/Controllers/AttachPermissionsToUserController.php new file mode 100644 index 000000000..c6868f5d3 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Controllers/AttachPermissionsToUserController.php @@ -0,0 +1,22 @@ +run($request); + return $this->transform($user, UserTransformer::class, ['permissions']); + } + +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Controllers/DetachPermissionsFromUserController.php b/app/Containers/AppSection/Authorization/UI/API/Controllers/DetachPermissionsFromUserController.php new file mode 100644 index 000000000..9240db299 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Controllers/DetachPermissionsFromUserController.php @@ -0,0 +1,21 @@ +run($request); + return $this->transform($user, UserTransformer::class, ['permissions']); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Controllers/GetRolePermissionsController.php b/app/Containers/AppSection/Authorization/UI/API/Controllers/GetRolePermissionsController.php new file mode 100644 index 000000000..461ca5435 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Controllers/GetRolePermissionsController.php @@ -0,0 +1,23 @@ +run($request); + return $this->transform($permission, PermissionTransformer::class); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserPermissionsController.php b/app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserPermissionsController.php new file mode 100644 index 000000000..b521634d3 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserPermissionsController.php @@ -0,0 +1,23 @@ +run($request); + return $this->transform($permission, PermissionTransformer::class); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserRolesController.php b/app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserRolesController.php new file mode 100644 index 000000000..a503cce6e --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Controllers/GetUserRolesController.php @@ -0,0 +1,18 @@ +run($request); + return $this->transform($roles, RoleTransformer::class); + } + +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Requests/AttachPermissionsToUserRequest.php b/app/Containers/AppSection/Authorization/UI/API/Requests/AttachPermissionsToUserRequest.php new file mode 100644 index 000000000..a62822a81 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Requests/AttachPermissionsToUserRequest.php @@ -0,0 +1,51 @@ + 'manage-permissions', + 'roles' => '', + ]; + + /** + * Id's that needs decoding before applying the validation rules. + */ + protected array $decode = [ + 'id', + ]; + + /** + * Defining the URL parameters (e.g, `/user/{id}`) allows applying + * validation rules on them and allows accessing them like request data. + */ + protected array $urlParameters = [ + 'id', + ]; + + /** + * Get the validation rules that apply to the request. + */ + public function rules(): array + { + return [ + 'permissions_ids' => 'required' + ]; + } + + /** + * Determine if the user is authorized to make this request. + */ + public function authorize(): bool + { + return $this->check([ + 'hasAccess', + ]); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Requests/DetachPermissionsFromUserRequest.php b/app/Containers/AppSection/Authorization/UI/API/Requests/DetachPermissionsFromUserRequest.php new file mode 100644 index 000000000..adc6dd9c3 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Requests/DetachPermissionsFromUserRequest.php @@ -0,0 +1,51 @@ + 'manage-permissions', + 'roles' => '', + ]; + + /** + * Id's that needs decoding before applying the validation rules. + */ + protected array $decode = [ + 'id', + ]; + + /** + * Defining the URL parameters (e.g, `/user/{id}`) allows applying + * validation rules on them and allows accessing them like request data. + */ + protected array $urlParameters = [ + 'id', + ]; + + /** + * Get the validation rules that apply to the request. + */ + public function rules(): array + { + return [ + 'permissions_ids' => 'required', + ]; + } + + /** + * Determine if the user is authorized to make this request. + */ + public function authorize(): bool + { + return $this->check([ + 'hasAccess', + ]); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Requests/GetRolePermissionsRequest.php b/app/Containers/AppSection/Authorization/UI/API/Requests/GetRolePermissionsRequest.php new file mode 100644 index 000000000..c0bd78d9f --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Requests/GetRolePermissionsRequest.php @@ -0,0 +1,51 @@ + '', + 'roles' => '', + ]; + + /** + * Id's that needs decoding before applying the validation rules. + */ + protected array $decode = [ + 'id', + ]; + + /** + * Defining the URL parameters (e.g, `/user/{id}`) allows applying + * validation rules on them and allows accessing them like request data. + */ + protected array $urlParameters = [ + 'id', + ]; + + /** + * Get the validation rules that apply to the request. + */ + public function rules(): array + { + return [ + 'id' => 'required', + ]; + } + + /** + * Determine if the user is authorized to make this request. + */ + public function authorize(): bool + { + return $this->check([ + 'hasAccess', + ]); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Requests/GetUserPermissionsRequest.php b/app/Containers/AppSection/Authorization/UI/API/Requests/GetUserPermissionsRequest.php new file mode 100644 index 000000000..66336915b --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Requests/GetUserPermissionsRequest.php @@ -0,0 +1,51 @@ + '', + 'roles' => '', + ]; + + /** + * Id's that needs decoding before applying the validation rules. + */ + protected array $decode = [ + 'id', + ]; + + /** + * Defining the URL parameters (e.g, `/user/{id}`) allows applying + * validation rules on them and allows accessing them like request data. + */ + protected array $urlParameters = [ + 'id', + ]; + + /** + * Get the validation rules that apply to the request. + */ + public function rules(): array + { + return [ + 'id' => 'required', + ]; + } + + /** + * Determine if the user is authorized to make this request. + */ + public function authorize(): bool + { + return $this->check([ + 'hasAccess', + ]); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Requests/GetUserRolesRequest.php b/app/Containers/AppSection/Authorization/UI/API/Requests/GetUserRolesRequest.php new file mode 100644 index 000000000..77dc94a62 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Requests/GetUserRolesRequest.php @@ -0,0 +1,51 @@ + '', + 'roles' => '', + ]; + + /** + * Id's that needs decoding before applying the validation rules. + */ + protected array $decode = [ + 'id', + ]; + + /** + * Defining the URL parameters (e.g, `/user/{id}`) allows applying + * validation rules on them and allows accessing them like request data. + */ + protected array $urlParameters = [ + 'id', + ]; + + /** + * Get the validation rules that apply to the request. + */ + public function rules(): array + { + return [ + // 'id' => 'required', + ]; + } + + /** + * Determine if the user is authorized to make this request. + */ + public function authorize(): bool + { + return $this->check([ + 'hasAccess', + ]); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Routes/AttachPermissionsToUser.v1.private.php b/app/Containers/AppSection/Authorization/UI/API/Routes/AttachPermissionsToUser.v1.private.php new file mode 100644 index 000000000..4215a396d --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Routes/AttachPermissionsToUser.v1.private.php @@ -0,0 +1,27 @@ + 'manage-permissions', 'roles' => ''] + * + * @apiHeader {String} accept=application/json + * @apiHeader {String} authorization=Bearer + * + * @apiBody {String} id user's id + * @apiBody {Array} permission_ids Permission ID or Array of Permissions ID's + * + * @apiUse UserSuccessSingleResponse + */ + +use App\Containers\AppSection\Authorization\UI\API\Controllers\AttachPermissionsToUserController; +use Illuminate\Support\Facades\Route; + +Route::post('users/{id}/permissions/attach', [AttachPermissionsToUserController::class, 'attachPermissionsToUser']) + ->middleware(['auth:api']); + diff --git a/app/Containers/AppSection/Authorization/UI/API/Routes/DetachPermissionsFromUser.v1.private.php b/app/Containers/AppSection/Authorization/UI/API/Routes/DetachPermissionsFromUser.v1.private.php new file mode 100644 index 000000000..6ecdd9167 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Routes/DetachPermissionsFromUser.v1.private.php @@ -0,0 +1,27 @@ + 'manage-permissions', 'roles' => ''] + * + * @apiHeader {String} accept=application/json + * @apiHeader {String} authorization=Bearer + * + * @apiBody {String} id user's id + * @apiBody {String-Array} permissions_ids Permission ID or Array of Permissions ID's + * + * @apiUse UserSuccessSingleResponse + */ + +use App\Containers\AppSection\Authorization\UI\API\Controllers\DetachPermissionsFromUserController; +use Illuminate\Support\Facades\Route; + +Route::post('users/{id}/permissions/detach', [DetachPermissionsFromUserController::class, 'detachPermissionFromUser']) + ->middleware(['auth:api']); + diff --git a/app/Containers/AppSection/Authorization/UI/API/Routes/GetRolePermissions.v1.private.php b/app/Containers/AppSection/Authorization/UI/API/Routes/GetRolePermissions.v1.private.php new file mode 100644 index 000000000..b8c9b1554 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Routes/GetRolePermissions.v1.private.php @@ -0,0 +1,25 @@ + '', 'roles' => ''] + * + * @apiHeader {String} accept=application/json + * @apiHeader {String} authorization=Bearer + * + * @apiParam {String} id id of role + * + * @apiUse GeneralSuccessMultipleResponse + */ + +use App\Containers\AppSection\Authorization\UI\API\Controllers\GetRolePermissionsController; +use Illuminate\Support\Facades\Route; + +Route::get('roles/{id}/permissions', [GetRolePermissionsController::class, 'getRolePermissions']) + ->middleware(['auth:api']); + diff --git a/app/Containers/AppSection/Authorization/UI/API/Routes/GetUserPermissions.v1.private.php b/app/Containers/AppSection/Authorization/UI/API/Routes/GetUserPermissions.v1.private.php new file mode 100644 index 000000000..e710c8171 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Routes/GetUserPermissions.v1.private.php @@ -0,0 +1,25 @@ + '', 'roles' => ''] + * + * @apiHeader {String} accept=application/json + * @apiHeader {String} authorization=Bearer + * + * @apiParam {String} id id of user + * + * @apiUse GeneralSuccessMultipleResponse + */ + +use App\Containers\AppSection\Authorization\UI\API\Controllers\GetUserPermissionsController; +use Illuminate\Support\Facades\Route; + +Route::get('users/{id}/permissions', [GetUserPermissionsController::class, 'getUserPermissions']) + ->middleware(['auth:api']); + diff --git a/app/Containers/AppSection/Authorization/UI/API/Routes/GetUserRoles.v1.private.php b/app/Containers/AppSection/Authorization/UI/API/Routes/GetUserRoles.v1.private.php new file mode 100644 index 000000000..c0216fa6d --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Routes/GetUserRoles.v1.private.php @@ -0,0 +1,25 @@ + '', 'roles' => ''] + * + * @apiHeader {String} accept=application/json + * @apiHeader {String} authorization=Bearer + * + * @apiParam {String} id user id + * + * @apiUse GeneralSuccessMultipleResponse + */ + +use App\Containers\AppSection\Authorization\UI\API\Controllers\GetUserRolesController; +use Illuminate\Support\Facades\Route; + +Route::get('users/{id}/roles', [GetUserRolesController::class, 'getUserRoles']) + ->middleware(['auth:api']); + diff --git a/app/Containers/AppSection/Authorization/UI/API/Routes/_userPermissions.v1.public.php b/app/Containers/AppSection/Authorization/UI/API/Routes/_userPermissions.v1.public.php new file mode 100644 index 000000000..ad7811790 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Routes/_userPermissions.v1.public.php @@ -0,0 +1,48 @@ + 'manage-permissions', + 'roles' => '', + ]; + + public function testAttachSinglePermissionToUser(): void + { + $user = User::factory()->create(); + + $permission = Permission::factory()->create(); + $data = [ + 'permissions_ids' => $permission->id + ]; + + // send the HTTP request + $response = $this->injectId($user->id)->makeCall($data); + + // assert the response status + $response->assertStatus(200); + $response->assertJson( + fn(AssertableJson $json) => $json->has('data') + ->where('data.object', 'User') + ->where('data.id', $user->getHashedKey()) + ->has('data.permissions.data',1) + ->where('data.permissions.data.0.object', 'Permission') + ->where('data.permissions.data.0.id', $permission->getHashedKey()) + ->etc() + ); + } + + public function testAttachMultiplePermissionsToUser(): void + { + $user = User::factory()->create(); + $permissionA = Permission::factory()->create(); + $permissionB = Permission::factory()->create(); + $data = [ + 'permissions_ids' => [$permissionA->id, $permissionB->id] + ]; + + $response = $this->injectId($user->id)->makeCall($data); + + $response->assertStatus(200); + $response->assertJson( + fn(AssertableJson $json) => $json->has('data') + ->where('data.object', 'User') + ->where('data.id', $user->getHashedKey()) + ->has('data.permissions.data', 2) + ->where('data.permissions.data.0.object', 'Permission') + ->where('data.permissions.data.0.id', $permissionA->getHashedKey()) + ->where('data.permissions.data.1.id', $permissionB->getHashedKey()) + ->etc() + ); + } + + public function testAttachNonExistingPermissionToUser(): void + { + $user = User::factory()->create(); + $invalidId = 3333; + $data = [ + 'permissions_ids' => $invalidId + ]; + + $response = $this->injectId($user->id)->makeCall($data); + + $response->assertStatus(404); + } + + public function testAttachPermissionToNonExistingUser(): void + { + $permission = Permission::factory()->create(); + $invalidId = 7777; + $data = [ + 'permissions_ids' => $permission->id + ]; + + $response = $this->injectId($invalidId)->makeCall($data); + + $response->assertStatus(404); + } +} + + + diff --git a/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/DetachPermissionFromUserTest.php b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/DetachPermissionFromUserTest.php new file mode 100644 index 000000000..b283f56eb --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/DetachPermissionFromUserTest.php @@ -0,0 +1,109 @@ + 'manage-permissions', + 'roles' => '', + ]; + + + public function testDetachSinglePermissionFromUser(): void + { + $user = User::factory()->create(); + $permissionA = Permission::factory()->create(); + $permissionB = Permission::factory()->create(); + $user->givePermissionTo([$permissionA, $permissionB]); + + $data = [ + 'permissions_ids' => [$permissionA->id] + ]; + + // send the HTTP request + $response = $this->injectId($user->id)->makeCall($data); + // assert the response status + $response->assertStatus(200); + $response->assertJson( + fn(AssertableJson $json) => $json->has('data') + ->where('data.object', 'User') + ->where('data.id', $user->getHashedKey()) + ->has('data.permissions.data',1) + ->where('data.permissions.data.0.object', 'Permission') + ->where('data.permissions.data.0.id', $permissionB->getHashedKey()) + ->etc() + ); + } + + public function testDetachMultiplePermissionFromUser(): void + { + $user = User::factory()->create(); + $permissionA = Permission::factory()->create(); + $permissionB = Permission::factory()->create(); + $permissionC = Permission::factory()->create(); + + $user->givePermissionTo([$permissionA, $permissionB, $permissionC]); + + $data = [ + 'permissions_ids' => [$permissionA->id, $permissionB->id] + ]; + + // send the HTTP request + $response = $this->injectId($user->id)->makeCall($data); + + // assert the response status + $response->assertStatus(200); + $response->assertJson( + fn(AssertableJson $json) => $json->has('data') + ->where('data.object', 'User') + ->where('data.id', $user->getHashedKey()) + ->count('data.permissions.data', 1) + ->where('data.permissions.data.0.id', $permissionC->getHashedKey()) + ->etc() + ); + } + + public function testDetachNonExistingPermissionFromUser() + { + $invalidId = 3333; + $user = User::factory()->create(); + $data = [ + 'permissions_ids' => [Hashids::encode($invalidId)] + ]; + + $response = $this->injectId($user->id)->makeCall($data); + + $response->assertStatus(404); + } + + public function testDetachPermissionFromNonExistingUser() + { + $invalidId = 3333; + $permission = Permission::factory()->create(); + $data = [ + //'user_id' => Hashids::encode($invalidId), + 'permissions_ids' => [$permission->getHashedKey()] + ]; + + $response = $this->injectId($invalidId)->makeCall($data); + + $response->assertStatus(404); + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetRolePermissionsTest.php b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetRolePermissionsTest.php new file mode 100644 index 000000000..07c6290e9 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetRolePermissionsTest.php @@ -0,0 +1,39 @@ + '', + 'roles' => '', + ]; + + public function testGetRolePermissions(): void + { + $role = Role::factory()->create(); + $permission = Permission::factory()->create(); + $role->givePermissionTo([$permission]); + // send the HTTP request + $response = $this->injectId($role->id)->makeCall(); + // assert the response status + $response->assertStatus(200); + $responseContent = $this->getResponseContentObject(); + $this->assertEquals($permission->name, $responseContent->data[0]->name); + + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserPermissionsTest.php b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserPermissionsTest.php new file mode 100644 index 000000000..af9733832 --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserPermissionsTest.php @@ -0,0 +1,40 @@ + '', + 'roles' => '', + ]; + + public function testGetUserPermissions(): void + { + $user = User::factory()->create(); + $permission = Permission::factory()->create(); + $user->givePermissionTo([$permission]); + // send the HTTP request + $response = $this->injectId($user->id)->makeCall(); + + // assert the response status + $response->assertStatus(200); + $responseContent = $this->getResponseContentObject(); + $this->assertEquals($permission->name, $responseContent->data[0]->name); + + } +} diff --git a/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserRolesTest.php b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserRolesTest.php new file mode 100644 index 000000000..f112e61bf --- /dev/null +++ b/app/Containers/AppSection/Authorization/UI/API/Tests/Functional/GetUserRolesTest.php @@ -0,0 +1,40 @@ + '', + 'roles' => '', + ]; + + public function testGetUserRoles(): void + { + $user = User::factory()->create(); + $role = Role::factory()->create(); + $user->assignRole($role); + // send the HTTP request + $response = $this->injectId($user->id)->makeCall(); + + // assert the response status + $response->assertStatus(200); + $responseContent = $this->getResponseContentObject(); + $this->assertEquals($role->name, $responseContent->data[0]->name); + + } +} diff --git a/app/Containers/AppSection/User/UI/API/Transformers/UserTransformer.php b/app/Containers/AppSection/User/UI/API/Transformers/UserTransformer.php index 5b7cee1d2..980cf432b 100644 --- a/app/Containers/AppSection/User/UI/API/Transformers/UserTransformer.php +++ b/app/Containers/AppSection/User/UI/API/Transformers/UserTransformer.php @@ -2,6 +2,7 @@ namespace App\Containers\AppSection\User\UI\API\Transformers; +use App\Containers\AppSection\Authorization\UI\API\Transformers\PermissionTransformer; use App\Containers\AppSection\Authorization\UI\API\Transformers\RoleTransformer; use App\Containers\AppSection\User\Models\User; use App\Ship\Parents\Transformers\Transformer as ParentTransformer; @@ -11,6 +12,7 @@ class UserTransformer extends ParentTransformer { protected array $availableIncludes = [ 'roles', + 'permissions' ]; protected array $defaultIncludes = [ @@ -42,4 +44,9 @@ public function includeRoles(User $user): Collection { return $this->collection($user->roles, new RoleTransformer()); } + + public function includePermissions(User $user): Collection + { + return $this->collection($user->permissions, new PermissionTransformer()); + } }