packages/security/src/Services/ResetPassword.php
<?php
namespace Moox\Security\Services;
use DanHarrin\LivewireRateLimiting\Exceptions\TooManyRequestsException;
use DanHarrin\LivewireRateLimiting\WithRateLimiting;
use Filament\Actions\Action;
use Filament\Actions\ActionGroup;
use Filament\Facades\Filament;
use Filament\Forms\Components\Component;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Form;
use Filament\Http\Responses\Auth\Contracts\PasswordResetResponse;
use Filament\Notifications\Notification;
use Filament\Pages\Concerns\InteractsWithFormActions;
use Filament\Pages\SimplePage;
use Illuminate\Auth\Events\PasswordReset;
use Illuminate\Contracts\Auth\Authenticatable;
use Illuminate\Contracts\Auth\CanResetPassword;
use Illuminate\Contracts\Support\Htmlable;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Support\Facades\Password;
use Illuminate\Support\Str;
use Livewire\Attributes\Locked;
use Moox\Security\Helper\PasswordHash;
/**
* @property Form $form
*/
class ResetPassword extends SimplePage
{
use InteractsWithFormActions;
use WithRateLimiting;
/**
* @var view-string
*/
protected static string $view = 'filament-panels::pages.auth.password-reset.reset-password';
#[Locked]
public ?string $email = null;
public ?string $password = '';
public ?string $passwordConfirmation = '';
#[Locked]
public ?string $token = null;
public function mount(?string $email = null, ?string $token = null): void
{
if (Filament::auth()->check()) {
redirect()->intended(Filament::getUrl());
}
$this->token = $token ?? request()->query('token');
$this->form->fill([
'email' => $email ?? request()->query('email'),
]);
}
public function resetPassword(): ?PasswordResetResponse
{
try {
$this->rateLimit(2);
} catch (TooManyRequestsException $exception) {
Notification::make()
->title(__('filament-panels::pages/auth/password-reset/reset-password.notifications.throttled.title', [
'seconds' => $exception->secondsUntilAvailable,
'minutes' => ceil($exception->secondsUntilAvailable / 60),
]))
->body(array_key_exists('body', __('filament-panels::pages/auth/password-reset/reset-password.notifications.throttled') ?: []) ? __('filament-panels::pages/auth/password-reset/reset-password.notifications.throttled.body', [
'seconds' => $exception->secondsUntilAvailable,
'minutes' => ceil($exception->secondsUntilAvailable / 60),
]) : null)
->danger()
->send();
return null;
}
$data = $this->form->getState();
$data['email'] = $this->email;
$data['token'] = $this->token;
$status = Password::broker(Filament::getAuthPasswordBroker())->reset(
$data,
function (CanResetPassword|Model|Authenticatable $user) use ($data) {
$passwordHash = new PasswordHash(8, true);
$user->forceFill([
'user_pass' => $passwordHash->HashPassword($data['password']),
'remember_token' => Str::random(60),
])->save();
event(new PasswordReset($user));
},
);
if ($status === Password::PASSWORD_RESET) {
Notification::make()
->title(__($status))
->success()
->send();
return app(PasswordResetResponse::class);
}
Notification::make()
->title(__($status))
->danger()
->send();
return null;
}
public function form(Form $form): Form
{
return $form
->schema([
$this->getEmailFormComponent(),
$this->getPasswordFormComponent(),
$this->getPasswordConfirmationFormComponent(),
]);
}
protected function getEmailFormComponent(): Component
{
return TextInput::make('email')
->label(__('filament-panels::pages/auth/password-reset/reset-password.form.email.label'))
->disabled()
->autofocus();
}
protected function getPasswordFormComponent(): Component
{
return TextInput::make('password')
->label(__('filament-panels::pages/auth/password-reset/reset-password.form.password.label'))
->password()
->revealable(filament()->arePasswordsRevealable())
->required()
->rules(config('press.password.validation'))
->same('passwordConfirmation')
->validationAttribute(__('filament-panels::pages/auth/password-reset/reset-password.form.password.validation_attribute'))
->helperText(config('press.password.helperText'));
}
protected function getPasswordConfirmationFormComponent(): Component
{
return TextInput::make('passwordConfirmation')
->label(__('filament-panels::pages/auth/password-reset/reset-password.form.password_confirmation.label'))
->password()
->revealable(filament()->arePasswordsRevealable())
->required()
->dehydrated(false);
}
public function getTitle(): string|Htmlable
{
return __('filament-panels::pages/auth/password-reset/reset-password.title');
}
public function getHeading(): string|Htmlable
{
return __('filament-panels::pages/auth/password-reset/reset-password.heading');
}
/**
* @return array<Action | ActionGroup>
*/
protected function getFormActions(): array
{
return [
$this->getResetPasswordFormAction(),
];
}
public function getResetPasswordFormAction(): Action
{
return Action::make('resetPassword')
->label(__('filament-panels::pages/auth/password-reset/reset-password.form.actions.reset.label'))
->submit('resetPassword');
}
protected function hasFullWidthFormActions(): bool
{
return true;
}
}