Вы не можете установить свойства role токена.Как вы видели, TokenInterface
не включает в себя установщик для ролей.
И вы не можете setAuthenticated(false)
, потому что тогда вы очистите первую аутентификацию factor ,и возвращаемся к нулю.
Но я полагаю, что вы подходите к проблеме неправильно.
Оставьте токен как есть, и используйте слушатель для onAuthenticationSuccess
, чтобы проверить User::organization::getMfa()
чтобы решить, нужно ли вам:
- отправить пользователю дополнительный код аутентификации,
- сохранить где-то, что вы не завершили аутентификацию (например, в сеансе, в БД впользовательское свойство, отдельное хранилище для этого и т. д.).
(Лично я бы сделал требование MFA другой ролью, чтобы ее можно было хранить на токене, и вам не нужно было быпроверьте дополнительные вещи для этого).
Затем создайте подписчика на события для каждого запроса.
На этом слушателе вы добавите use Symfony\Component\Security\Core\Security
получить доступ к пользователю и посмотреть, если User::organization::getMfa() === true
;и все, что означает, чтобы проверить, если аутентификация не завершена (например, если вы сохранили в сеансе, или проверить пользовательский объект непосредственно и т. д.).
Если у вас нет токена, вы return
, это представление без аутентификации, и это не проблема слушателей.
Если пользователю не требуется MFA, вы просто return
;тут делать нечего.
Если пользователю требуется MFA, но он не находится в процессе аутентификации, вы return
.Это означает, что пользователь уже полностью прошел аутентификацию и может продолжить.
Если пользователю требуется MFA и он находится в середине процесса аутентификации, вы отображаете любые средства для его завершения.,Обычно это форма.
Если пользователю требуется MFA, и он находится в процессе аутентификации , и они отправляют вам средства для проверки этой аутентификации (например,POST-запрос с соответствующими полями), вы проверяете, что аутентификация в порядке.
Проверка правильна?Вы очищаете бит «аутентификация в процессе» и возвращаетесь.Дальнейшие запросы не будут перехвачены.
Проверка плоха?Вы либо позволяете им проверять снова, повторно отправлять проверочные коды, полностью выходить из системы, чтобы они снова начали ... Что именно будет очень зависеть от ваших требований.
ПолныйРеализация кода будет зависеть от вас, поскольку невозможно охватить все возможные сценарии.
Но для начала ваш подписчик может быть чем-то вроде этого (обратите внимание, что есть также пара нереализованных методов).
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpKernel\Event\RequestEvent;
use Symfony\Component\HttpKernel\HttpKernelInterface;
use Symfony\Component\HttpKernel\KernelEvents;
use Symfony\Component\Security\Core\Security;
class MfaSubscriber implements EventSubscriberInterface
{
private $security;
public function __construct(Security $security)
{
$this->security = $security;
}
public function onRequest(RequestEvent $event)
{
if ($event->getRequestType() === HttpKernelInterface::SUB_REQUEST) {
return;
}
$request = $event->getRequest();
if ( null == $token = $this->security->getToken()) {
return;
}
if ( ! in_array('ROLE_NEEDS_MFA', $token->getRoleNames(), true)) {
return;
}
if ( ! $request->hasPreviousSession() || ! $request->getSession()->get('AUTHENTICATION_IN_PROCESS')) {
return;
}
if (
$request->getMethod() === Request::METHOD_POST
&& $request->get('verification_code')
&& $this->verifyCode($request->get('verification_code')) // verify code needs to be implemented
) {
$request->getSession()->set('AUTHENTICATION_IN_PROCESS', null);
return;
}
$event->setResponse($this->displayMfaVerificationMethod($request, $token));
}
public static function getSubscribedEvents(): array
{
return [
KernelEvents::REQUEST => 'onRequest',
];
}
}