* @link https://www.oxidmodule.com */ declare(strict_types=1); namespace D3\Webauthn\Application\Model; use D3\TestingTools\Production\IsMockable; use D3\Webauthn\Application\Model\Exceptions\WebauthnException; use D3\Webauthn\Application\Model\Exceptions\WebauthnGetException; use D3\Webauthn\Application\Model\Exceptions\WebauthnLoginErrorException; use D3\Webauthn\Modules\Application\Model\d3_User_Webauthn; use OxidEsales\Eshop\Application\Model\User; use OxidEsales\Eshop\Core\Config; use OxidEsales\Eshop\Core\Exception\CookieException; use OxidEsales\Eshop\Core\Exception\UserException; use OxidEsales\Eshop\Core\Session; use OxidEsales\Eshop\Core\Str; use OxidEsales\Eshop\Core\SystemEventHandler; use OxidEsales\Eshop\Core\Utils; use OxidEsales\Eshop\Core\UtilsServer; use OxidEsales\Eshop\Core\UtilsView; use OxidEsales\EshopCommunity\Application\Component\UserComponent; use Psr\Log\LoggerInterface; class WebauthnLogin { use IsMockable; public $credential; public $errorMsg; /** * @param string $credential * @param string|null $error * @throws WebauthnGetException */ public function __construct(string $credential, string $error = null) { $this->setCredential($credential); $this->setErrorMsg($error); } /** * @return string * @throws WebauthnGetException */ public function getCredential(): string { if (!strlen(trim((string) $this->credential))) { /** @var WebauthnGetException $e */ $e = oxNew(WebauthnGetException::class, 'missing credential data'); throw $e; } return trim($this->credential); } /** * @param string $credential * @throws WebauthnGetException */ public function setCredential(string $credential): void { if (!strlen(trim($credential))) { /** @var WebauthnGetException $e */ $e = oxNew(WebauthnGetException::class, 'missing credential data'); throw $e; } $this->credential = trim($credential); } /** * @return ?string */ public function getErrorMsg(): ?string { return $this->errorMsg; } /** * @param string|null $errorMsg */ public function setErrorMsg(?string $errorMsg): void { $this->errorMsg = $errorMsg; } /** * @param UserComponent $usrCmp * @param bool $setSessionCookie * @return void */ public function frontendLogin(UserComponent $usrCmp, bool $setSessionCookie = false) { $myUtilsView = d3GetOxidDIC()->get('d3ox.webauthn.'.UtilsView::class); /** @var d3_User_Webauthn $user */ $user = d3GetOxidDIC()->get('d3ox.webauthn.'.User::class); $userId = $this->getUserId(); try { $this->handleErrorMessage(); $user = $this->assertUser($userId); $this->assertAuthn(); // relogin, don't extract from this try block /** @var User $user */ $user = d3GetOxidDIC()->get('d3ox.webauthn.'.User::class); $usrCmp->setUser($user); $this->setFrontendSession($user); $usrCmp->setLoginStatus(USER_LOGIN_SUCCESS); if ($setSessionCookie) { $this->setSessionCookie($user); } $this->regenerateSessionId(); $usrCmp->setUser($user); return; } catch (UserException $oEx) { // for login component send exception text to a custom component (if defined) $myUtilsView->addErrorToDisplay($oEx, false, true); } catch (CookieException $oEx) { $myUtilsView->addErrorToDisplay($oEx); } catch (WebauthnException $e) { $myUtilsView->addErrorToDisplay($e); d3GetOxidDIC()->get('d3ox.webauthn.'.LoggerInterface::class)->error($e->getDetailedErrorMessage(), ['UserId' => $userId]); d3GetOxidDIC()->get('d3ox.webauthn.'.LoggerInterface::class)->debug($e->getTraceAsString()); } $user->logout(); throw oxNew(WebauthnLoginErrorException::class); } /** * @param string $selectedProfile * @return string */ public function adminLogin(string $selectedProfile): string { $myUtilsView = d3GetOxidDIC()->get('d3ox.webauthn.'.UtilsView::class); /** @var d3_User_Webauthn $user */ $user = d3GetOxidDIC()->get('d3ox.webauthn.'.User::class); $userId = $this->getUserId(); try { $this->handleErrorMessage(); $this->assertUser($userId, true); $this->handleBlockedUser($user); $this->assertAuthn(); $session = $this->setAdminSession($userId); $this->handleBackendCookie(); $this->handleBackendSubshopRights($user, $session); $oEventHandler = d3GetOxidDIC()->get('d3ox.webauthn.'.SystemEventHandler::class); $oEventHandler->onAdminLogin(); $afterLogin = d3GetOxidDIC()->get(WebauthnAfterLogin::class); $afterLogin->setDisplayProfile(); $afterLogin->changeLanguage(); $this->regenerateSessionId(); $this->updateBasket(); return "admin_start"; } catch (UserException $oEx) { $myUtilsView->addErrorToDisplay('LOGIN_ERROR'); } catch (CookieException $oEx) { $myUtilsView->addErrorToDisplay('LOGIN_NO_COOKIE_SUPPORT'); } catch (WebauthnException $e) { $myUtilsView->addErrorToDisplay($e); d3GetOxidDIC()->get('d3ox.webauthn.'.LoggerInterface::class)->error($e->getDetailedErrorMessage(), ['UserId' => $userId]); d3GetOxidDIC()->get('d3ox.webauthn.'.LoggerInterface::class)->debug($e->getTraceAsString()); } $user->logout(); $oStr = Str::getStr(); d3GetOxidDIC()->get('d3ox.webauthn.'.Config::class)->getActiveView() ->addTplParam('user', $oStr->htmlspecialchars($userId)); d3GetOxidDIC()->get('d3ox.webauthn.'.Config::class)->getActiveView() ->addTplParam('profile', $oStr->htmlspecialchars($selectedProfile)); return 'login'; } /** * @throws WebauthnGetException */ public function handleErrorMessage() { $error = $this->getErrorMsg(); if (strlen((string)$error)) { /** @var WebauthnGetException $e */ $e = oxNew(WebauthnGetException::class, $error); throw $e; } } /** * @throws WebauthnGetException|WebauthnException */ public function assertAuthn(): void { $credential = $this->getCredential(); $webAuthn = d3GetOxidDIC()->get(Webauthn::class); $webAuthn->assertAuthn($credential); } /** * @param $userId * @return Session */ public function setAdminSession($userId): Session { /** @var Session $session */ $session = d3GetOxidDIC()->get('d3ox.webauthn.'.Session::class); $adminProfiles = $session->getVariable("aAdminProfiles"); $session->initNewSession(); $session->setVariable("aAdminProfiles", $adminProfiles); $session->setVariable(WebauthnConf::OXID_ADMIN_AUTH, $userId); return $session; } /** * @param User $user * @return void */ public function setSessionCookie(User $user) { if (d3GetOxidDIC()->get('d3ox.webauthn.'.Config::class)->getConfigParam('blShowRememberMe')) { d3GetOxidDIC()->get('d3ox.webauthn.'.UtilsServer::class)->setUserCookie( $user->getFieldData('oxusername'), $user->getFieldData('oxpassword'), d3GetOxidDIC()->get('d3ox.webauthn.'.Config::class)->getShopId() ); } } /** * @param $userId * @param bool $isBackend * @return User * @throws UserException */ public function assertUser($userId, bool $isBackend = false): User { /** @var User $user */ $user = d3GetOxidDIC()->get('d3ox.webauthn.'.User::class); $user->load($userId); if (!$user->isLoaded() || ($isBackend && $user->getFieldData('oxrights') === 'user') ) { /** @var UserException $exc */ $exc = oxNew(UserException::class, 'ERROR_MESSAGE_USER_NOVALIDLOGIN'); throw $exc; } return $user; } /** * @return void * @throws CookieException */ public function handleBackendCookie(): void { $cookie = d3GetOxidDIC()->get('d3ox.webauthn.'.UtilsServer::class)->getOxCookie(); if ($cookie === null) { /** @var CookieException $exc */ $exc = oxNew(CookieException::class, 'ERROR_MESSAGE_COOKIE_NOCOOKIE'); throw $exc; } } /** * @param User $user * @param Session $session * @return void */ public function handleBackendSubshopRights(User $user, Session $session): void { $iSubshop = (int)$user->getFieldData('oxrights'); if ($iSubshop) { $session->setVariable("shp", $iSubshop); $session->setVariable('currentadminshop', $iSubshop); d3GetOxidDIC()->get('d3ox.webauthn.'.Config::class)->setShopId((string) $iSubshop); } } /** * @return void */ public function regenerateSessionId(): void { $oSession = d3GetOxidDIC()->get('d3ox.webauthn.'.Session::class); if ($oSession->isSessionStarted()) { $oSession->regenerateSessionId(); } } public function handleBlockedUser(User $user) { // this user is blocked, deny him if ($user->inGroup('oxidblocked')) { $sUrl = d3GetOxidDIC()->get('d3ox.webauthn.'.Config::class)->getShopHomeUrl() . 'cl=content&tpl=user_blocked.tpl'; d3GetOxidDIC()->get('d3ox.webauthn.'.Utils::class)->redirect($sUrl); } } /** * @return void */ public function updateBasket(): void { $oBasket = d3GetOxidDIC()->get('d3ox.webauthn.'.Session::class)->getBasket(); $oBasket->onUpdate(); } /** * @return bool */ public function isAdmin(): bool { return isAdmin(); } /** * @return string */ public function getUserId(): string { return $this->isAdmin() ? d3GetOxidDIC()->get('d3ox.webauthn.'.Session::class) ->getVariable(WebauthnConf::WEBAUTHN_ADMIN_SESSION_CURRENTUSER) : d3GetOxidDIC()->get('d3ox.webauthn.'.Session::class) ->getVariable(WebauthnConf::WEBAUTHN_SESSION_CURRENTUSER); } /** * @param User $user * @return void * @throws WebauthnGetException */ public function setFrontendSession(User $user): void { $session = d3GetOxidDIC()->get('d3ox.webauthn.'.Session::class); $session->setVariable(WebauthnConf::WEBAUTHN_SESSION_AUTH, $this->getCredential()); $session->setVariable(WebauthnConf::OXID_FRONTEND_AUTH, $user->getId()); } }