UserManagerController.php 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. <?php
  2. namespace App\Api\v1\Controllers;
  3. use App\Api\v1\Requests\UserManagerStoreRequest;
  4. use App\Api\v1\Requests\UserManagerPromoteRequest;
  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. }
  58. else {
  59. return response()->json([
  60. 'message' => 'bad request',
  61. 'reason' => is_string($response) ? __($response) : __('errors.no_pwd_reset_for_this_user_type')
  62. ], 400);
  63. }
  64. return $response == Password::RESET_LINK_SENT
  65. ? new UserManagerResource($user)
  66. : response()->json([
  67. 'message' => 'bad request',
  68. 'reason' => __($response)
  69. ], 400);
  70. }
  71. /**
  72. * Store a newly created user in storage.
  73. *
  74. * @return \Illuminate\Http\JsonResponse
  75. */
  76. public function store(UserManagerStoreRequest $request)
  77. {
  78. $validated = $request->validated();
  79. $user = User::create([
  80. 'name' => $validated['name'],
  81. 'email' => $validated['email'],
  82. 'password' => Hash::make($validated['password']),
  83. ]);
  84. Log::info(sprintf('User ID #%s created by user ID #%s', $user->id, $request->user()->id));
  85. if ($validated['is_admin']) {
  86. $user->promoteToAdministrator();
  87. $user->save();
  88. Log::notice(sprintf('User ID #%s set as administrator at creation by user ID #%s', $user->id, $request->user()->id));
  89. }
  90. $user->refresh();
  91. return (new UserManagerResource($user))
  92. ->response()
  93. ->setStatusCode(201);
  94. }
  95. /**
  96. * Purge user's PATs.
  97. *
  98. * @return \Illuminate\Http\JsonResponse
  99. */
  100. public function revokePATs(Request $request, User $user, TokenRepository $tokenRepository)
  101. {
  102. Log::info(sprintf('Deletion of all personal access tokens for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
  103. $tokens = $tokenRepository->forUser($user->getAuthIdentifier());
  104. $tokens->load('client')->filter(function ($token) {
  105. return $token->client->personal_access_client && ! $token->revoked;
  106. })->each(function ($token) {
  107. $token->revoke();
  108. });
  109. Log::info(sprintf('All personal access tokens for User ID #%s have been revoked', $user->id));
  110. return response()->json(null, 204);
  111. }
  112. /**
  113. * Purge user's webauthn credentials.
  114. *
  115. * @return \Illuminate\Http\JsonResponse
  116. */
  117. public function revokeWebauthnCredentials(Request $request, User $user)
  118. {
  119. Log::info(sprintf('Deletion of all security devices for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
  120. $user->flushCredentials();
  121. // WebauthnOnly user options need to be reset to prevent impossible login when
  122. // no more registered device exists.
  123. // See #110
  124. if (blank($user->webAuthnCredentials()->WhereEnabled()->get())) {
  125. $user['preferences->useWebauthnOnly'] = false;
  126. $user->save();
  127. Log::notice(sprintf('No more Webauthn credential for user ID #%s, useWebauthnOnly user preference reset to false', $user->id));
  128. }
  129. Log::info(sprintf('All security devices for User ID #%s have been revoked', $user->id));
  130. return response()->json(null, 204);
  131. }
  132. /**
  133. * Remove the specified user from storage.
  134. *
  135. * @return \Illuminate\Http\JsonResponse
  136. */
  137. public function destroy(Request $request, User $user)
  138. {
  139. // This will delete the user and all its 2FAs & Groups thanks to the onCascadeDelete constrains.
  140. // Deletion will not be done (and returns False) if the user is the only existing admin (see UserObserver clas)
  141. return $user->delete() === false
  142. ? response()->json([
  143. 'message' => __('errors.cannot_delete_the_only_admin'),
  144. ], 403)
  145. : response()->json(null, 204);
  146. }
  147. /**
  148. * Update a user
  149. *
  150. * @return \App\Api\v1\Resources\UserManagerResource
  151. */
  152. public function promote(UserManagerPromoteRequest $request, User $user)
  153. {
  154. $user->promoteToAdministrator($request->validated('is_admin'));
  155. $user->save();
  156. Log::info(sprintf('User ID #%s set is_admin=%s for User ID #%s', $request->user()->id, $user->isAdministrator(), $user->id));
  157. return new UserManagerResource($user);
  158. }
  159. /**
  160. * Get the broker to be used during password reset.
  161. *
  162. * @return \Illuminate\Contracts\Auth\PasswordBroker|\Illuminate\Auth\Passwords\PasswordBroker
  163. */
  164. protected function broker()
  165. {
  166. return Password::broker();
  167. }
  168. }