2024-01-29 08:53:46 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Api\v1\Controllers;
|
|
|
|
|
2024-03-14 15:09:05 +01:00
|
|
|
use App\Api\v1\Requests\UserManagerPromoteRequest;
|
2024-03-29 09:42:54 +01:00
|
|
|
use App\Api\v1\Requests\UserManagerStoreRequest;
|
2024-06-28 16:13:45 +02:00
|
|
|
use App\Api\v1\Resources\UserAuthenticationResource;
|
2024-01-29 08:53:46 +01:00
|
|
|
use App\Api\v1\Resources\UserManagerResource;
|
|
|
|
use App\Http\Controllers\Controller;
|
|
|
|
use App\Models\User;
|
|
|
|
use Illuminate\Http\Request;
|
2024-05-24 13:50:19 +02:00
|
|
|
use Illuminate\Support\Facades\Artisan;
|
2024-01-29 08:53:46 +01:00
|
|
|
use Illuminate\Support\Facades\Hash;
|
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
use Illuminate\Support\Facades\Password;
|
|
|
|
use Laravel\Passport\TokenRepository;
|
|
|
|
|
|
|
|
class UserManagerController extends Controller
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Display all users.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Http\Resources\Json\AnonymousResourceCollection
|
|
|
|
*/
|
|
|
|
public function index(Request $request)
|
|
|
|
{
|
2024-04-25 17:00:32 +02:00
|
|
|
return UserManagerResource::collection(User::withCount('twofaccounts')->get());
|
2024-01-29 08:53:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a user
|
|
|
|
*
|
|
|
|
* @return \App\Api\v1\Resources\UserManagerResource
|
|
|
|
*/
|
|
|
|
public function show(User $user)
|
|
|
|
{
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('view', $user);
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
return new UserManagerResource($user);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-03-29 09:42:54 +01:00
|
|
|
* Reset user's password
|
2024-01-29 08:53:46 +01:00
|
|
|
*
|
|
|
|
* @return \Illuminate\Http\JsonResponse
|
|
|
|
*/
|
|
|
|
public function resetPassword(Request $request, User $user)
|
|
|
|
{
|
|
|
|
Log::info(sprintf('Password reset for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
|
|
|
|
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('update', $user);
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
$credentials = [
|
|
|
|
'token' => $this->broker()->createToken($user),
|
|
|
|
'email' => $user->email,
|
|
|
|
'password' => $user->password,
|
|
|
|
];
|
|
|
|
|
|
|
|
$response = $this->broker()->reset(
|
|
|
|
$credentials, function ($user) {
|
|
|
|
$user->resetPassword();
|
|
|
|
$user->save();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($response == Password::PASSWORD_RESET) {
|
|
|
|
Log::info(sprintf('Temporary password set for User ID #%s', $user->id));
|
2024-03-29 09:42:54 +01:00
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
$response = $this->broker()->sendResetLink(
|
|
|
|
['email' => $credentials['email']]
|
|
|
|
);
|
2024-03-29 09:42:54 +01:00
|
|
|
} else {
|
2024-01-29 08:53:46 +01:00
|
|
|
return response()->json([
|
|
|
|
'message' => 'bad request',
|
2024-03-29 09:42:54 +01:00
|
|
|
'reason' => is_string($response) ? __($response) : __('errors.no_pwd_reset_for_this_user_type'),
|
2024-01-29 08:53:46 +01:00
|
|
|
], 400);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response == Password::RESET_LINK_SENT
|
|
|
|
? new UserManagerResource($user)
|
|
|
|
: response()->json([
|
|
|
|
'message' => 'bad request',
|
2024-03-29 09:42:54 +01:00
|
|
|
'reason' => __($response),
|
2024-01-29 08:53:46 +01:00
|
|
|
], 400);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Store a newly created user in storage.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Http\JsonResponse
|
|
|
|
*/
|
|
|
|
public function store(UserManagerStoreRequest $request)
|
|
|
|
{
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('create', User::class);
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
$validated = $request->validated();
|
|
|
|
|
|
|
|
$user = User::create([
|
2024-03-29 09:42:54 +01:00
|
|
|
'name' => $validated['name'],
|
|
|
|
'email' => $validated['email'],
|
|
|
|
'password' => Hash::make($validated['password']),
|
2024-01-29 08:53:46 +01:00
|
|
|
]);
|
|
|
|
|
|
|
|
Log::info(sprintf('User ID #%s created by user ID #%s', $user->id, $request->user()->id));
|
|
|
|
|
|
|
|
if ($validated['is_admin']) {
|
2024-04-09 14:41:26 +02:00
|
|
|
if ($user->promoteToAdministrator()) {
|
|
|
|
$user->save();
|
|
|
|
Log::notice(sprintf('User ID #%s set as administrator at creation by user ID #%s', $user->id, $request->user()->id));
|
|
|
|
}
|
2024-01-29 08:53:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$user->refresh();
|
|
|
|
|
|
|
|
return (new UserManagerResource($user))
|
|
|
|
->response()
|
|
|
|
->setStatusCode(201);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Purge user's PATs.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Http\JsonResponse
|
|
|
|
*/
|
|
|
|
public function revokePATs(Request $request, User $user, TokenRepository $tokenRepository)
|
|
|
|
{
|
|
|
|
Log::info(sprintf('Deletion of all personal access tokens for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
|
|
|
|
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('update', $user);
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
$tokens = $tokenRepository->forUser($user->getAuthIdentifier());
|
|
|
|
|
|
|
|
$tokens->load('client')->filter(function ($token) {
|
2024-04-20 19:03:44 +02:00
|
|
|
return $token->client->personal_access_client && ! $token->revoked; /** @phpstan-ignore-line */
|
2024-01-29 08:53:46 +01:00
|
|
|
})->each(function ($token) {
|
2024-04-20 19:03:44 +02:00
|
|
|
$token->revoke(); /** @phpstan-ignore-line */
|
2024-01-29 08:53:46 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
Log::info(sprintf('All personal access tokens for User ID #%s have been revoked', $user->id));
|
|
|
|
|
|
|
|
return response()->json(null, 204);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Purge user's webauthn credentials.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Http\JsonResponse
|
|
|
|
*/
|
|
|
|
public function revokeWebauthnCredentials(Request $request, User $user)
|
|
|
|
{
|
|
|
|
Log::info(sprintf('Deletion of all security devices for User ID #%s requested by User ID #%s', $user->id, $request->user()->id));
|
|
|
|
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('update', $user);
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
$user->flushCredentials();
|
|
|
|
|
|
|
|
// WebauthnOnly user options need to be reset to prevent impossible login when
|
|
|
|
// no more registered device exists.
|
|
|
|
// See #110
|
|
|
|
if (blank($user->webAuthnCredentials()->WhereEnabled()->get())) {
|
|
|
|
$user['preferences->useWebauthnOnly'] = false;
|
|
|
|
$user->save();
|
|
|
|
Log::notice(sprintf('No more Webauthn credential for user ID #%s, useWebauthnOnly user preference reset to false', $user->id));
|
|
|
|
}
|
|
|
|
|
|
|
|
Log::info(sprintf('All security devices for User ID #%s have been revoked', $user->id));
|
|
|
|
|
|
|
|
return response()->json(null, 204);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove the specified user from storage.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Http\JsonResponse
|
|
|
|
*/
|
|
|
|
public function destroy(Request $request, User $user)
|
|
|
|
{
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('delete', $user);
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
// This will delete the user and all its 2FAs & Groups thanks to the onCascadeDelete constrains.
|
|
|
|
// Deletion will not be done (and returns False) if the user is the only existing admin (see UserObserver clas)
|
|
|
|
return $user->delete() === false
|
|
|
|
? response()->json([
|
|
|
|
'message' => __('errors.cannot_delete_the_only_admin'),
|
|
|
|
], 403)
|
|
|
|
: response()->json(null, 204);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-03-15 08:02:36 +01:00
|
|
|
* Promote (or demote) a user
|
2024-01-29 08:53:46 +01:00
|
|
|
*
|
2024-04-20 19:03:44 +02:00
|
|
|
* @return \App\Api\v1\Resources\UserManagerResource|\Illuminate\Http\JsonResponse
|
2024-01-29 08:53:46 +01:00
|
|
|
*/
|
2024-03-14 15:09:05 +01:00
|
|
|
public function promote(UserManagerPromoteRequest $request, User $user)
|
2024-01-29 08:53:46 +01:00
|
|
|
{
|
2024-03-30 15:42:34 +01:00
|
|
|
$this->authorize('promote', $user);
|
|
|
|
|
2024-04-15 00:44:18 +02:00
|
|
|
if ($user->promoteToAdministrator($request->validated('is_admin'))) {
|
2024-04-09 14:41:26 +02:00
|
|
|
$user->save();
|
|
|
|
Log::info(sprintf('User ID #%s set is_admin=%s for User ID #%s', $request->user()->id, $user->isAdministrator(), $user->id));
|
2024-01-29 08:53:46 +01:00
|
|
|
|
2024-04-09 14:41:26 +02:00
|
|
|
return new UserManagerResource($user);
|
|
|
|
}
|
2024-01-29 08:53:46 +01:00
|
|
|
|
2024-04-09 14:41:26 +02:00
|
|
|
return response()->json([
|
|
|
|
'message' => __('errors.cannot_demote_the_only_admin'),
|
|
|
|
], 403);
|
2024-01-29 08:53:46 +01:00
|
|
|
}
|
|
|
|
|
2024-04-15 00:44:18 +02:00
|
|
|
/**
|
|
|
|
* Get the user's authentication logs
|
|
|
|
*
|
2024-04-20 19:03:44 +02:00
|
|
|
* @return \Illuminate\Http\Resources\Json\AnonymousResourceCollection
|
2024-04-15 00:44:18 +02:00
|
|
|
*/
|
|
|
|
public function authentications(Request $request, User $user)
|
|
|
|
{
|
|
|
|
$this->authorize('view', $user);
|
|
|
|
|
2024-05-24 13:50:19 +02:00
|
|
|
// Here we purge the authentication log.
|
|
|
|
// Running the purge command when someone fetchs the auth log
|
|
|
|
// is not very elegant but it's straitforward compared
|
|
|
|
// to a scheduled task, and the delete query is light.
|
|
|
|
// => To enhance.
|
|
|
|
Artisan::call('2fauth:purge-log');
|
|
|
|
|
2024-04-15 00:44:18 +02:00
|
|
|
$validated = $this->validate($request, [
|
2024-04-15 18:39:37 +02:00
|
|
|
'period' => 'sometimes|numeric',
|
2024-04-20 19:03:44 +02:00
|
|
|
'limit' => 'sometimes|numeric',
|
2024-04-15 00:44:18 +02:00
|
|
|
]);
|
|
|
|
|
2024-04-16 15:28:38 +02:00
|
|
|
$authentications = $request->has('period') ? $user->authenticationsByPeriod($validated['period']) : $user->authentications;
|
2024-04-15 18:39:37 +02:00
|
|
|
$authentications = $request->has('limit') ? $authentications->take($validated['limit']) : $authentications;
|
2024-04-15 00:44:18 +02:00
|
|
|
|
2024-06-28 16:13:45 +02:00
|
|
|
return UserAuthenticationResource::collection($authentications);
|
2024-04-15 00:44:18 +02:00
|
|
|
}
|
|
|
|
|
2024-01-29 08:53:46 +01:00
|
|
|
/**
|
|
|
|
* Get the broker to be used during password reset.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Contracts\Auth\PasswordBroker|\Illuminate\Auth\Passwords\PasswordBroker
|
|
|
|
*/
|
|
|
|
protected function broker()
|
|
|
|
{
|
|
|
|
return Password::broker();
|
|
|
|
}
|
|
|
|
}
|