<?php
namespace App\Services;
use App\Enums\Acl\Role;
use App\Exceptions\InvitationNotFoundException;
use App\Helpers\Uuid;
use App\Mail\UserInvite;
use App\Models\User;
use App\Repositories\UserRepository;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Facades\Mail;
class UserInvitationService
{
public function __construct(private readonly UserRepository $userRepository)
{
}
/** @return Collection<array-key, User> */
public function invite(array $emails, Role $role, User $invitor): Collection
{
$role->assertAvailable();
return DB::transaction(function () use ($emails, $role, $invitor) {
return collect($emails)->map(fn ($email) => $this->inviteOne($email, $role, $invitor));
});
}
public function getUserProspectByToken(string $token): User
{
return User::query()->where('invitation_token', $token)->firstOr(static function (): never {
throw new InvitationNotFoundException();
});
}
public function revokeByEmail(string $email): void
{
$user = $this->userRepository->findOneByEmail($email);
throw_unless($user?->is_prospect, new InvitationNotFoundException());
$user->delete();
}
private function inviteOne(string $email, Role $role, User $invitor): User
{
$role->assertAvailable();
/** @var User $invitee */
$invitee = $invitor->organization->users()->create([
'name' => '',
'email' => $email,
'password' => '',
'invited_by_id' => $invitor->id,
'invitation_token' => Uuid::generate(),
'invited_at' => now(),
]);
$invitee->syncRoles($role);
Mail::to($email)->queue(new UserInvite($invitee));
return $invitee;
}
public function accept(string $token, string $name, string $password): User
{
$user = $this->getUserProspectByToken($token);
$user->update(attributes: [
'name' => $name,
'password' => Hash::make($password),
'invitation_token' => null,
'invitation_accepted_at' => now(),
]);
return $user;
}
}