From db66a09e3b81808044e58431af4581002b066260 Mon Sep 17 00:00:00 2001 From: Elise Lebeau Date: Fri, 29 Sep 2023 11:10:48 +0200 Subject: [PATCH] add linked ACE to user and group detail page --- src/app/app-routing.module.ts | 5 +- .../group-details.component.html | 37 +++++++--- .../group-details.component.scss | 4 ++ .../group-details/group-details.component.ts | 67 +++++++------------ .../resource-pool-details.component.html | 4 +- .../user-detail/user-detail.component.html | 31 +++++++++ .../user-detail/user-detail.component.scss | 4 ++ .../user-detail/user-detail.component.ts | 30 +++++++-- src/app/models/api/ACE.ts | 4 ++ .../group-ace.resolver.ts.resolver.ts | 41 ++++++++++++ src/app/resolvers/user-aces.resolver.ts | 38 +++++++++++ 11 files changed, 207 insertions(+), 58 deletions(-) create mode 100644 src/app/resolvers/group-ace.resolver.ts.resolver.ts create mode 100644 src/app/resolvers/user-aces.resolver.ts diff --git a/src/app/app-routing.module.ts b/src/app/app-routing.module.ts index 7f6ea897..2fe06586 100644 --- a/src/app/app-routing.module.ts +++ b/src/app/app-routing.module.ts @@ -73,6 +73,8 @@ import {AceManagementComponent} from "@components/ace-management/ace-management. import {ResourcePoolsManagementComponent} from "@components/resource-pools-management/resource-pools-management.component"; import {ResourcePoolDetailsComponent} from "@components/resource-pool-details/resource-pool-details.component"; import {ResourcePoolsResolver} from "@resolvers/resource-pools.resolver"; +import {GroupAcesResolver} from "@resolvers/group-ace.resolver.ts.resolver"; +import {UserAcesResolver} from "@resolvers/user-aces.resolver"; const routes: Routes = [ { @@ -101,6 +103,7 @@ const routes: Routes = [ resolve: { user: UserDetailResolver, groups: UserGroupsResolver, + aces: UserAcesResolver, controller: ControllerResolve}, }, { @@ -262,7 +265,7 @@ const routes: Routes = [ members: GroupMembersResolver, controller: ControllerResolve, group: GroupResolver, - roles: GroupRoleResolver + aces: GroupAcesResolver } }, { diff --git a/src/app/components/group-details/group-details.component.html b/src/app/components/group-details/group-details.component.html index 0c378354..cf0df37d 100644 --- a/src/app/components/group-details/group-details.component.html +++ b/src/app/components/group-details/group-details.component.html @@ -56,15 +56,34 @@ [pageSizeOptions]="[5, 20, 50, 100]"> - -
-
-
{{role.name}}
-
- -
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + +
Endpoint {{element.endpoint_name}} Role {{element.role_name}} Propagate {{element.propagate}} Allowed {{element.allowed}}
diff --git a/src/app/components/group-details/group-details.component.scss b/src/app/components/group-details/group-details.component.scss index e8417eab..b2b1f702 100644 --- a/src/app/components/group-details/group-details.component.scss +++ b/src/app/components/group-details/group-details.component.scss @@ -1,3 +1,7 @@ +table { + width: 100%; +} + .main { display: flex; justify-content: space-around; diff --git a/src/app/components/group-details/group-details.component.ts b/src/app/components/group-details/group-details.component.ts index eb0eb475..6dbe26c8 100644 --- a/src/app/components/group-details/group-details.component.ts +++ b/src/app/components/group-details/group-details.component.ts @@ -22,8 +22,14 @@ import {RemoveToGroupDialogComponent} from "@components/group-details/remove-to- import {GroupService} from "@services/group.service"; import {ToasterService} from "@services/toaster.service"; import {PageEvent} from "@angular/material/paginator"; +import {ACE, ACEDetailed, AceType} from "@models/api/ACE"; +import {UserService} from "@services/user.service"; +import {RoleService} from "@services/role.service"; import {Role} from "@models/api/role"; -import {AddRoleToGroupComponent} from "@components/group-details/add-role-to-group/add-role-to-group.component"; +import {AclService} from "@services/acl.service"; +import {Endpoint} from "@models/api/endpoint"; +import {interval} from "rxjs"; +import {MatTableDataSource} from "@angular/material/table"; @Component({ selector: 'app-group-details', @@ -37,28 +43,43 @@ export class GroupDetailsComponent implements OnInit { editGroupForm: UntypedFormGroup; pageEvent: PageEvent | undefined; searchMembers: string; - roles: Role[]; + aces: ACE[]; + aceDatasource = new MatTableDataSource(); + public aceDisplayedColumns = ['endpoint', 'role', 'propagate', 'allowed']; constructor(private route: ActivatedRoute, private dialog: MatDialog, private groupService: GroupService, - private toastService: ToasterService) { + private toastService: ToasterService, + private aclService: AclService, + private roleService: RoleService) { this.editGroupForm = new UntypedFormGroup({ groupname: new UntypedFormControl(''), }); - this.route.data.subscribe((d: { controller: Controller; group: Group, members: User[], roles: Role[] }) => { + this.route.data.subscribe((d: { controller: Controller; group: Group, members: User[], aces: ACE[] }) => { this.controller = d.controller; this.group = d.group; - this.roles = d.roles; + this.aces = d.aces; this.members = d.members.sort((a: User, b: User) => a.username.toLowerCase().localeCompare(b.username.toLowerCase())); this.editGroupForm.setValue({groupname: this.group.name}); }); + + } ngOnInit(): void { + this.roleService.get(this.controller).subscribe((roles: Role[]) => { + this.aclService.getEndpoints(this.controller).subscribe((endps: Endpoint[]) => { + this.aceDatasource.data = this.aces.map((ace: ACE) => { + const endpoint = endps.filter((endp: Endpoint) => endp.endpoint === ace.path)[0] + const role = roles.filter((r: Role) => r.role_id === ace.role_id)[0] + return {...ace, endpoint_name: endpoint.name, role_name: role.name} + }) + }) + }) } @@ -72,18 +93,6 @@ export class GroupDetailsComponent implements OnInit { }); } - openAddRoleDialog() { - this.dialog - .open(AddRoleToGroupComponent, - { - width: '700px', height: '500px', - data: {controller: this.controller, group: this.group} - }) - .afterClosed() - .subscribe(() => { - this.reloadRoles(); - }); - } openAddUserDialog() { this.dialog .open(AddUserToGroupDialogComponent, @@ -117,24 +126,6 @@ export class GroupDetailsComponent implements OnInit { } - openRemoveRoleDialog(role: Role) { - this.dialog.open(RemoveToGroupDialogComponent, - {width: '500px', height: '200px', data: {name: role.name}}) - .afterClosed() - .subscribe((confirm: string) => { - if (confirm) { - this.groupService.removeRole(this.controller, this.group, role) - .subscribe(() => { - this.toastService.success(`Role ${role.name} was removed`); - this.reloadRoles(); - }, - (error) => { - this.toastService.error(`Error while removing role ${role.name} from ${this.group.name}`); - console.log(error); - }); - } - }); - } reloadMembers() { this.groupService.getGroupMember(this.controller, this.group.user_group_id) @@ -143,10 +134,4 @@ export class GroupDetailsComponent implements OnInit { }); } - reloadRoles() { - this.groupService.getGroupRole(this.controller, this.group.user_group_id) - .subscribe((roles: Role[]) => { - this.roles = roles; - }); - } } diff --git a/src/app/components/resource-pool-details/resource-pool-details.component.html b/src/app/components/resource-pool-details/resource-pool-details.component.html index bc2a77cc..51db7933 100644 --- a/src/app/components/resource-pool-details/resource-pool-details.component.html +++ b/src/app/components/resource-pool-details/resource-pool-details.component.html @@ -8,7 +8,7 @@ [routerLink]="['/controller', controller.id, 'management', 'resourcePools']"> keyboard_arrow_left -

role {{pool.name}} details

+

Pool {{pool.name}} details

@@ -23,7 +23,7 @@
uuid: {{pool.resource_pool_id}}
diff --git a/src/app/components/user-management/user-detail/user-detail.component.html b/src/app/components/user-management/user-detail/user-detail.component.html index 11462e39..78c1626a 100644 --- a/src/app/components/user-management/user-detail/user-detail.component.html +++ b/src/app/components/user-management/user-detail/user-detail.component.html @@ -80,6 +80,37 @@
+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + +
Endpoint {{element.endpoint_name}} Role {{element.role_name}} Propagate {{element.propagate}} Allowed {{element.allowed}}
+
+
diff --git a/src/app/components/user-management/user-detail/user-detail.component.scss b/src/app/components/user-management/user-detail/user-detail.component.scss index a64b61ba..2a8103d5 100644 --- a/src/app/components/user-management/user-detail/user-detail.component.scss +++ b/src/app/components/user-management/user-detail/user-detail.component.scss @@ -5,3 +5,7 @@ .button-div { float: right; } + +table { + width: 100%; +} diff --git a/src/app/components/user-management/user-detail/user-detail.component.ts b/src/app/components/user-management/user-detail/user-detail.component.ts index 64a7c150..ee0200e0 100644 --- a/src/app/components/user-management/user-detail/user-detail.component.ts +++ b/src/app/components/user-management/user-detail/user-detail.component.ts @@ -8,11 +8,15 @@ import {Controller} from "@models/controller"; import {userNameAsyncValidator} from "@components/user-management/add-user-dialog/userNameAsyncValidator"; import {userEmailAsyncValidator} from "@components/user-management/add-user-dialog/userEmailAsyncValidator"; import {ActivatedRoute, Router} from "@angular/router"; -import {Role} from "@models/api/role"; -import {AddUserDialogComponent} from "@components/user-management/add-user-dialog/add-user-dialog.component"; import {MatDialog} from "@angular/material/dialog"; import {ChangeUserPasswordComponent} from "@components/user-management/user-detail/change-user-password/change-user-password.component"; -import {RemoveToGroupDialogComponent} from "@components/group-details/remove-to-group-dialog/remove-to-group-dialog.component"; +import {ACE, ACEDetailed} from "@models/api/ACE"; +import {MatTableDataSource} from "@angular/material/table"; +import {AclService} from "@services/acl.service"; +import {RoleService} from "@services/role.service"; +import {Role} from "@models/api/role"; +import {Endpoint} from "@models/api/endpoint"; + @Component({ selector: 'app-user-detail', @@ -27,12 +31,17 @@ export class UserDetailComponent implements OnInit { controller: Controller; user_id: string; changingPassword: boolean = false; + aces: ACE[]; + aceDatasource = new MatTableDataSource(); + public aceDisplayedColumns = ['endpoint', 'role', 'propagate', 'allowed']; constructor(public userService: UserService, private toasterService: ToasterService, private route: ActivatedRoute, private router: Router, - public dialog: MatDialog) { + public dialog: MatDialog, + private aclService: AclService, + private roleService: RoleService) { } @@ -40,11 +49,22 @@ export class UserDetailComponent implements OnInit { this.controller = this.route.snapshot.data['controller']; if (!this.controller) this.router.navigate(['/controllers']); - this.route.data.subscribe((d: { controller: Controller; user: User, groups: Group[]}) => { + this.route.data.subscribe((d: { controller: Controller; user: User, groups: Group[], aces: ACE[]}) => { this.user = d.user; this.user_id = this.user.user_id; this.groups = d.groups; + this.aces = d.aces; this.initForm(); + + this.roleService.get(this.controller).subscribe((roles: Role[]) => { + this.aclService.getEndpoints(this.controller).subscribe((endps: Endpoint[]) => { + this.aceDatasource.data = this.aces.map((ace: ACE) => { + const endpoint = endps.filter((endp: Endpoint) => endp.endpoint === ace.path)[0] + const role = roles.filter((r: Role) => r.role_id === ace.role_id)[0] + return {...ace, endpoint_name: endpoint.name, role_name: role.name} + }) + }) + }) }); } diff --git a/src/app/models/api/ACE.ts b/src/app/models/api/ACE.ts index dab2ebcc..cecee5ac 100644 --- a/src/app/models/api/ACE.ts +++ b/src/app/models/api/ACE.ts @@ -3,6 +3,10 @@ export enum AceType { user = "user" } +export interface ACEDetailed extends ACE{ + endpoint_name: string; + role_name: string; +} export interface ACE { ace_id: string; diff --git a/src/app/resolvers/group-ace.resolver.ts.resolver.ts b/src/app/resolvers/group-ace.resolver.ts.resolver.ts new file mode 100644 index 00000000..edf75ec4 --- /dev/null +++ b/src/app/resolvers/group-ace.resolver.ts.resolver.ts @@ -0,0 +1,41 @@ +import { Injectable } from '@angular/core'; +import { + Router, Resolve, + RouterStateSnapshot, + ActivatedRouteSnapshot +} from '@angular/router'; +import {Observable,Subscriber} from 'rxjs'; +import {ACE} from "../models/api/ACE"; +import {ControllerService} from "../services/controller.service"; +import {AclService} from "../services/acl.service"; +import {Controller} from "../models/controller"; +import {UserService} from "@services/user.service"; +import {GroupService} from "@services/group.service"; +import {RoleService} from "@services/role.service"; + +@Injectable({ + providedIn: 'root' +}) +export class GroupAcesResolver implements Resolve { + + constructor(private controllerService: ControllerService, + private aceService: AclService) { + } + + resolve(route: ActivatedRouteSnapshot, state: RouterStateSnapshot): Observable { + return new Observable((subscriber: Subscriber) => { + + const controllerId = route.paramMap.get('controller_id'); + const groupId = route.paramMap.get('user_group_id'); + + this.controllerService.get(+controllerId).then((controller: Controller) => { + this.aceService.list(controller).subscribe((aces: ACE[]) => { + const filter = aces.filter((ace: ACE) => ace.group_id === groupId) + + subscriber.next(filter); + subscriber.complete() + }) + }); + }); + } +} diff --git a/src/app/resolvers/user-aces.resolver.ts b/src/app/resolvers/user-aces.resolver.ts new file mode 100644 index 00000000..d74bc3ff --- /dev/null +++ b/src/app/resolvers/user-aces.resolver.ts @@ -0,0 +1,38 @@ +import { Injectable } from '@angular/core'; +import { + Router, Resolve, + RouterStateSnapshot, + ActivatedRouteSnapshot +} from '@angular/router'; +import {Observable, of, Subscriber} from 'rxjs'; +import {ACE} from "@models/api/ACE"; +import {Controller} from "@models/controller"; +import {ControllerService} from "@services/controller.service"; +import {AclService} from "@services/acl.service"; + +@Injectable({ + providedIn: 'root' +}) +export class UserAcesResolver implements Resolve { + + constructor(private controllerService: ControllerService, + private aceService: AclService) { + } + + resolve(route: ActivatedRouteSnapshot, state: RouterStateSnapshot): Observable { + return new Observable((subscriber: Subscriber) => { + + const controllerId = route.paramMap.get('controller_id'); + const userId = route.paramMap.get('user_id'); + + this.controllerService.get(+controllerId).then((controller: Controller) => { + this.aceService.list(controller).subscribe((aces: ACE[]) => { + const filter = aces.filter((ace: ACE) => ace.user_id === userId) + + subscriber.next(filter); + subscriber.complete() + }) + }); + }); + } +}