173 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
| <?php
 | |
| 
 | |
| namespace kernel\modules\secure\controllers;
 | |
| 
 | |
| use JetBrains\PhpStorm\NoReturn;
 | |
| use kernel\AdminController;
 | |
| use kernel\App;
 | |
| use kernel\Flash;
 | |
| use kernel\helpers\Debug;
 | |
| use kernel\Mailing;
 | |
| use kernel\modules\secure\models\forms\LoginEmailForm;
 | |
| use kernel\modules\secure\models\forms\LoginForm;
 | |
| use kernel\modules\secure\models\forms\RegisterForm;
 | |
| use kernel\modules\secure\services\SecureService;
 | |
| use kernel\modules\user\models\User;
 | |
| use kernel\modules\user\service\UserService;
 | |
| use kernel\Request;
 | |
| use PHPMailer\PHPMailer\Exception;
 | |
| use Random\RandomException;
 | |
| 
 | |
| class SecureController extends AdminController
 | |
| {
 | |
|     protected UserService $userService;
 | |
| 
 | |
|     protected function init(): void
 | |
|     {
 | |
|         parent::init();
 | |
|         $this->cgView->viewPath = KERNEL_MODULES_DIR. "/secure/views/";
 | |
|         $this->cgView->layout = "/login.php";
 | |
|         $this->userService = new UserService();
 | |
|     }
 | |
| 
 | |
|     public function actionLogin(): void
 | |
|     {
 | |
|         $this->cgView->render(match (App::$secure['web_auth_type']) {
 | |
|             "login_password" => "login.php",
 | |
|             "email_code" => "email_login.php",
 | |
|         });
 | |
| 
 | |
| //        $this->cgView->render('login.php');
 | |
|     }
 | |
| 
 | |
|     #[NoReturn] public function actionAuth(): void
 | |
|     {
 | |
|         $loginForm = new LoginForm();
 | |
|         $loginForm->load($_REQUEST);
 | |
| 
 | |
|         if(filter_var($loginForm->getItem("username"), FILTER_VALIDATE_EMAIL)) {
 | |
|             $field = "email";
 | |
|         }
 | |
|         else {
 | |
|             $field = "username";
 | |
|         }
 | |
| 
 | |
|         $user = $this->userService->getByField($field, $loginForm->getItem("username"));
 | |
|         if (!$user){
 | |
|             Flash::setMessage("error", "User not found.");
 | |
|             $this->redirect("/admin/login", code: 302);
 | |
|         }
 | |
| 
 | |
|         if (password_verify($loginForm->getItem("password"), $user->password_hash)) {
 | |
|             setcookie('user_id', $user->id, time()+60*60*24, '/', $_SERVER['SERVER_NAME'], false);
 | |
|             $this->redirect("/admin", code: 302);
 | |
|         } else {
 | |
|             Flash::setMessage("error", "Username or password incorrect.");
 | |
|             $this->redirect("/admin/login", code: 302);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @throws RandomException
 | |
|      * @throws Exception
 | |
|      */
 | |
|     public function actionEmailAuth(): void
 | |
|     {
 | |
|         $mailing = new Mailing();
 | |
| 
 | |
|         $loginForm = new LoginEmailForm();
 | |
|         $loginForm->load($_REQUEST);
 | |
| 
 | |
|         $email = $loginForm->getItem("email");
 | |
|         $user = $this->userService->getByField('email', $email);
 | |
| 
 | |
|         if (!$user){
 | |
|             $password = bin2hex(random_bytes(8));
 | |
| 
 | |
|             UserService::createUserByEmailAndPassword($email, $password);
 | |
|             $user = $this->userService->getByField('email', $email);
 | |
| 
 | |
|             SecureService::createSecretCode($user);
 | |
|             $secretCode = SecureService::getByField("user_id", $user->id);
 | |
| 
 | |
| 
 | |
|             $mailing->send_html("register_by_code.php", ['code' => $secretCode->code, 'password' => $password], [
 | |
|                 'address' => $email,
 | |
|                 'subject' => "Код регистрации",
 | |
|                 "from_name" => $_ENV['APP_NAME']
 | |
|             ]);
 | |
|         } else {
 | |
|             SecureService::updateSecretCode($user);
 | |
|             $secretCode = SecureService::getByField("user_id", $user->id);
 | |
|             $mailing->send_html("login_by_code.php", ['code' => $secretCode->code], [
 | |
|                 'address' => $email,
 | |
|                 'subject' => "Код авторизации",
 | |
|                 "from_name" => $_ENV['APP_NAME']
 | |
|             ]);
 | |
|         }
 | |
| 
 | |
|         setcookie('user_email', $email, time()+60*15, '/', $_SERVER['SERVER_NAME'], false);
 | |
|         $this->cgView->render("enter_code.php", ['email' => $email]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @throws Exception
 | |
|      */
 | |
|     public function actionCodeCheck(): void
 | |
|     {
 | |
|         $request = new Request();
 | |
| 
 | |
|         if (isset($_COOKIE['user_email'])) {
 | |
|             $user = User::where('email', $_COOKIE["user_email"])->first();
 | |
|             if (!$user) {
 | |
|                 throw new exception("User not found.");
 | |
|             }
 | |
|             $code = $request->post("code");
 | |
|             $secretCode = SecureService::getByField("user_id", $user->id);
 | |
|             if ($secretCode->code == $code && time() <= strtotime($secretCode->code_expires_at)) {
 | |
|                 setcookie('user_id', $user->id, time() + 60 * 60 * 24, '/', $_SERVER['SERVER_NAME'], false);
 | |
|                 $this->redirect("/admin", code: 302);
 | |
|             } else {
 | |
|                 Flash::setMessage("error", "Wrong code.");
 | |
|                 $this->cgView->render("enter_code.php", ['email' => $_COOKIE["user_email"]]);
 | |
|             }
 | |
| 
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[NoReturn] public function actionLogout(): void
 | |
|     {
 | |
|         unset($_COOKIE['user_id']);
 | |
|         setcookie('user_id', "", -1, '/', ".".$_SERVER['SERVER_NAME'], false);
 | |
|         setcookie('user_email', "", -1, '/', ".".$_SERVER['SERVER_NAME'], false);
 | |
|         $this->redirect("/", code: 302);
 | |
|     }
 | |
| 
 | |
|     public function actionRegister(): void
 | |
|     {
 | |
|         $this->cgView->render('register.php');
 | |
|     }
 | |
| 
 | |
|     public function actionRegistration(): void
 | |
|     {
 | |
|         $regForm = new RegisterForm();
 | |
|         $regForm->load($_REQUEST);
 | |
| 
 | |
|         if ($this->userService->getByField('username', $regForm->getItem("username"))) {
 | |
|             Flash::setMessage("error", "Username already exists.");
 | |
|             $this->redirect("/admin/register", code: 302);
 | |
|         }
 | |
| 
 | |
|         if ($this->userService->getByField('email', $regForm->getItem("email"))) {
 | |
|             Flash::setMessage("error", "Email already exists.");
 | |
|             $this->redirect("/admin/register", code: 302);
 | |
|         }
 | |
| 
 | |
|         $user = $this->userService->create($regForm);
 | |
|         if ($user){
 | |
|             setcookie('user_id', $user->id, time()+60*60*24, '/', $_SERVER['SERVER_NAME'], false);
 | |
|             $this->redirect("/admin", code: 302);
 | |
|         }
 | |
|     }
 | |
| 
 | |
| } |