Implement user registration and entity
This commit is contained in:
		| @@ -30,6 +30,7 @@ | |||||||
|         "symfony/proxy-manager-bridge": "6.0.*", |         "symfony/proxy-manager-bridge": "6.0.*", | ||||||
|         "symfony/runtime": "6.0.*", |         "symfony/runtime": "6.0.*", | ||||||
|         "symfony/security-bundle": "6.0.*", |         "symfony/security-bundle": "6.0.*", | ||||||
|  |         "symfony/security-csrf": "6.0.*", | ||||||
|         "symfony/serializer": "6.0.*", |         "symfony/serializer": "6.0.*", | ||||||
|         "symfony/string": "6.0.*", |         "symfony/string": "6.0.*", | ||||||
|         "symfony/translation": "6.0.*", |         "symfony/translation": "6.0.*", | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							| @@ -4,7 +4,7 @@ | |||||||
|         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", |         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", | ||||||
|         "This file is @generated automatically" |         "This file is @generated automatically" | ||||||
|     ], |     ], | ||||||
|     "content-hash": "dabb783b3f2096007e6fbd0ea80f7505", |     "content-hash": "d4f8936e80f8eda092bcbd5412193f15", | ||||||
|     "packages": [ |     "packages": [ | ||||||
|         { |         { | ||||||
|             "name": "composer/package-versions-deprecated", |             "name": "composer/package-versions-deprecated", | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| # see https://symfony.com/doc/current/reference/configuration/framework.html | # see https://symfony.com/doc/current/reference/configuration/framework.html | ||||||
| framework: | framework: | ||||||
|     secret: '%env(APP_SECRET)%' |     secret: '%env(APP_SECRET)%' | ||||||
|     #csrf_protection: true |     csrf_protection: ~ | ||||||
|     http_method_override: false |     http_method_override: false | ||||||
|  |  | ||||||
|     # Enables session support. Note that the session will ONLY be started if you read or write from it. |     # Enables session support. Note that the session will ONLY be started if you read or write from it. | ||||||
|   | |||||||
| @@ -3,16 +3,23 @@ security: | |||||||
|     # https://symfony.com/doc/current/security.html#registering-the-user-hashing-passwords |     # https://symfony.com/doc/current/security.html#registering-the-user-hashing-passwords | ||||||
|     password_hashers: |     password_hashers: | ||||||
|         Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface: 'auto' |         Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface: 'auto' | ||||||
|  |         App\Entity\User: | ||||||
|  |             algorithm: auto | ||||||
|  |  | ||||||
|     # https://symfony.com/doc/current/security.html#loading-the-user-the-user-provider |     # https://symfony.com/doc/current/security.html#loading-the-user-the-user-provider | ||||||
|     providers: |     providers: | ||||||
|         users_in_memory: { memory: null } |         # used to reload user from session & other features (e.g. switch_user) | ||||||
|  |         app_user_provider: | ||||||
|  |             entity: | ||||||
|  |                 class: App\Entity\User | ||||||
|  |                 property: username | ||||||
|     firewalls: |     firewalls: | ||||||
|         dev: |         dev: | ||||||
|             pattern: ^/(_(profiler|wdt)|css|images|js)/ |             pattern: ^/(_(profiler|wdt)|css|images|js)/ | ||||||
|             security: false |             security: false | ||||||
|         main: |         main: | ||||||
|             lazy: true |             lazy: true | ||||||
|             provider: users_in_memory |             provider: app_user_provider | ||||||
|  |  | ||||||
|             # activate different ways to authenticate |             # activate different ways to authenticate | ||||||
|             # https://symfony.com/doc/current/security.html#the-firewall |             # https://symfony.com/doc/current/security.html#the-firewall | ||||||
| @@ -23,7 +30,7 @@ security: | |||||||
|     # Easy way to control access for large sections of your site |     # Easy way to control access for large sections of your site | ||||||
|     # Note: Only the *first* access control that matches will be used |     # Note: Only the *first* access control that matches will be used | ||||||
|     access_control: |     access_control: | ||||||
|         # - { path: ^/admin, roles: ROLE_ADMIN } | #        - { path: ^/admin, roles: ROLE_ADMIN } | ||||||
|         # - { path: ^/profile, roles: ROLE_USER } |         # - { path: ^/profile, roles: ROLE_USER } | ||||||
|  |  | ||||||
| when@test: | when@test: | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| twig: | twig: | ||||||
|     default_path: '%kernel.project_dir%/templates' |     default_path: '%kernel.project_dir%/templates' | ||||||
|  |     form_themes: ['bootstrap_5_horizontal_layout.html.twig'] | ||||||
|  |  | ||||||
| when@test: | when@test: | ||||||
|     twig: |     twig: | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
| namespace App\Controller\Admin; | namespace App\Controller\Admin; | ||||||
|  |  | ||||||
| use App\Entity\Product; | use App\Entity\Product; | ||||||
|  | use App\Entity\User; | ||||||
| use EasyCorp\Bundle\EasyAdminBundle\Config\Dashboard; | use EasyCorp\Bundle\EasyAdminBundle\Config\Dashboard; | ||||||
| use EasyCorp\Bundle\EasyAdminBundle\Config\MenuItem; | use EasyCorp\Bundle\EasyAdminBundle\Config\MenuItem; | ||||||
| use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractDashboardController; | use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractDashboardController; | ||||||
| @@ -35,5 +36,8 @@ class DashboardController extends AbstractDashboardController | |||||||
|         yield MenuItem::section('Products', 'fas fa-folder-open'); |         yield MenuItem::section('Products', 'fas fa-folder-open'); | ||||||
|         yield MenuItem::linkToDashboard('Dashboard', 'fa fa-home'); |         yield MenuItem::linkToDashboard('Dashboard', 'fa fa-home'); | ||||||
|         yield MenuItem::linkToCrud('Products', 'fas fa-list', Product::class); |         yield MenuItem::linkToCrud('Products', 'fas fa-list', Product::class); | ||||||
|  |  | ||||||
|  |         yield MenuItem::section('Administration', 'fas fa-folder-open'); | ||||||
|  |         yield MenuItem::linkToCrud('User', 'fas fa-list', User::class); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								src/Controller/Admin/UserCrudController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/Controller/Admin/UserCrudController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Controller\Admin; | ||||||
|  |  | ||||||
|  | use App\Entity\User; | ||||||
|  | use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController; | ||||||
|  |  | ||||||
|  | class UserCrudController extends AbstractCrudController | ||||||
|  | { | ||||||
|  |     public static function getEntityFqcn(): string | ||||||
|  |     { | ||||||
|  |         return User::class; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |     public function configureFields(string $pageName): iterable | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             IdField::new('id'), | ||||||
|  |             TextField::new('title'), | ||||||
|  |             TextEditorField::new('description'), | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  |     */ | ||||||
|  | } | ||||||
							
								
								
									
										43
									
								
								src/Controller/RegistrationController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								src/Controller/RegistrationController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Controller; | ||||||
|  |  | ||||||
|  | use App\Entity\User; | ||||||
|  | use App\Form\RegistrationFormType; | ||||||
|  | use Doctrine\ORM\EntityManagerInterface; | ||||||
|  | use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; | ||||||
|  | use Symfony\Component\HttpFoundation\Request; | ||||||
|  | use Symfony\Component\HttpFoundation\Response; | ||||||
|  | use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface; | ||||||
|  | use Symfony\Component\Routing\Annotation\Route; | ||||||
|  |  | ||||||
|  | class RegistrationController extends AbstractController | ||||||
|  | { | ||||||
|  |     #[Route('/register', name: 'app_register')] | ||||||
|  |     public function register(Request $request, UserPasswordHasherInterface $userPasswordHasher, EntityManagerInterface $entityManager): Response | ||||||
|  |     { | ||||||
|  |         $user = new User(); | ||||||
|  |         $form = $this->createForm(RegistrationFormType::class, $user); | ||||||
|  |         $form->handleRequest($request); | ||||||
|  |  | ||||||
|  |         if ($form->isSubmitted() && $form->isValid()) { | ||||||
|  |             // encode the plain password | ||||||
|  |             $user->setPassword( | ||||||
|  |             $userPasswordHasher->hashPassword( | ||||||
|  |                     $user, | ||||||
|  |                     $form->get('plainPassword')->getData() | ||||||
|  |                 ) | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             $entityManager->persist($user); | ||||||
|  |             $entityManager->flush(); | ||||||
|  |             // do anything else you need here, like send an email | ||||||
|  |  | ||||||
|  |             return $this->redirectToRoute('app_test_test1'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return $this->render('registration/register.html.twig', [ | ||||||
|  |             'registrationForm' => $form->createView(), | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										100
									
								
								src/Entity/User.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								src/Entity/User.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Entity; | ||||||
|  |  | ||||||
|  | use App\Repository\UserRepository; | ||||||
|  | use Doctrine\ORM\Mapping as ORM; | ||||||
|  | use Symfony\Bridge\Doctrine\Validator\Constraints\UniqueEntity; | ||||||
|  | use Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface; | ||||||
|  | use Symfony\Component\Security\Core\User\UserInterface; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @UniqueEntity(fields={"username"}, message="There is already an account with this username") | ||||||
|  |  */ | ||||||
|  | #[ORM\Entity(repositoryClass: UserRepository::class)] | ||||||
|  | class User implements UserInterface, PasswordAuthenticatedUserInterface | ||||||
|  | { | ||||||
|  |     #[ORM\Id] | ||||||
|  |     #[ORM\GeneratedValue] | ||||||
|  |     #[ORM\Column(type: 'integer')] | ||||||
|  |     private $id; | ||||||
|  |  | ||||||
|  |     #[ORM\Column(type: 'string', length: 180, unique: true)] | ||||||
|  |     private $username; | ||||||
|  |  | ||||||
|  |     #[ORM\Column(type: 'json')] | ||||||
|  |     private $roles = []; | ||||||
|  |  | ||||||
|  |     #[ORM\Column(type: 'string')] | ||||||
|  |     private $password; | ||||||
|  |  | ||||||
|  |     public function getId(): ?int | ||||||
|  |     { | ||||||
|  |         return $this->id; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getUsername(): ?string | ||||||
|  |     { | ||||||
|  |         return $this->username; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function setUsername(string $username): self | ||||||
|  |     { | ||||||
|  |         $this->username = $username; | ||||||
|  |  | ||||||
|  |         return $this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * A visual identifier that represents this user. | ||||||
|  |      * | ||||||
|  |      * @see UserInterface | ||||||
|  |      */ | ||||||
|  |     public function getUserIdentifier(): string | ||||||
|  |     { | ||||||
|  |         return (string) $this->username; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @see UserInterface | ||||||
|  |      */ | ||||||
|  |     public function getRoles(): array | ||||||
|  |     { | ||||||
|  |         $roles = $this->roles; | ||||||
|  |         // guarantee every user at least has ROLE_USER | ||||||
|  |         $roles[] = 'ROLE_USER'; | ||||||
|  |  | ||||||
|  |         return array_unique($roles); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function setRoles(array $roles): self | ||||||
|  |     { | ||||||
|  |         $this->roles = $roles; | ||||||
|  |  | ||||||
|  |         return $this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @see PasswordAuthenticatedUserInterface | ||||||
|  |      */ | ||||||
|  |     public function getPassword(): string | ||||||
|  |     { | ||||||
|  |         return $this->password; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function setPassword(string $password): self | ||||||
|  |     { | ||||||
|  |         $this->password = $password; | ||||||
|  |  | ||||||
|  |         return $this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @see UserInterface | ||||||
|  |      */ | ||||||
|  |     public function eraseCredentials() | ||||||
|  |     { | ||||||
|  |         // If you store any temporary, sensitive data on the user, clear it here | ||||||
|  |         // $this->plainPassword = null; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								src/Form/RegistrationFormType.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								src/Form/RegistrationFormType.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Form; | ||||||
|  |  | ||||||
|  | use App\Entity\User; | ||||||
|  | use Symfony\Component\Form\AbstractType; | ||||||
|  | use Symfony\Component\Form\Extension\Core\Type\CheckboxType; | ||||||
|  | use Symfony\Component\Form\Extension\Core\Type\PasswordType; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  | use Symfony\Component\OptionsResolver\OptionsResolver; | ||||||
|  | use Symfony\Component\Validator\Constraints\IsTrue; | ||||||
|  | use Symfony\Component\Validator\Constraints\Length; | ||||||
|  | use Symfony\Component\Validator\Constraints\NotBlank; | ||||||
|  |  | ||||||
|  | class RegistrationFormType extends AbstractType | ||||||
|  | { | ||||||
|  |     public function buildForm(FormBuilderInterface $builder, array $options): void | ||||||
|  |     { | ||||||
|  |         $builder | ||||||
|  |             ->add('username') | ||||||
|  |             ->add('agreeTerms', CheckboxType::class, [ | ||||||
|  |                 'mapped' => false, | ||||||
|  |                 'constraints' => [ | ||||||
|  |                     new IsTrue([ | ||||||
|  |                         'message' => 'You should agree to our terms.', | ||||||
|  |                     ]), | ||||||
|  |                 ], | ||||||
|  |             ]) | ||||||
|  |             ->add('plainPassword', PasswordType::class, [ | ||||||
|  |                 // instead of being set onto the object directly, | ||||||
|  |                 // this is read and encoded in the controller | ||||||
|  |                 'mapped' => false, | ||||||
|  |                 'attr' => ['autocomplete' => 'new-password'], | ||||||
|  |                 'constraints' => [ | ||||||
|  |                     new NotBlank([ | ||||||
|  |                         'message' => 'Please enter a password', | ||||||
|  |                     ]), | ||||||
|  |                     new Length([ | ||||||
|  |                         'min' => 3, | ||||||
|  |                         'minMessage' => 'Your password should be at least {{ limit }} characters', | ||||||
|  |                         // max length allowed by Symfony for security reasons | ||||||
|  |                         'max' => 4096, | ||||||
|  |                     ]), | ||||||
|  |                 ], | ||||||
|  |             ]) | ||||||
|  |         ; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function configureOptions(OptionsResolver $resolver): void | ||||||
|  |     { | ||||||
|  |         $resolver->setDefaults([ | ||||||
|  |             'data_class' => User::class, | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										67
									
								
								src/Repository/UserRepository.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								src/Repository/UserRepository.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Repository; | ||||||
|  |  | ||||||
|  | use App\Entity\User; | ||||||
|  | use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository; | ||||||
|  | use Doctrine\Persistence\ManagerRegistry; | ||||||
|  | use Symfony\Component\Security\Core\Exception\UnsupportedUserException; | ||||||
|  | use Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface; | ||||||
|  | use Symfony\Component\Security\Core\User\PasswordUpgraderInterface; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @method User|null find($id, $lockMode = null, $lockVersion = null) | ||||||
|  |  * @method User|null findOneBy(array $criteria, array $orderBy = null) | ||||||
|  |  * @method User[]    findAll() | ||||||
|  |  * @method User[]    findBy(array $criteria, array $orderBy = null, $limit = null, $offset = null) | ||||||
|  |  */ | ||||||
|  | class UserRepository extends ServiceEntityRepository implements PasswordUpgraderInterface | ||||||
|  | { | ||||||
|  |     public function __construct(ManagerRegistry $registry) | ||||||
|  |     { | ||||||
|  |         parent::__construct($registry, User::class); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Used to upgrade (rehash) the user's password automatically over time. | ||||||
|  |      */ | ||||||
|  |     public function upgradePassword(PasswordAuthenticatedUserInterface $user, string $newHashedPassword): void | ||||||
|  |     { | ||||||
|  |         if (!$user instanceof User) { | ||||||
|  |             throw new UnsupportedUserException(sprintf('Instances of "%s" are not supported.', \get_class($user))); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $user->setPassword($newHashedPassword); | ||||||
|  |         $this->_em->persist($user); | ||||||
|  |         $this->_em->flush(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // /** | ||||||
|  |     //  * @return User[] Returns an array of User objects | ||||||
|  |     //  */ | ||||||
|  |     /* | ||||||
|  |     public function findByExampleField($value) | ||||||
|  |     { | ||||||
|  |         return $this->createQueryBuilder('u') | ||||||
|  |             ->andWhere('u.exampleField = :val') | ||||||
|  |             ->setParameter('val', $value) | ||||||
|  |             ->orderBy('u.id', 'ASC') | ||||||
|  |             ->setMaxResults(10) | ||||||
|  |             ->getQuery() | ||||||
|  |             ->getResult() | ||||||
|  |         ; | ||||||
|  |     } | ||||||
|  |     */ | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |     public function findOneBySomeField($value): ?User | ||||||
|  |     { | ||||||
|  |         return $this->createQueryBuilder('u') | ||||||
|  |             ->andWhere('u.exampleField = :val') | ||||||
|  |             ->setParameter('val', $value) | ||||||
|  |             ->getQuery() | ||||||
|  |             ->getOneOrNullResult() | ||||||
|  |         ; | ||||||
|  |     } | ||||||
|  |     */ | ||||||
|  | } | ||||||
| @@ -3,7 +3,7 @@ | |||||||
| <head> | <head> | ||||||
|     <meta charset="utf-8"> |     <meta charset="utf-8"> | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1"> |     <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|     <title>{% block title %}Welcome!{% endblock %}</title> |     <title>{% block title %}IceCold!{% endblock %}</title> | ||||||
|     <link rel="icon" |     <link rel="icon" | ||||||
|           href="data:image/svg+xml,<svg xmlns=%22http://www.w3.org/2000/svg%22 viewBox=%220 0 128 128%22><text y=%221.2em%22 font-size=%2296%22>⚫️</text></svg>"> |           href="data:image/svg+xml,<svg xmlns=%22http://www.w3.org/2000/svg%22 viewBox=%220 0 128 128%22><text y=%221.2em%22 font-size=%2296%22>⚫️</text></svg>"> | ||||||
|     {# Run `composer require symfony/webpack-encore-bundle` to start using Symfony UX #} |     {# Run `composer require symfony/webpack-encore-bundle` to start using Symfony UX #} | ||||||
| @@ -17,7 +17,7 @@ | |||||||
| </head> | </head> | ||||||
| <body> | <body> | ||||||
| {% include 'base/navbar.html.twig' %} | {% include 'base/navbar.html.twig' %} | ||||||
| <main> | <main style="padding-top: 64px;"> | ||||||
|     {% block body %} |     {% block body %} | ||||||
|     {% endblock %} |     {% endblock %} | ||||||
| </main> | </main> | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								templates/registration/register.html.twig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								templates/registration/register.html.twig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | {% extends 'base/base.html.twig' %} | ||||||
|  |  | ||||||
|  | {% block body %} | ||||||
|  |     <div class="container"> | ||||||
|  |         <h1>Register</h1> | ||||||
|  |  | ||||||
|  |         {{ form_start(registrationForm) }} | ||||||
|  |         {{ form_row(registrationForm.username) }} | ||||||
|  |         {{ form_row(registrationForm.plainPassword, { | ||||||
|  |             label: 'Password' | ||||||
|  |         }) }} | ||||||
|  |         {{ form_row(registrationForm.agreeTerms) }} | ||||||
|  |  | ||||||
|  |         <button type="submit" class="btn">Register</button> | ||||||
|  |         {{ form_end(registrationForm) }} | ||||||
|  |     </div> | ||||||
|  | {% endblock %} | ||||||
		Reference in New Issue
	
	Block a user