diff --git a/src/app/admin-dashboard/admin-dashboard.component.html b/src/app/admin-dashboard/admin-dashboard.component.html index afa9155cbeb4cde0d718505b588fe35870f258d8..ba3bcbcc83206126de51f465621415703f281190 100644 --- a/src/app/admin-dashboard/admin-dashboard.component.html +++ b/src/app/admin-dashboard/admin-dashboard.component.html @@ -8,6 +8,9 @@ (click)="clickAddUser()"> Add User </button> + <button type="button" class="btn btn-danger" (click)="logout()"> + Logout + </button> </div> </div> </nav> diff --git a/src/app/admin-dashboard/admin-dashboard.component.ts b/src/app/admin-dashboard/admin-dashboard.component.ts index 400634a4ec75505124e442d6a247ba4c176e33b7..14ca77d7d5b1d86b3070d51abc6f6092c24f5713 100644 --- a/src/app/admin-dashboard/admin-dashboard.component.ts +++ b/src/app/admin-dashboard/admin-dashboard.component.ts @@ -2,6 +2,7 @@ import {Component, OnInit} from '@angular/core'; import {FormBuilder, FormGroup} from "@angular/forms"; import {UserModel} from "./user.model"; import {ApiService} from "../shared/api.service"; +import {Router} from "@angular/router"; @Component({ selector: 'app-admin-dashboard', @@ -17,7 +18,7 @@ export class AdminDashboardComponent implements OnInit { showAddButton!: boolean; showUpdateButton!: boolean; - constructor(private fb: FormBuilder, private api: ApiService) { + constructor(private fb: FormBuilder, private api: ApiService, private router: Router) { this.form = this.fb.group({ username: [''], email: [''], @@ -106,4 +107,10 @@ export class AdminDashboardComponent implements OnInit { alert('An error has occurred.'); }); } + + logout() { + localStorage.removeItem('token'); + localStorage.removeItem('role'); + this.router.navigate(['/login']); + } } diff --git a/src/app/admin-dashboard/user.model.ts b/src/app/admin-dashboard/user.model.ts index 5f20eabfeab3a7926a3a7419f48ecb6ff7d96a18..d4688281908ddba4218bcb5d7eb9df75c834f2a6 100644 --- a/src/app/admin-dashboard/user.model.ts +++ b/src/app/admin-dashboard/user.model.ts @@ -1,6 +1,7 @@ export class UserModel { id: string = ''; + token: string = ''; username: string = ''; email: string = ''; password: string = ''; diff --git a/src/app/app-routing.module.ts b/src/app/app-routing.module.ts index 02972627f8df364102ce4ede71c8bd5f3660e1d8..90f3b91f95049e34e379c5ffb493d7b59f43082b 100644 --- a/src/app/app-routing.module.ts +++ b/src/app/app-routing.module.ts @@ -1,10 +1,21 @@ -import { NgModule } from '@angular/core'; -import { RouterModule, Routes } from '@angular/router'; +import {NgModule} from '@angular/core'; +import {RouterModule, Routes} from '@angular/router'; +import {LoginComponent} from "./login/login.component"; +import {SignupComponent} from "./signup/signup.component"; +import {AdminDashboardComponent} from "./admin-dashboard/admin-dashboard.component"; +import {AuthGuard} from "./shared/auth/auth.guard"; +import {RoleGuard} from "./shared/auth/role.guard"; -const routes: Routes = []; +const routes: Routes = [ + {path: '', redirectTo: 'login', pathMatch: 'full'}, + {path: 'login', component: LoginComponent}, + {path: 'signup', component: SignupComponent}, + {path: 'admin', component: AdminDashboardComponent, canActivate: [RoleGuard], data: {roles: ['ROLE_ADMIN']}}, +]; @NgModule({ imports: [RouterModule.forRoot(routes)], exports: [RouterModule] }) -export class AppRoutingModule { } +export class AppRoutingModule { +} diff --git a/src/app/app.component.html b/src/app/app.component.html index a1e346b4f9900c17f2565f398c12a5014dea6015..0680b43f9c6ae05df91c576141f20ed411d07c7d 100644 --- a/src/app/app.component.html +++ b/src/app/app.component.html @@ -1 +1 @@ -<app-admin-dashboard></app-admin-dashboard> +<router-outlet></router-outlet> diff --git a/src/app/app.module.ts b/src/app/app.module.ts index 791980c077beeda13b5bd16ac953a7f2343b52d9..42bc1f682583804c5b97cf1ca5fbafb16276f0cf 100644 --- a/src/app/app.module.ts +++ b/src/app/app.module.ts @@ -6,11 +6,15 @@ import {AppComponent} from './app.component'; import {AdminDashboardComponent} from './admin-dashboard/admin-dashboard.component'; import {ReactiveFormsModule} from "@angular/forms"; import {HttpClientModule} from "@angular/common/http"; +import { LoginComponent } from './login/login.component'; +import { SignupComponent } from './signup/signup.component'; @NgModule({ declarations: [ AppComponent, - AdminDashboardComponent + AdminDashboardComponent, + LoginComponent, + SignupComponent ], imports: [ BrowserModule, diff --git a/src/app/login/login.component.css b/src/app/login/login.component.css new file mode 100644 index 0000000000000000000000000000000000000000..2954033bef26e309c8b01b346dc4317e854de181 --- /dev/null +++ b/src/app/login/login.component.css @@ -0,0 +1,24 @@ +.card { + margin-top: 10px; + border: none; + width: 500px; + padding: 40px; + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + background: hsla(205, 46%, 30%, 0.8); + background: linear-gradient(45deg, hsla(205, 46%, 30%, 0.8), hsla(24, 46%, 30%, 0.8)); +} + +.card h1 { + color: #fff; +} + +.form-text, h6 { + color: darkgray; +} + +label { + color: #fff; +} diff --git a/src/app/login/login.component.html b/src/app/login/login.component.html new file mode 100644 index 0000000000000000000000000000000000000000..9b68571f63e8eb7408e13abcbc59a13c337a7f46 --- /dev/null +++ b/src/app/login/login.component.html @@ -0,0 +1,27 @@ +<div class="container"> + <div class="row"> + <div class="col-md-6"> + <div class="card"> + <div class="text-center"> + <!-- <img src="https://www.w3schools.com/bootstrap4/img_avatar1.png" alt="John" style="width:20%">--> + <h1>Login</h1> + <h6>Enter your username and password</h6> + <form [formGroup]="formLogin" (ngSubmit)="login()"> + <div class="mb-3"> + <label for="username" class="form-label">Username</label> + <input type="text" class="form-control" id="username" aria-describedby="emailHelp" + formControlName="username"> + <div id="emailHelp" class="form-text">We'll never share your username with anyone else.</div> + </div> + <div class="mb-3"> + <label for="password" class="form-label">Password</label> + <input type="password" class="form-control" id="password" formControlName="password"> + </div> + <button type="submit" class="btn btn-primary">Login</button> + </form> + <a class="form-text" routerLink="/signup">New user? Click to sign up!</a> + </div> + </div> + </div> + </div> +</div> diff --git a/src/app/login/login.component.spec.ts b/src/app/login/login.component.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..10eca249d528f4bcf8ef434696978a242a03e487 --- /dev/null +++ b/src/app/login/login.component.spec.ts @@ -0,0 +1,23 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { LoginComponent } from './login.component'; + +describe('LoginComponent', () => { + let component: LoginComponent; + let fixture: ComponentFixture<LoginComponent>; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ LoginComponent ] + }) + .compileComponents(); + + fixture = TestBed.createComponent(LoginComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src/app/login/login.component.ts b/src/app/login/login.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..531aaca1d0d47ad044361179948883370cc0ae86 --- /dev/null +++ b/src/app/login/login.component.ts @@ -0,0 +1,40 @@ +import {Component} from '@angular/core'; +import {Router} from "@angular/router"; +import {FormBuilder, FormGroup} from "@angular/forms"; +import {ApiService} from "../shared/api.service"; + +@Component({ + selector: 'app-login', + templateUrl: './login.component.html', + styleUrls: ['./login.component.css'] +}) +export class LoginComponent { + + formLogin!: FormGroup; + + constructor(private fb: FormBuilder, private router: Router, private api: ApiService) { + this.formLogin = fb.group({ + username: [''], + password: [''] + }); + } + + login() { + this.api.post("http://localhost:8080/api/v1/auth/login", this.formLogin.value).subscribe((res: any) => { + if (res && res.token) { + const role = res.roles[0]; + localStorage.setItem('token', res.token); + localStorage.setItem('role', role); + if (role === 'ROLE_ADMIN') { + this.router.navigate(['/admin']); + } else { + this.router.navigate(['/']); + } + this.formLogin.reset(); + } + }, err => { + alert("Wrong username or password. Try again!") + this.formLogin.controls['password'].setValue(''); + }); + } +} diff --git a/src/app/shared/auth/auth.guard.spec.ts b/src/app/shared/auth/auth.guard.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..68889d22d920e3b78febae9e5d323c717887961e --- /dev/null +++ b/src/app/shared/auth/auth.guard.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { AuthGuard } from './auth.guard'; + +describe('AuthGuard', () => { + let guard: AuthGuard; + + beforeEach(() => { + TestBed.configureTestingModule({}); + guard = TestBed.inject(AuthGuard); + }); + + it('should be created', () => { + expect(guard).toBeTruthy(); + }); +}); diff --git a/src/app/shared/auth/auth.guard.ts b/src/app/shared/auth/auth.guard.ts new file mode 100644 index 0000000000000000000000000000000000000000..5ca3570b698bb92e58bb3b5a01b6d777a0aabeb2 --- /dev/null +++ b/src/app/shared/auth/auth.guard.ts @@ -0,0 +1,21 @@ +import {Injectable} from '@angular/core'; +import {ActivatedRouteSnapshot, CanActivate, Router, RouterStateSnapshot, UrlTree} from '@angular/router'; +import {Observable} from 'rxjs'; +import {ApiService} from "../api.service"; +import {AuthService} from "./auth.service"; + +@Injectable({ + providedIn: 'root' +}) +export class AuthGuard implements CanActivate { + constructor(private auth: AuthService, private router: Router) { + } + + canActivate(): boolean { + if (this.auth.isLoggedIn()) { + return true; + } + this.router.navigate(['/login']); + return false; + } +} diff --git a/src/app/shared/auth/auth.service.spec.ts b/src/app/shared/auth/auth.service.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..f1251cacf9dda3b3bd9dc5222583fafa110ffe8f --- /dev/null +++ b/src/app/shared/auth/auth.service.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { AuthService } from './auth.service'; + +describe('AuthService', () => { + let service: AuthService; + + beforeEach(() => { + TestBed.configureTestingModule({}); + service = TestBed.inject(AuthService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); +}); diff --git a/src/app/shared/auth/auth.service.ts b/src/app/shared/auth/auth.service.ts new file mode 100644 index 0000000000000000000000000000000000000000..c12feb28a3ba10800fac642aff1a2552849a74b1 --- /dev/null +++ b/src/app/shared/auth/auth.service.ts @@ -0,0 +1,12 @@ +import { Injectable } from '@angular/core'; + +@Injectable({ + providedIn: 'root' +}) +export class AuthService { + + constructor() { } + isLoggedIn() { + return !!localStorage.getItem('token') && !!localStorage.getItem('role'); + } +} diff --git a/src/app/shared/auth/role.guard.spec.ts b/src/app/shared/auth/role.guard.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..5b6db9fd568a85482d99640792a1acde7497962b --- /dev/null +++ b/src/app/shared/auth/role.guard.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { RoleGuard } from './role.guard'; + +describe('RoleGuard', () => { + let guard: RoleGuard; + + beforeEach(() => { + TestBed.configureTestingModule({}); + guard = TestBed.inject(RoleGuard); + }); + + it('should be created', () => { + expect(guard).toBeTruthy(); + }); +}); diff --git a/src/app/shared/auth/role.guard.ts b/src/app/shared/auth/role.guard.ts new file mode 100644 index 0000000000000000000000000000000000000000..a7d1c1e989b4f91af9976a20087ddbe40b0dab71 --- /dev/null +++ b/src/app/shared/auth/role.guard.ts @@ -0,0 +1,16 @@ +import { Injectable } from '@angular/core'; +import { ActivatedRouteSnapshot, CanActivate, RouterStateSnapshot, UrlTree } from '@angular/router'; +import { Observable } from 'rxjs'; + +@Injectable({ + providedIn: 'root' +}) +export class RoleGuard implements CanActivate { + canActivate(): boolean { + if (localStorage.getItem('role') === 'ROLE_ADMIN') { + return true; + } + alert('You are not authorized to view this page!'); + return false; + } +} diff --git a/src/app/signup/signup.component.css b/src/app/signup/signup.component.css new file mode 100644 index 0000000000000000000000000000000000000000..2954033bef26e309c8b01b346dc4317e854de181 --- /dev/null +++ b/src/app/signup/signup.component.css @@ -0,0 +1,24 @@ +.card { + margin-top: 10px; + border: none; + width: 500px; + padding: 40px; + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + background: hsla(205, 46%, 30%, 0.8); + background: linear-gradient(45deg, hsla(205, 46%, 30%, 0.8), hsla(24, 46%, 30%, 0.8)); +} + +.card h1 { + color: #fff; +} + +.form-text, h6 { + color: darkgray; +} + +label { + color: #fff; +} diff --git a/src/app/signup/signup.component.html b/src/app/signup/signup.component.html new file mode 100644 index 0000000000000000000000000000000000000000..3ae7c3e47db0c9fab22914164d2e5eff1acca7f1 --- /dev/null +++ b/src/app/signup/signup.component.html @@ -0,0 +1,31 @@ +<div class="container"> + <div class="row"> + <div class="col-md-6"> + <div class="card"> + <div class="text-center"> + <!-- <img src="https://www.w3schools.com/bootstrap4/img_avatar1.png" alt="John" style="width:20%">--> + <h1>Sign up</h1> + <h6>Register yourself!</h6> + <form [formGroup]="formSignup" (ngSubmit)="singUp()"> + <div class="mb-3"> + <label for="username" class="form-label">Username</label> + <input type="text" class="form-control" id="username" aria-describedby="emailHelp" + formControlName="username"> + </div> + <div class="mb-3"> + <label for="email" class="form-label">Email</label> + <input type="email" class="form-control" id="email" aria-describedby="emailHelp" formControlName="email"> + <div id="emailHelp" class="form-text">We'll never share your email with anyone else.</div> + </div> + <div class="mb-3"> + <label for="password" class="form-label">Password</label> + <input type="password" class="form-control" id="password" formControlName="password"> + </div> + <button type="submit" class="btn btn-primary">Sign up</button> + </form> + <a class="form-text" routerLink="/login">Already a user? Click to login!</a> + </div> + </div> + </div> + </div> +</div> diff --git a/src/app/signup/signup.component.spec.ts b/src/app/signup/signup.component.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..2643b204064d3c0d0c4ccf7495229ee39ce8d8b2 --- /dev/null +++ b/src/app/signup/signup.component.spec.ts @@ -0,0 +1,23 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { SignupComponent } from './signup.component'; + +describe('SignupComponent', () => { + let component: SignupComponent; + let fixture: ComponentFixture<SignupComponent>; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ SignupComponent ] + }) + .compileComponents(); + + fixture = TestBed.createComponent(SignupComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src/app/signup/signup.component.ts b/src/app/signup/signup.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..b19b19ab17f05056a7a8a6cd49ea204e8b8657d5 --- /dev/null +++ b/src/app/signup/signup.component.ts @@ -0,0 +1,40 @@ +import {Component} from '@angular/core'; +import {FormBuilder, FormGroup} from "@angular/forms"; +import {ApiService} from "../shared/api.service"; +import {Router} from "@angular/router"; + +@Component({ + selector: 'app-signup', + templateUrl: './signup.component.html', + styleUrls: ['./signup.component.css'] +}) +export class SignupComponent { + + formSignup!: FormGroup; + + constructor(private fb: FormBuilder, private api: ApiService, private router: Router) { + this.formSignup = this.fb.group({ + username: [''], + email: [''], + password: [''] + }); + } + + singUp() { + const payload = { + username: this.formSignup.value.username, + email: this.formSignup.value.email, + password: this.formSignup.value.password, + roles: ['user'] + } + return this.api.post('http://localhost:8080/api/v1/auth/signup', payload) + .subscribe(res => { + console.log(res); + this.formSignup.reset(); + alert('Sign up successful.'); + this.router.navigate(['login']); + }, err => { + alert('An error has occurred. Try again.'); + }); + } +}