vendor/pimcore/portal-engine/src/Controller/AuthController.php line 44

Open in your IDE?
  1. <?php
  2. /**
  3.  * Pimcore
  4.  *
  5.  * This source file is available under following license:
  6.  * - Pimcore Commercial License (PCL)
  7.  *
  8.  *  @copyright  Copyright (c) Pimcore GmbH (http://www.pimcore.org)
  9.  *  @license    http://www.pimcore.org/license     PCL
  10.  */
  11. namespace Pimcore\Bundle\PortalEngineBundle\Controller;
  12. use GuzzleHttp\ClientInterface;
  13. use Pimcore\Bundle\OpenIdConnectBundle\Client\Provider\OpenIdConnectProvider;
  14. use Pimcore\Bundle\OpenIdConnectBundle\Session\Configurator;
  15. use Pimcore\Bundle\PortalEngineBundle\Exception\OutputErrorException;
  16. use Pimcore\Bundle\PortalEngineBundle\Form\LoginForm;
  17. use Pimcore\Bundle\PortalEngineBundle\Form\RecoverPasswordForm;
  18. use Pimcore\Bundle\PortalEngineBundle\Model\View\Notification;
  19. use Pimcore\Bundle\PortalEngineBundle\Service\Content\HeadTitleService;
  20. use Pimcore\Bundle\PortalEngineBundle\Service\Frontend\FrontendNotificationService;
  21. use Pimcore\Bundle\PortalEngineBundle\Service\PortalConfig\PortalConfigService;
  22. use Pimcore\Bundle\PortalEngineBundle\Service\Security\Authentication\OpenIdConnectAuthenticator;
  23. use Pimcore\Bundle\PortalEngineBundle\Service\Security\Authentication\User\PasswordChangeableService;
  24. use Pimcore\Bundle\PortalEngineBundle\Service\Security\Authentication\User\RecoverPasswordService;
  25. use Pimcore\Tool;
  26. use Pimcore\Tool\Session;
  27. use Symfony\Component\Form\FormFactoryInterface;
  28. use Symfony\Component\Form\FormInterface;
  29. use Symfony\Component\HttpFoundation\Request;
  30. use Symfony\Component\HttpFoundation\Response;
  31. use Symfony\Component\HttpFoundation\Session\Attribute\AttributeBagInterface;
  32. use Symfony\Component\HttpFoundation\Session\Attribute\NamespacedAttributeBag;
  33. use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
  34. use Symfony\Component\Routing\Annotation\Route;
  35. use Symfony\Contracts\Translation\LocaleAwareInterface;
  36. use Symfony\Contracts\Translation\TranslatorInterface;
  37. /**
  38.  * @Route("/auth", condition="request.attributes.get('isPortalEngineSite')")
  39.  */
  40. class AuthController extends AbstractSiteController
  41. {
  42.     protected ClientInterface $httpClient;
  43.     /**
  44.      * @param ClientInterface $httpClient
  45.      */
  46.     public function __construct(ClientInterface $httpClient)
  47.     {
  48.         $this->httpClient $httpClient;
  49.     }
  50.     /**
  51.      * @Route("/login",
  52.      *     name="pimcore_portalengine_auth_login"
  53.      * )
  54.      *
  55.      * @return Response
  56.      */
  57.     public function loginAction(Request $requestFormFactoryInterface $formFactoryHeadTitleService $headTitleServiceTranslatorInterface $translatorPortalConfigService $portalConfigServicePasswordChangeableService $passwordChangeableService)
  58.     {
  59.         /** @var string $locale */
  60.         $locale $request->getPreferredLanguage(Tool::getValidLanguages());
  61.         if ($translator instanceof LocaleAwareInterface) {
  62.             $translator->setLocale($locale);
  63.         }
  64.         $request->setLocale($locale);
  65.         $portalName $portalConfigService->getPortalName();
  66.         $oidcProviders $portalConfigService->getCurrentPortalConfig()->getOidcConfigs();
  67.         $guestUserActivated = !empty($portalConfigService->getCurrentPortalConfig()->getGuestUser());
  68.         $headTitleService->setTitle($translator->trans('portal-engine.content.title.auth-login', ['%name%' => $portalName]));
  69.         $loginForm $formFactory->create(LoginForm::class);
  70.         return $this->renderTemplate('@PimcorePortalEngine/auth/login.html.twig', [
  71.             'form' => $loginForm->createView(),
  72.             'loginFailed' => (bool)$request->query->get('loginFailed'),
  73.             'portalName' => $portalName,
  74.             'oidcProviderNames' => array_keys($oidcProviders),
  75.             'guestUserActivated' => $guestUserActivated,
  76.             'showRecoverPassword' => $passwordChangeableService->isPasswordChangeable()
  77.         ]);
  78.     }
  79.     /**
  80.      * @Route("/logout",
  81.      *     name="pimcore_portalengine_auth_logout"
  82.      * )
  83.      *
  84.      * @return \Symfony\Component\HttpFoundation\RedirectResponse
  85.      */
  86.     public function logoutAction(Request $request)
  87.     {
  88.         return $this->redirectToRoute('pimcore_portalengine_auth_login');
  89.     }
  90.     /**
  91.      * @Route("/recover-password",
  92.      *     name="pimcore_portalengine_auth_recover_password"
  93.      * )
  94.      *
  95.      * @return Response
  96.      */
  97.     public function recoverPasswordAction(Request $requestTranslatorInterface $translatorFormFactoryInterface $formFactoryFrontendNotificationService $frontendNotificationServiceRecoverPasswordService $recoverPasswordServicePasswordChangeableService $passwordChangeableServiceHeadTitleService $headTitleService)
  98.     {
  99.         if (!$passwordChangeableService->isPasswordChangeable()) {
  100.             throw new NotFoundHttpException('password not changeable');
  101.         }
  102.         /** @var string $locale */
  103.         $locale $request->getPreferredLanguage(Tool::getValidLanguages());
  104.         if ($translator instanceof LocaleAwareInterface) {
  105.             $translator->setLocale($locale);
  106.         }
  107.         $request->setLocale($locale);
  108.         $headTitleService->setTitle($translator->trans('portal-engine.content.title.auth-recover-password'));
  109.         /** @var FormInterface $recoverPasswordForm */
  110.         $recoverPasswordForm $formFactory
  111.             ->create(RecoverPasswordForm::class)
  112.             ->handleRequest($request);
  113.         if ($request->isMethod(Request::METHOD_POST) && $recoverPasswordForm->isSubmitted() && $recoverPasswordForm->isValid()) {
  114.             try {
  115.                 /** @var string $userIdentifier */
  116.                 $userIdentifier = (string)$recoverPasswordForm->get('userIdentifier')->getData();
  117.                 if (empty($userIdentifier)) {
  118.                     throw new OutputErrorException($translator->trans('portal-engine.auth.user-not-found'));
  119.                 }
  120.                 $recoverPasswordService->recoverPassword($userIdentifier);
  121.                 $frontendNotificationService
  122.                     ->addNotification($translator->trans('portal-engine.auth.password-recover-email'), Notification::HTML_CLASS_SUCCESS);
  123.             } catch (\Exception $e) {
  124.                 if ($e instanceof OutputErrorException) {
  125.                     $frontendNotificationService->addNotification($e->getMessage(), Notification::HTML_CLASS_DANGER);
  126.                 }
  127.             }
  128.         }
  129.         return $this->renderTemplate('@PimcorePortalEngine/auth/recover_password.html.twig', [
  130.             'recoverPasswordForm' => $recoverPasswordForm->createView(),
  131.             'notification' => $frontendNotificationService->getNotification()
  132.         ]);
  133.     }
  134.     /**
  135.      * @Route("/oidc/endpoint",
  136.      *     name="pimcore_portalengine_auth_oidc"
  137.      * )
  138.      *
  139.      * @return \Symfony\Component\HttpFoundation\RedirectResponse|Response
  140.      *
  141.      * @throws \Exception
  142.      */
  143.     public function oidcAction(Request $requestPortalConfigService $portalConfigService)
  144.     {
  145.         if (($request->get('code') || $request->get('error')) && !$request->get('redirect')) {
  146.             $url $request->getSchemeAndHttpHost() . $request->getRequestUri() . '&redirect=1';
  147.             $content = <<<RESPONSE
  148. <html>
  149. <head>
  150. <meta http-equiv="refresh" content="0;URL=' {$url}'"/>
  151. </head>
  152. <body><p>Redirecting to <a href="{$url}">{$url}</a>.</p></body>
  153. </html>
  154. RESPONSE;
  155.             return new Response($content);
  156.         }
  157.         if (!class_exists('Pimcore\\Bundle\\OpenIdConnectBundle\\PimcoreOpenIdConnectBundle')) {
  158.             throw new \Exception('OpenID Connect Bundle not available');
  159.         }
  160.         if ($request->get('error')) {
  161.             throw new \Exception(strip_tags($request->get('error')) . ': ' strip_tags($request->get('error_description')));
  162.         }
  163.         $portalConfig $portalConfigService->getCurrentPortalConfig();
  164.         $session $request->getSession();
  165.         /** @var NamespacedAttributeBag $sessionBag */
  166.         $sessionBag $session->getBag(Configurator::SESSION_BAG_NAME); // @phpstan-ignore-line
  167.         // if coming back from provider, use provider from session
  168.         if ($request->get('code')) {
  169.             $providerName $sessionBag->get(OpenIdConnectAuthenticator::SESSION_KEY_PROVIDER);
  170.         } else {
  171.             $providerName $request->get('provider');
  172.             Session::useSession(function (AttributeBagInterface $sessionBag) use ($providerName) {
  173.                 $sessionBag->set(OpenIdConnectAuthenticator::SESSION_KEY_PROVIDER$providerName);
  174.             }, Configurator::SESSION_BAG_NAME); // @phpstan-ignore-line
  175.             Session::writeClose();
  176.         }
  177.         $oidcConfig $portalConfig->getOidcConfig($providerName);
  178.         if (empty($oidcConfig)) {
  179.             throw new \InvalidArgumentException(sprintf('Provider `%s` unknown.'strip_tags($request->get('provider'))));
  180.         }
  181.         $provider = new OpenIdConnectProvider([   // @phpstan-ignore-line
  182.             'clientId' => $oidcConfig->getClientId(),
  183.             'clientSecret' => $oidcConfig->getClientSecret(),
  184.             'urlDiscovery' => $oidcConfig->getDiscoveryUrl(),
  185.             'redirectUri' => $request->getSchemeAndHttpHost() . $request->getPathInfo(),
  186.             'scopes' => $oidcConfig->getScopes()
  187.         ]);
  188.         $provider->setHttpClient($this->httpClient); // @phpstan-ignore-line
  189.         if (!$request->get('code')) {
  190.             // If we don't have an authorization code then get one
  191.             $authUrl $provider->getAuthorizationUrl(); // @phpstan-ignore-line
  192.             Session::useSession(function (AttributeBagInterface $sessionBag) use ($provider) {
  193.                 $sessionBag->set(OpenIdConnectAuthenticator::SESSION_KEY_STATE$provider->getState()); // @phpstan-ignore-line
  194.             }, Configurator::SESSION_BAG_NAME); // @phpstan-ignore-line
  195.             return $this->redirect($authUrl);
  196.         } elseif (!$request->get('state') || ($request->get('state') !== $sessionBag->get('oauth2state'))) {
  197.             // Check given state against previously stored one to mitigate CSRF attack
  198.             $sessionBag->remove(OpenIdConnectAuthenticator::SESSION_KEY_STATE);
  199.             throw new \Exception('Invalid state');
  200.         }
  201.         return $this->redirectToRoute('pimcore_portalengine_auth_login');
  202.     }
  203. }