UserManagerController.php 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. <?php
  2. namespace App\Api\v1\Controllers;
  3. use App\Api\v1\Requests\UserManagerPromoteRequest;
  4. use App\Api\v1\Requests\UserManagerStoreRequest;
  5. use App\Api\v1\Resources\UserManagerResource;
  6. use App\Http\Controllers\Controller;
  7. use App\Models\User;
  8. use Illuminate\Http\Request;
  9. use Illuminate\Support\Facades\Hash;
  10. use Illuminate\Support\Facades\Log;
  11. use Illuminate\Support\Facades\Password;
  12. use Laravel\Passport\TokenRepository;
  13. class UserManagerController extends Controller
  14. {
  15. /**
  16. * Display all users.
  17. *
  18. * @return \Illuminate\Http\Resources\Json\AnonymousResourceCollection
  19. */
  20. public function index(Request $request)
  21. {
  22. return UserManagerResource::collection(User::all());
  23. }
  24. /**
  25. * Get a user
  26. *
  27. * @return \App\Api\v1\Resources\UserManagerResource
  28. */
  29. public function show(User $user)
  30. {
  31. return new UserManagerResource($user);
  32. }
  33. /**
  34. * Reset user's password
  35. *
  36. * @return \Illuminate\Http\JsonResponse
  37. */
  38. public function resetPassword(Request $request, User $user)
  39. {
  40. Log::info(sprintf('Password reset for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
  41. $credentials = [
  42. 'token' => $this->broker()->createToken($user),
  43. 'email' => $user->email,
  44. 'password' => $user->password,
  45. ];
  46. $response = $this->broker()->reset(
  47. $credentials, function ($user) {
  48. $user->resetPassword();
  49. $user->save();
  50. }
  51. );
  52. if ($response == Password::PASSWORD_RESET) {
  53. Log::info(sprintf('Temporary password set for User ID #%s', $user->id));
  54. $response = $this->broker()->sendResetLink(
  55. ['email' => $credentials['email']]
  56. );
  57. } else {
  58. return response()->json([
  59. 'message' => 'bad request',
  60. 'reason' => is_string($response) ? __($response) : __('errors.no_pwd_reset_for_this_user_type'),
  61. ], 400);
  62. }
  63. return $response == Password::RESET_LINK_SENT
  64. ? new UserManagerResource($user)
  65. : response()->json([
  66. 'message' => 'bad request',
  67. 'reason' => __($response),
  68. ], 400);
  69. }
  70. /**
  71. * Store a newly created user in storage.
  72. *
  73. * @return \Illuminate\Http\JsonResponse
  74. */
  75. public function store(UserManagerStoreRequest $request)
  76. {
  77. $validated = $request->validated();
  78. $user = User::create([
  79. 'name' => $validated['name'],
  80. 'email' => $validated['email'],
  81. 'password' => Hash::make($validated['password']),
  82. ]);
  83. Log::info(sprintf('User ID #%s created by user ID #%s', $user->id, $request->user()->id));
  84. if ($validated['is_admin']) {
  85. $user->promoteToAdministrator();
  86. $user->save();
  87. Log::notice(sprintf('User ID #%s set as administrator at creation by user ID #%s', $user->id, $request->user()->id));
  88. }
  89. $user->refresh();
  90. return (new UserManagerResource($user))
  91. ->response()
  92. ->setStatusCode(201);
  93. }
  94. /**
  95. * Purge user's PATs.
  96. *
  97. * @return \Illuminate\Http\JsonResponse
  98. */
  99. public function revokePATs(Request $request, User $user, TokenRepository $tokenRepository)
  100. {
  101. Log::info(sprintf('Deletion of all personal access tokens for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
  102. $tokens = $tokenRepository->forUser($user->getAuthIdentifier());
  103. $tokens->load('client')->filter(function ($token) {
  104. return $token->client->personal_access_client && ! $token->revoked;
  105. })->each(function ($token) {
  106. $token->revoke();
  107. });
  108. Log::info(sprintf('All personal access tokens for User ID #%s have been revoked', $user->id));
  109. return response()->json(null, 204);
  110. }
  111. /**
  112. * Purge user's webauthn credentials.
  113. *
  114. * @return \Illuminate\Http\JsonResponse
  115. */
  116. public function revokeWebauthnCredentials(Request $request, User $user)
  117. {
  118. Log::info(sprintf('Deletion of all security devices for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
  119. $user->flushCredentials();
  120. // WebauthnOnly user options need to be reset to prevent impossible login when
  121. // no more registered device exists.
  122. // See #110
  123. if (blank($user->webAuthnCredentials()->WhereEnabled()->get())) {
  124. $user['preferences->useWebauthnOnly'] = false;
  125. $user->save();
  126. Log::notice(sprintf('No more Webauthn credential for user ID #%s, useWebauthnOnly user preference reset to false', $user->id));
  127. }
  128. Log::info(sprintf('All security devices for User ID #%s have been revoked', $user->id));
  129. return response()->json(null, 204);
  130. }
  131. /**
  132. * Remove the specified user from storage.
  133. *
  134. * @return \Illuminate\Http\JsonResponse
  135. */
  136. public function destroy(Request $request, User $user)
  137. {
  138. // This will delete the user and all its 2FAs & Groups thanks to the onCascadeDelete constrains.
  139. // Deletion will not be done (and returns False) if the user is the only existing admin (see UserObserver clas)
  140. return $user->delete() === false
  141. ? response()->json([
  142. 'message' => __('errors.cannot_delete_the_only_admin'),
  143. ], 403)
  144. : response()->json(null, 204);
  145. }
  146. /**
  147. * Promote (or demote) a user
  148. *
  149. * @return \App\Api\v1\Resources\UserManagerResource
  150. */
  151. public function promote(UserManagerPromoteRequest $request, User $user)
  152. {
  153. $user->promoteToAdministrator($request->validated('is_admin'));
  154. $user->save();
  155. Log::info(sprintf('User ID #%s set is_admin=%s for User ID #%s', $request->user()->id, $user->isAdministrator(), $user->id));
  156. return new UserManagerResource($user);
  157. }
  158. /**
  159. * Get the broker to be used during password reset.
  160. *
  161. * @return \Illuminate\Contracts\Auth\PasswordBroker|\Illuminate\Auth\Passwords\PasswordBroker
  162. */
  163. protected function broker()
  164. {
  165. return Password::broker();
  166. }
  167. }