<?php
namespace App\Controller;
use Monolog\Logger;
use Psr\Log\LoggerInterface;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\HttpFoundation\RedirectResponse;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\Routing\Annotation\Route;
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
use Symfony\Component\Security\Http\Authentication\AuthenticationUtils;
use Symfony\Component\HttpFoundation\Request;
use App\Entity\User;
use Intis\SDK\Exception\MessageSendingResultException;
use App\Service\Sender;
class SecurityController extends ParentController {
protected $error = '';
protected $logger = null;
protected $ip = '';
protected $user = null;
/**
* @Route("/login", name="app_login")
*/
public function login(AuthenticationUtils $authenticationUtils, Request $request, TokenStorageInterface $tokenStorage, LoggerInterface $dbLogger): Response {
$this->logger = $dbLogger;
$token = $tokenStorage->getToken();
$this->user = $this->getUser();
$this->ip = $request->getClientIp();
if ($token && in_array('ROLE_PREVIOUS_ADMIN', $token->getRoleNames()) && $this->user) {
return $this->redirectToRoute('admin_dashboard');
}
if ($this->user) {
$this->user->setLastLoginAt(date_create());
$entityManager = $this->getDoctrine()->getManager();
$entityManager->persist($this->user);
$entityManager->flush();
$authData = $this->valide2wayAuth($request);
if ($authData === true) {
$this->logger->info('valide2wayAuth', ['User' => $this->user->getEmail(), 'IP' => $this->ip, 'Success' => true]);
return $this->redirectToRoute('admin_dashboard');
}
$this->logger->info('valide2wayAuth', ['User' => $this->user->getEmail(), 'IP' => $this->ip, 'Success' => false]);
}
// get the login error if there is one
$error = $authenticationUtils->getLastAuthenticationError();
// last username entered by the user
$email = $authenticationUtils->getLastUsername();
if (!empty($error)) {
$this->logger->error('valide2wayAuth', ['Message' => $error->getMessage(), 'IP' => $this->ip]);
}
$data = [
'last_username' => $email,
'error' => (!empty($this->error) ? $this->error : $error),
'template' => 'login',
'auth' => ($authData ?? []),
'systemName' => $this->systemName,
];
return $this->render('security/login.html.twig', $data);
}
protected function valide2wayAuth($request) {
$authService = $this->getConfigValue('2wayAuthType');
$authMessage = $this->getConfigValue('2wayAuthMessage');
if (empty($authService) || (!empty($this->user->getAllowedIps()) && strpos($this->user->getAllowedIps(), $this->ip) !== false)) {
return true;
}
$authCodeReceived = $request->get('authCode');
$session = $this->get('session');
$authCode = $session->get($authService.'Code');
$data = [
'message' => $authMessage,
'code' => true,
'type' => $authService,
];
if (!empty($authCode) && !empty($authCodeReceived) && $authCode != $authCodeReceived) {
$this->error = 'Code set is incorrect';
return $data;
} elseif (!empty($authCode) && $authCode == $authCodeReceived) {
$session->remove($authService.'Code');
return true;
}
if (empty($authCode)) {
$authData = explode(':', $authService);
$authType = $authData[0];
$provider = $authData[1];
switch ($authType) {
case 'sms' :
$authCode = mt_rand(10000, 99999);
/** @var Sender $sender */
$sender = $this->get('sender');
$sendData = [
'login' => $this->getConfigValue('smsApiLogin'),
'apiKey' => $this->getConfigValue('smsApiKey'),
'apiHost' => $this->getConfigValue('smsApiHost'),
'phone' => trim($this->getUser()->getAdminPhone(), '+'),
'sender' => $this->getConfigValue('smsSender'),
'provider' => $provider,
'text' => 'Login confirm code: '.$authCode,
];
$session->set($authService.'Code', $authCode);
try {
$sender->sendSms($sendData);
} catch (MessageSendingResultException $e) {
$this->error = $e->getMessage();
$this->logger->error('sendSms', ['Message' => $this->error, 'IP' => $this->ip]);
return true;
}
break;
}
}
return $data;
}
/**
* @Route("/logout", name="app_logout")
*/
public function logout() {
throw new \Exception('This method can be blank - it will be intercepted by the logout key on your firewall');
}
public function forgot(Request $request) {
$this->initialize($request);
$success = '';
$error = '';
$email = $request->get('restore_email');
if (!empty($email)) {
$entityManager = $this->getDoctrine()->getManager();
$repo = $this->getDoctrine()->getRepository(User::class);
$user = $repo->findOneBy(['email' => $email, 'enabled' => 1]);
if (empty($user)) {
$error = 'No such email on the platform';
} else {
$helper = $this->get('helper');
$sender = $this->get('sender');
$user->setPasswRestoreKey($helper->getRandomString(50));
$user->setPasswRestoreKeyValidTil(date_create('+ 3 days'));
$entityManager->persist($user);
$entityManager->flush();
$data = [
'from' => $this->mailSender,
'to' => $user->getEmail(),
'toName' => $user->getEmail(),
'subject' => 'corpHedge platform password restore',
'content' => '
Hello,<br>
<br>
CorpHedge Platform has received a request to restore your password from IP: '.$request->getClientIp().'. <br>
<a href="'.$request->getSchemeAndHttpHost().'/restore?key='.$user->getPasswRestoreKey().'"> Click here to set your new password</a><br>
<br>
If you did not initiate this request, please disregard this email and notify CorpHedge administration.<br>
<br>
Best regards,<br>
<br>
CorpHedge platform team<br>
<img src="'.$request->getSchemeAndHttpHost().'/assets/img/logo_email.png">
',
];
if ($sender->sendEmail($data)) {
$success = 'We have sent an email to '.$email.'. Please check your inbox and follow the instructions';
} else {
$error = 'Some error ';
}
}
}
$data = [
'last_username' => $email,
'success' => $success,
'error' => $error,
'template' => 'forgot',
];
return $this->render('security/login.html.twig', $data);
}
public function restore(Request $request) {
$this->initialize($request);
$success = '';
$error = '';
$key = $request->get('key');
$passw1 = $request->get('password1');
$passw2 = $request->get('password2');
$email = '';
if (!empty($key)) {
/** @var User $user */
$user = $this->getDoctrine()->getRepository(User::class)->findOneBy(['passwRestoreKey' => $key]);
if (empty($user)) {
$error = 'This restore key is invalid or expired';
} else {
$email = $user->getEmail();
}
if (!empty($passw1)) {
if (empty($user)) {
$error = 'User not found';
} elseif ($user->getPasswRestoreKeyValidTil() < date_create()) {
$error = 'Password restore period is expired. Make a new request';
} elseif (!empty($passw1) && $passw1 !== $passw2) {
$error = 'The repeated password is not the same';
} elseif (!$this->passwordSecure($passw1, $user)) {
$error = 'Password is not secure. Please ensure that it:
<li> Contains at least 8 characters </li>
<li>Requires uppercase and lowercase letters</li>
<li>Contains at least one number</li>
<li>Cannot match previous passwords.</li> ';
} elseif (!empty($user->getPasswChanged()) && date_create($user->getPasswChanged()->format('Y-m-d H:i:s').' +1 hour') > date_create()) {
if ((int)$user->getPasswChangeCount() > (int)$this->getConfigValue('maxPasswordChangesInHour')) {
$error = 'Too much changes in one hour. Try it later';
}
}
if (empty($error)) {
$entityManager = $this->getDoctrine()->getManager();
$userController = new UserController();
$userController->encodePassword($user, $passw1);
$user->setPassword($userController->encodePassword($user, $passw1));
$user->setPasswRestoreKey(null);
$user->setPasswRestoreKeyValidTil(null);
if (!empty($user->getPasswChanged()) && date_create($user->getPasswChanged()->getTimestamp().' +1 hour') > date_create()) {
$user->setPasswChangeCount($user->getPasswChangeCount() + 1);
}
$user->setPasswChanged(date_create());
$user->setLastLoginAt(date_create());
$entityManager->persist($user);
$entityManager->flush();
$sender = $this->get('sender');
$data = [
'from' => $this->mailSender,
'to' => $user->getEmail(),
'toName' => $user->getUsername(),
'subject' => 'corpHedge platform password restore',
'content' => '
<a href="https://www.corphedge.com/" target="_blank"> <img
moz-do-not-send="false"
src=""
alt="" width="200"></a><br><br>
Dear User,<br>
<br>
We would like to inform you that the password for the account '.$user->getEmail().' has been successfully changed.
<br>
Please log in to the platform using your new password.
<br>
<br>
Best regards,<br>
<br>
CorpHedge support team
',
];
$sender->sendEmail($data);
$success = 'Password was changed successfully.';
}
}
}
$data = [
'last_username' => $email,
'success' => $success,
'error' => $error,
'passw1' => $passw1,
'passw2' => $passw2,
'template' => 'restore',
];
return $this->render('security/login.html.twig', $data);
}
protected function passwordSecure($password, $user) {
$userController = new UserController();
$passwordEncoded = $userController->encodePassword($user, $password);
if (!empty(trim($password, 'a..z'))
&& !empty(trim($password, 'A..Z'))
&& strlen($password) >= 8
&& !empty(trim($password, '1..9'))
&& $passwordEncoded !== $user->getPassword()) {
return true;
}
return false;
}
}