src/Controller/SecurityController.php line 130

Open in your IDE?
  1. <?php
  2. namespace App\Controller;
  3. use Monolog\Logger;
  4. use Psr\Log\LoggerInterface;
  5. use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
  6. use Symfony\Component\HttpFoundation\RedirectResponse;
  7. use Symfony\Component\HttpFoundation\Response;
  8. use Symfony\Component\Routing\Annotation\Route;
  9. use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
  10. use Symfony\Component\Security\Http\Authentication\AuthenticationUtils;
  11. use Symfony\Component\HttpFoundation\Request;
  12. use App\Entity\User;
  13. use Intis\SDK\Exception\MessageSendingResultException;
  14. use App\Service\Sender;
  15. class SecurityController extends ParentController {
  16. protected $error = '';
  17. protected $logger = null;
  18. protected $ip = '';
  19. protected $user = null;
  20. /**
  21. * @Route("/login", name="app_login")
  22. */
  23. public function login(AuthenticationUtils $authenticationUtils, Request $request, TokenStorageInterface $tokenStorage, LoggerInterface $dbLogger): Response {
  24. $this->logger = $dbLogger;
  25. $token = $tokenStorage->getToken();
  26. $this->user = $this->getUser();
  27. $this->ip = $request->getClientIp();
  28. if ($token && in_array('ROLE_PREVIOUS_ADMIN', $token->getRoleNames()) && $this->user) {
  29. return $this->redirectToRoute('admin_dashboard');
  30. }
  31. if ($this->user) {
  32. $this->user->setLastLoginAt(date_create());
  33. $entityManager = $this->getDoctrine()->getManager();
  34. $entityManager->persist($this->user);
  35. $entityManager->flush();
  36. $authData = $this->valide2wayAuth($request);
  37. if ($authData === true) {
  38. $this->logger->info('valide2wayAuth', ['User' => $this->user->getEmail(), 'IP' => $this->ip, 'Success' => true]);
  39. return $this->redirectToRoute('admin_dashboard');
  40. }
  41. $this->logger->info('valide2wayAuth', ['User' => $this->user->getEmail(), 'IP' => $this->ip, 'Success' => false]);
  42. }
  43. // get the login error if there is one
  44. $error = $authenticationUtils->getLastAuthenticationError();
  45. // last username entered by the user
  46. $email = $authenticationUtils->getLastUsername();
  47. if (!empty($error)) {
  48. $this->logger->error('valide2wayAuth', ['Message' => $error->getMessage(), 'IP' => $this->ip]);
  49. }
  50. $data = [
  51. 'last_username' => $email,
  52. 'error' => (!empty($this->error) ? $this->error : $error),
  53. 'template' => 'login',
  54. 'auth' => ($authData ?? []),
  55. 'systemName' => $this->systemName,
  56. ];
  57. return $this->render('security/login.html.twig', $data);
  58. }
  59. protected function valide2wayAuth($request) {
  60. $authService = $this->getConfigValue('2wayAuthType');
  61. $authMessage = $this->getConfigValue('2wayAuthMessage');
  62. if (empty($authService) || (!empty($this->user->getAllowedIps()) && strpos($this->user->getAllowedIps(), $this->ip) !== false)) {
  63. return true;
  64. }
  65. $authCodeReceived = $request->get('authCode');
  66. $session = $this->get('session');
  67. $authCode = $session->get($authService.'Code');
  68. $data = [
  69. 'message' => $authMessage,
  70. 'code' => true,
  71. 'type' => $authService,
  72. ];
  73. if (!empty($authCode) && !empty($authCodeReceived) && $authCode != $authCodeReceived) {
  74. $this->error = 'Code set is incorrect';
  75. return $data;
  76. } elseif (!empty($authCode) && $authCode == $authCodeReceived) {
  77. $session->remove($authService.'Code');
  78. return true;
  79. }
  80. if (empty($authCode)) {
  81. $authData = explode(':', $authService);
  82. $authType = $authData[0];
  83. $provider = $authData[1];
  84. switch ($authType) {
  85. case 'sms' :
  86. $authCode = mt_rand(10000, 99999);
  87. /** @var Sender $sender */
  88. $sender = $this->get('sender');
  89. $sendData = [
  90. 'login' => $this->getConfigValue('smsApiLogin'),
  91. 'apiKey' => $this->getConfigValue('smsApiKey'),
  92. 'apiHost' => $this->getConfigValue('smsApiHost'),
  93. 'phone' => trim($this->getUser()->getAdminPhone(), '+'),
  94. 'sender' => $this->getConfigValue('smsSender'),
  95. 'provider' => $provider,
  96. 'text' => 'Login confirm code: '.$authCode,
  97. ];
  98. $session->set($authService.'Code', $authCode);
  99. try {
  100. $sender->sendSms($sendData);
  101. } catch (MessageSendingResultException $e) {
  102. $this->error = $e->getMessage();
  103. $this->logger->error('sendSms', ['Message' => $this->error, 'IP' => $this->ip]);
  104. return true;
  105. }
  106. break;
  107. }
  108. }
  109. return $data;
  110. }
  111. /**
  112. * @Route("/logout", name="app_logout")
  113. */
  114. public function logout() {
  115. throw new \Exception('This method can be blank - it will be intercepted by the logout key on your firewall');
  116. }
  117. public function forgot(Request $request) {
  118. $this->initialize($request);
  119. $success = '';
  120. $error = '';
  121. $email = $request->get('restore_email');
  122. if (!empty($email)) {
  123. $entityManager = $this->getDoctrine()->getManager();
  124. $repo = $this->getDoctrine()->getRepository(User::class);
  125. $user = $repo->findOneBy(['email' => $email, 'enabled' => 1]);
  126. if (empty($user)) {
  127. $error = 'No such email on the platform';
  128. } else {
  129. $helper = $this->get('helper');
  130. $sender = $this->get('sender');
  131. $user->setPasswRestoreKey($helper->getRandomString(50));
  132. $user->setPasswRestoreKeyValidTil(date_create('+ 3 days'));
  133. $entityManager->persist($user);
  134. $entityManager->flush();
  135. $data = [
  136. 'from' => $this->mailSender,
  137. 'to' => $user->getEmail(),
  138. 'toName' => $user->getEmail(),
  139. 'subject' => 'corpHedge platform password restore',
  140. 'content' => '
  141. Hello,<br>
  142. <br>
  143. CorpHedge Platform has received a request to restore your password from IP: '.$request->getClientIp().'. <br>
  144. <a href="'.$request->getSchemeAndHttpHost().'/restore?key='.$user->getPasswRestoreKey().'"> Click here to set your new password</a><br>
  145. <br>
  146. If you did not initiate this request, please disregard this email and notify CorpHedge administration.<br>
  147. <br>
  148. Best regards,<br>
  149. <br>
  150. CorpHedge platform team<br>
  151. <img src="'.$request->getSchemeAndHttpHost().'/assets/img/logo_email.png">
  152. ',
  153. ];
  154. if ($sender->sendEmail($data)) {
  155. $success = 'We have sent an email to '.$email.'. Please check your inbox and follow the instructions';
  156. } else {
  157. $error = 'Some error ';
  158. }
  159. }
  160. }
  161. $data = [
  162. 'last_username' => $email,
  163. 'success' => $success,
  164. 'error' => $error,
  165. 'template' => 'forgot',
  166. ];
  167. return $this->render('security/login.html.twig', $data);
  168. }
  169. public function restore(Request $request) {
  170. $this->initialize($request);
  171. $success = '';
  172. $error = '';
  173. $key = $request->get('key');
  174. $passw1 = $request->get('password1');
  175. $passw2 = $request->get('password2');
  176. $email = '';
  177. if (!empty($key)) {
  178. /** @var User $user */
  179. $user = $this->getDoctrine()->getRepository(User::class)->findOneBy(['passwRestoreKey' => $key]);
  180. if (empty($user)) {
  181. $error = 'This restore key is invalid or expired';
  182. } else {
  183. $email = $user->getEmail();
  184. }
  185. if (!empty($passw1)) {
  186. if (empty($user)) {
  187. $error = 'User not found';
  188. } elseif ($user->getPasswRestoreKeyValidTil() < date_create()) {
  189. $error = 'Password restore period is expired. Make a new request';
  190. } elseif (!empty($passw1) && $passw1 !== $passw2) {
  191. $error = 'The repeated password is not the same';
  192. } elseif (!$this->passwordSecure($passw1, $user)) {
  193. $error = 'Password is not secure. Please ensure that it:
  194. <li> Contains at least 8 characters </li>
  195. <li>Requires uppercase and lowercase letters</li>
  196. <li>Contains at least one number</li>
  197. <li>Cannot match previous passwords.</li> ';
  198. } elseif (!empty($user->getPasswChanged()) && date_create($user->getPasswChanged()->format('Y-m-d H:i:s').' +1 hour') > date_create()) {
  199. if ((int)$user->getPasswChangeCount() > (int)$this->getConfigValue('maxPasswordChangesInHour')) {
  200. $error = 'Too much changes in one hour. Try it later';
  201. }
  202. }
  203. if (empty($error)) {
  204. $entityManager = $this->getDoctrine()->getManager();
  205. $userController = new UserController();
  206. $userController->encodePassword($user, $passw1);
  207. $user->setPassword($userController->encodePassword($user, $passw1));
  208. $user->setPasswRestoreKey(null);
  209. $user->setPasswRestoreKeyValidTil(null);
  210. if (!empty($user->getPasswChanged()) && date_create($user->getPasswChanged()->getTimestamp().' +1 hour') > date_create()) {
  211. $user->setPasswChangeCount($user->getPasswChangeCount() + 1);
  212. }
  213. $user->setPasswChanged(date_create());
  214. $user->setLastLoginAt(date_create());
  215. $entityManager->persist($user);
  216. $entityManager->flush();
  217. $sender = $this->get('sender');
  218. $data = [
  219. 'from' => $this->mailSender,
  220. 'to' => $user->getEmail(),
  221. 'toName' => $user->getUsername(),
  222. 'subject' => 'corpHedge platform password restore',
  223. 'content' => '
  224. <a href="https://www.corphedge.com/" target="_blank"> <img
  225. moz-do-not-send="false"
  226. src=""
  227. alt="" width="200"></a><br><br>
  228. Dear User,<br>
  229. <br>
  230. We would like to inform you that the password for the account '.$user->getEmail().' has been successfully changed.
  231. <br>
  232. Please log in to the platform using your new password.
  233. <br>
  234. <br>
  235. Best regards,<br>
  236. <br>
  237. CorpHedge support team
  238. ',
  239. ];
  240. $sender->sendEmail($data);
  241. $success = 'Password was changed successfully.';
  242. }
  243. }
  244. }
  245. $data = [
  246. 'last_username' => $email,
  247. 'success' => $success,
  248. 'error' => $error,
  249. 'passw1' => $passw1,
  250. 'passw2' => $passw2,
  251. 'template' => 'restore',
  252. ];
  253. return $this->render('security/login.html.twig', $data);
  254. }
  255. protected function passwordSecure($password, $user) {
  256. $userController = new UserController();
  257. $passwordEncoded = $userController->encodePassword($user, $password);
  258. if (!empty(trim($password, 'a..z'))
  259. && !empty(trim($password, 'A..Z'))
  260. && strlen($password) >= 8
  261. && !empty(trim($password, '1..9'))
  262. && $passwordEncoded !== $user->getPassword()) {
  263. return true;
  264. }
  265. return false;
  266. }
  267. }