src/Controller/ResetPasswordController.php line 133

Open in your IDE?
  1. <?php
  2. namespace App\Controller;
  3. use App\Entity\User;
  4. use App\Form\ChangePasswordFormType;
  5. use App\Form\ResetPasswordRequestFormType;
  6. use Symfony\Bridge\Twig\Mime\TemplatedEmail;
  7. use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
  8. use Symfony\Component\HttpFoundation\RedirectResponse;
  9. use Symfony\Component\HttpFoundation\Request;
  10. use Symfony\Component\HttpFoundation\Response;
  11. use Symfony\Component\Mailer\MailerInterface;
  12. use Symfony\Component\Mime\Address;
  13. use Symfony\Component\Routing\Annotation\Route;
  14. use Symfony\Component\Security\Core\Encoder\UserPasswordEncoderInterface;
  15. use SymfonyCasts\Bundle\ResetPassword\Controller\ResetPasswordControllerTrait;
  16. use SymfonyCasts\Bundle\ResetPassword\Exception\ResetPasswordExceptionInterface;
  17. use SymfonyCasts\Bundle\ResetPassword\ResetPasswordHelperInterface;
  18. /**
  19. * @Route("/reset-password")
  20. */
  21. class ResetPasswordController extends AbstractController
  22. {
  23. use ResetPasswordControllerTrait;
  24. private $resetPasswordHelper;
  25. public function __construct(ResetPasswordHelperInterface $resetPasswordHelper)
  26. {
  27. $this->resetPasswordHelper = $resetPasswordHelper;
  28. }
  29. /**
  30. * Display & process form to request a password reset.
  31. *
  32. * @Route("", name="app_forgot_password_request")
  33. */
  34. public function request(Request $request, MailerInterface $mailer): Response
  35. {
  36. $form = $this->createForm(ResetPasswordRequestFormType::class);
  37. $form->handleRequest($request);
  38. if ($form->isSubmitted() && $form->isValid()) {
  39. return $this->processSendingPasswordResetEmail(
  40. $form->get('email')->getData(),
  41. $mailer
  42. );
  43. }
  44. return $this->render(
  45. 'reset_password/request.html.twig',
  46. [
  47. 'requestForm' => $form->createView(),
  48. ]
  49. );
  50. }
  51. private function processSendingPasswordResetEmail(string $emailFormData, MailerInterface $mailer): RedirectResponse
  52. {
  53. $user = $this->getDoctrine()->getRepository(User::class)->findOneBy(
  54. [
  55. 'email' => $emailFormData,
  56. ]
  57. );
  58. // Marks that you are allowed to see the app_check_email page.
  59. $this->setCanCheckEmailInSession();
  60. // Do not reveal whether a user account was found or not.
  61. if (!$user) {
  62. return $this->redirectToRoute('app_check_email');
  63. }
  64. try {
  65. $resetToken = $this->resetPasswordHelper->generateResetToken($user);
  66. } catch (ResetPasswordExceptionInterface $e) {
  67. // If you want to tell the user why a reset email was not sent, uncomment
  68. // the lines below and change the redirect to 'app_forgot_password_request'.
  69. // Caution: This may reveal if a user is registered or not.
  70. //
  71. // $this->addFlash('reset_password_error', sprintf(
  72. // 'There was a problem handling your password reset request - %s',
  73. // $e->getReason()
  74. // ));
  75. return $this->redirectToRoute('app_check_email');
  76. }
  77. $email = (new TemplatedEmail())
  78. ->from(new Address('mailer@galab.com', 'Galab Mail Bot'))
  79. ->to($user->getEmail())
  80. ->subject('Your password reset request')
  81. ->htmlTemplate('reset_password/email.html.twig')
  82. ->context(
  83. [
  84. 'resetToken' => $resetToken,
  85. 'tokenLifetime' => $this->resetPasswordHelper->getTokenLifetime(),
  86. ]
  87. );
  88. $mailer->send($email);
  89. return $this->redirectToRoute('app_check_email');
  90. }
  91. /**
  92. * Confirmation page after a user has requested a password reset.
  93. *
  94. * @Route("/check-email", name="app_check_email")
  95. */
  96. public function checkEmail(): Response
  97. {
  98. // We prevent users from directly accessing this page
  99. if (!$this->canCheckEmail()) {
  100. return $this->redirectToRoute('app_forgot_password_request');
  101. }
  102. return $this->render(
  103. 'reset_password/check_email.html.twig',
  104. [
  105. 'tokenLifetime' => $this->resetPasswordHelper->getTokenLifetime(),
  106. ]
  107. );
  108. }
  109. /**
  110. * Validates and process the reset URL that the user clicked in their email.
  111. *
  112. * @Route("/reset/{token}", name="app_reset_password")
  113. */
  114. public function reset(Request $request, UserPasswordEncoderInterface $passwordEncoder, string $token = null): Response
  115. {
  116. if ($token) {
  117. // We store the token in session and remove it from the URL, to avoid the URL being
  118. // loaded in a browser and potentially leaking the token to 3rd party JavaScript.
  119. $this->storeTokenInSession($token);
  120. return $this->redirectToRoute('app_reset_password');
  121. }
  122. $token = $this->getTokenFromSession();
  123. if (null === $token) {
  124. throw $this->createNotFoundException('No reset password token found in the URL or in the session.');
  125. }
  126. try {
  127. $user = $this->resetPasswordHelper->validateTokenAndFetchUser($token);
  128. } catch (ResetPasswordExceptionInterface $e) {
  129. $this->addFlash(
  130. 'reset_password_error',
  131. sprintf(
  132. 'There was a problem validating your reset request - %s',
  133. $e->getReason()
  134. )
  135. );
  136. return $this->redirectToRoute('app_forgot_password_request');
  137. }
  138. // The token is valid; allow the user to change their password.
  139. $form = $this->createForm(ChangePasswordFormType::class);
  140. $form->handleRequest($request);
  141. if ($form->isSubmitted() && $form->isValid()) {
  142. // A password reset token should be used only once, remove it.
  143. $this->resetPasswordHelper->removeResetRequest($token);
  144. // Encode the plain password, and set it.
  145. $encodedPassword = $passwordEncoder->encodePassword(
  146. $user,
  147. $form->get('plainPassword')->getData()
  148. );
  149. $user->setPassword($encodedPassword);
  150. $this->getDoctrine()->getManager()->flush();
  151. // The session is cleaned up after the password has been changed.
  152. $this->cleanSessionAfterReset();
  153. return $this->redirectToRoute('app_home');
  154. }
  155. return $this->render(
  156. 'reset_password/reset.html.twig',
  157. [
  158. 'resetForm' => $form->createView(),
  159. ]
  160. );
  161. }
  162. }