From 9c0038cbf28b02d80007c685024e62747e169d75 Mon Sep 17 00:00:00 2001 From: Felipe Barreta Date: Tue, 5 Nov 2024 17:37:49 -0800 Subject: [PATCH] ALCS-2353 Constraints end error handling --- .../tag-category-dialog.component.html | 2 +- .../tag-category-dialog.component.ts | 4 +- .../tag/tag-dialog/tag-dialog.component.html | 2 +- .../tag/tag-dialog/tag-dialog.component.ts | 4 +- .../tag/tag-category/tag-category.service.ts | 8 ++-- .../tag-category/tag-category.controller.ts | 48 ++----------------- .../tag/tag-category/tag-category.entity.ts | 2 +- .../tag/tag-category/tag-category.service.ts | 19 +++++++- .../apps/alcs/src/alcs/tag/tag.controller.ts | 48 ++----------------- services/apps/alcs/src/alcs/tag/tag.dto.ts | 3 ++ services/apps/alcs/src/alcs/tag/tag.entity.ts | 4 +- .../apps/alcs/src/alcs/tag/tag.service.ts | 17 +++++++ ...1730856145155-remove_unique_constraints.ts | 16 +++++++ 13 files changed, 77 insertions(+), 100 deletions(-) create mode 100644 services/apps/alcs/src/providers/typeorm/migrations/1730856145155-remove_unique_constraints.ts diff --git a/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.html b/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.html index 374954022f..f4df4b696a 100644 --- a/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.html +++ b/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.html @@ -7,7 +7,7 @@

{{ isEdit ? 'Edit' : 'Create New' }} Category

Name - +
diff --git a/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.ts b/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.ts index e6ef70d084..8ade8f8cb9 100644 --- a/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.ts +++ b/alcs-frontend/src/app/features/admin/tag/tag-category/tag-category-dialog/tag-category-dialog.component.ts @@ -2,6 +2,7 @@ import { Component, Inject } from '@angular/core'; import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog'; import { TagCategoryDto } from '../../../../../services/tag/tag-category/tag-category.dto'; import { TagCategoryService } from '../../../../../services/tag/tag-category/tag-category.service'; +import { FormControl } from '@angular/forms'; @Component({ selector: 'app-tag-category-dialog', @@ -15,6 +16,7 @@ export class TagCategoryDialogComponent { isLoading = false; isEdit = false; showNameWarning = false; + nameControl = new FormControl(); constructor( @Inject(MAT_DIALOG_DATA) public data: TagCategoryDto | undefined, @@ -63,6 +65,6 @@ export class TagCategoryDialogComponent { private showWarning() { this.showNameWarning = true; - this.name = ''; + this.nameControl.setErrors({"invalid": true}); } } diff --git a/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.html b/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.html index c5a734706c..ca9a9bb41b 100644 --- a/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.html +++ b/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.html @@ -8,7 +8,7 @@

{{ isEdit ? 'Edit' : 'Create New' }} Tag

Name - +
diff --git a/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.ts b/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.ts index b03786f32e..b99a2f481b 100644 --- a/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.ts +++ b/alcs-frontend/src/app/features/admin/tag/tag-dialog/tag-dialog.component.ts @@ -5,6 +5,7 @@ import { TagService } from '../../../../services/tag/tag.service'; import { TagCategoryService } from '../../../../services/tag/tag-category/tag-category.service'; import { TagCategoryDto } from 'src/app/services/tag/tag-category/tag-category.dto'; import { Subject, takeUntil } from 'rxjs'; +import { FormControl } from '@angular/forms'; @Component({ selector: 'app-tag-dialog', @@ -25,6 +26,7 @@ export class TagDialogComponent implements OnInit { isLoading = false; isEdit = false; showNameWarning = false; + nameControl = new FormControl(); categories: TagCategoryDto[] = []; @@ -97,6 +99,6 @@ export class TagDialogComponent implements OnInit { private showWarning() { this.showNameWarning = true; - this.name = ''; + this.nameControl.setErrors({"invalid": true}); } } diff --git a/alcs-frontend/src/app/services/tag/tag-category/tag-category.service.ts b/alcs-frontend/src/app/services/tag/tag-category/tag-category.service.ts index 43449f1fc9..c9eab99cbd 100644 --- a/alcs-frontend/src/app/services/tag/tag-category/tag-category.service.ts +++ b/alcs-frontend/src/app/services/tag/tag-category/tag-category.service.ts @@ -50,8 +50,8 @@ export class TagCategoryService { return await firstValueFrom(this.http.post(`${this.url}`, createDto)); } catch (e) { const res = e as HttpErrorResponse; - if (res.error.statusCode === HttpStatusCode.Conflict && res.error.message.includes('duplicate key')) { - throw e as HttpErrorResponse; + if (res.error.statusCode === HttpStatusCode.Conflict) { + throw res; } else { console.error(e); this.toastService.showErrorToast('Failed to create tag category'); @@ -65,8 +65,8 @@ export class TagCategoryService { return await firstValueFrom(this.http.patch(`${this.url}/${uuid}`, updateDto)); } catch (e) { const res = e as HttpErrorResponse; - if (res.error.statusCode === HttpStatusCode.Conflict && res.error.message.includes('duplicate key')) { - throw e as HttpErrorResponse; + if (res.error.statusCode === HttpStatusCode.Conflict) { + throw res; } else { console.error(e); this.toastService.showErrorToast('Failed to update tag category'); diff --git a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.ts b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.ts index 263097760a..cf9a887f2f 100644 --- a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.ts +++ b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.ts @@ -1,16 +1,4 @@ -import { - Body, - Controller, - Delete, - Get, - HttpException, - HttpStatus, - Param, - Patch, - Post, - Query, - UseGuards, -} from '@nestjs/common'; +import { Body, Controller, Delete, Get, Param, Patch, Post, Query, UseGuards } from '@nestjs/common'; import { ApiOAuth2 } from '@nestjs/swagger'; import * as config from 'config'; import { RolesGuard } from '../../../common/authorization/roles-guard.service'; @@ -18,7 +6,6 @@ import { UserRoles } from '../../../common/authorization/roles.decorator'; import { AUTH_ROLE } from '../../../common/authorization/roles'; import { TagCategoryDto } from './tag-category.dto'; import { TagCategoryService } from './tag-category.service'; -import { QueryFailedError } from 'typeorm'; @Controller('tag-category') @ApiOAuth2(config.get('KEYCLOAK.SCOPES')) @@ -40,45 +27,18 @@ export class TagCategoryController { @Post('') @UserRoles(AUTH_ROLE.ADMIN) async create(@Body() createDto: TagCategoryDto) { - try { - return await this.service.create(createDto); - } catch (e) { - if (e.constructor === QueryFailedError) { - const msg = (e as QueryFailedError).message; - throw new HttpException(msg, HttpStatus.CONFLICT); - } else { - throw e; - } - } + return await this.service.create(createDto); } @Patch('/:uuid') @UserRoles(AUTH_ROLE.ADMIN) async update(@Param('uuid') uuid: string, @Body() updateDto: TagCategoryDto) { - try { - return await this.service.update(uuid, updateDto); - } catch (e) { - if (e.constructor === QueryFailedError) { - const msg = (e as QueryFailedError).message; - throw new HttpException(msg, HttpStatus.CONFLICT); - } else { - throw e; - } - } + return await this.service.update(uuid, updateDto); } @Delete('/:uuid') @UserRoles(AUTH_ROLE.ADMIN) async delete(@Param('uuid') uuid: string) { - try { - return await this.service.delete(uuid); - } catch (e) { - if (e.constructor === QueryFailedError) { - const msg = (e as QueryFailedError).message; - throw new HttpException(msg, HttpStatus.CONFLICT); - } else { - throw e; - } - } + return await this.service.delete(uuid); } } diff --git a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.entity.ts b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.entity.ts index cd50e163ef..f866e2a80d 100644 --- a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.entity.ts +++ b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.entity.ts @@ -16,6 +16,6 @@ export class TagCategory extends Base { uuid: string; @AutoMap() - @Column({ unique: true }) + @Column() name: string; } diff --git a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.ts b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.ts index 7f977ff785..37aaa79d83 100644 --- a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.ts +++ b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.ts @@ -35,6 +35,9 @@ export class TagCategoryService { } async create(dto: TagCategoryDto) { + if (await this.hasName(dto)) { + throw new ServiceConflictException('There is already a category with this name. Unable to create.'); + } const newTagCategory = new TagCategory(); newTagCategory.name = dto.name; return this.repository.save(newTagCategory); @@ -47,6 +50,10 @@ export class TagCategoryService { } async update(uuid: string, updateDto: TagCategoryDto) { + updateDto.uuid = uuid; + if (await this.hasName(updateDto)) { + throw new ServiceConflictException('There is already a category with this name. Unable to update.'); + } const tagCategory = await this.getOneOrFail(uuid); tagCategory.name = updateDto.name; return await this.repository.save(tagCategory); @@ -58,7 +65,7 @@ export class TagCategoryService { if (await this.isAssociated(tagCategory)) { throw new ServiceConflictException('Category is associated with tags. Unable to delete.'); } - return await this.repository.remove(tagCategory); + return await this.repository.softDelete(uuid); } async isAssociated(tagCategory: TagCategory) { @@ -66,4 +73,14 @@ export class TagCategoryService { return associatedTags.length > 0; } + + async hasName(tag: TagCategoryDto) { + let tags = await this.repository.find({ + where: { name: tag.name }, + }); + if (tag.uuid) { + tags = tags.filter((t) => t.uuid !== tag.uuid); + } + return tags.length > 0; + } } diff --git a/services/apps/alcs/src/alcs/tag/tag.controller.ts b/services/apps/alcs/src/alcs/tag/tag.controller.ts index 27cc01ff7e..a29446a9bf 100644 --- a/services/apps/alcs/src/alcs/tag/tag.controller.ts +++ b/services/apps/alcs/src/alcs/tag/tag.controller.ts @@ -1,16 +1,4 @@ -import { - Body, - Controller, - Delete, - Get, - HttpException, - HttpStatus, - Param, - Patch, - Post, - Query, - UseGuards, -} from '@nestjs/common'; +import { Body, Controller, Delete, Get, Param, Patch, Post, Query, UseGuards } from '@nestjs/common'; import { ApiOAuth2 } from '@nestjs/swagger'; import * as config from 'config'; import { RolesGuard } from '../../common/authorization/roles-guard.service'; @@ -18,7 +6,6 @@ import { UserRoles } from '../../common/authorization/roles.decorator'; import { TagService } from './tag.service'; import { AUTH_ROLE } from '../../common/authorization/roles'; import { TagDto } from './tag.dto'; -import { QueryFailedError } from 'typeorm'; @Controller('tag') @ApiOAuth2(config.get('KEYCLOAK.SCOPES')) @@ -40,45 +27,18 @@ export class TagController { @Post('') @UserRoles(AUTH_ROLE.ADMIN) async create(@Body() createDto: TagDto) { - try { - return await this.service.create(createDto); - } catch (e) { - if (e.constructor === QueryFailedError) { - const msg = (e as QueryFailedError).message; - throw new HttpException(msg, HttpStatus.CONFLICT); - } else { - throw e; - } - } + return await this.service.create(createDto); } @Patch('/:uuid') @UserRoles(AUTH_ROLE.ADMIN) async update(@Param('uuid') uuid: string, @Body() updateDto: TagDto) { - try { - return await this.service.update(uuid, updateDto); - } catch (e) { - if (e.constructor === QueryFailedError) { - const msg = (e as QueryFailedError).message; - throw new HttpException(msg, HttpStatus.CONFLICT); - } else { - throw e; - } - } + return await this.service.update(uuid, updateDto); } @Delete('/:uuid') @UserRoles(AUTH_ROLE.ADMIN) async delete(@Param('uuid') uuid: string) { - try { - return await this.service.delete(uuid); - } catch (e) { - if (e.constructor === QueryFailedError) { - const msg = (e as QueryFailedError).message; - throw new HttpException(msg, HttpStatus.CONFLICT); - } else { - throw e; - } - } + return await this.service.delete(uuid); } } diff --git a/services/apps/alcs/src/alcs/tag/tag.dto.ts b/services/apps/alcs/src/alcs/tag/tag.dto.ts index 8088d8dc69..75ce8d3423 100644 --- a/services/apps/alcs/src/alcs/tag/tag.dto.ts +++ b/services/apps/alcs/src/alcs/tag/tag.dto.ts @@ -2,6 +2,9 @@ import { IsBoolean, IsObject, IsOptional, IsString } from 'class-validator'; import { TagCategoryDto } from './tag-category/tag-category.dto'; export class TagDto { + @IsString() + uuid: string; + @IsString() name: string; diff --git a/services/apps/alcs/src/alcs/tag/tag.entity.ts b/services/apps/alcs/src/alcs/tag/tag.entity.ts index 80f1d2620c..071901c6a3 100644 --- a/services/apps/alcs/src/alcs/tag/tag.entity.ts +++ b/services/apps/alcs/src/alcs/tag/tag.entity.ts @@ -19,7 +19,7 @@ export class Tag extends Base { uuid: string; @AutoMap() - @Column({ unique: true }) + @Column() name: string; @AutoMap() @@ -33,7 +33,7 @@ export class Tag extends Base { @ManyToMany(() => Application, (application) => application.tags) applications: Application[]; - + @ManyToMany(() => NoticeOfIntent, (noticeOfIntent) => noticeOfIntent.tags) noticeOfIntents: NoticeOfIntent[]; } diff --git a/services/apps/alcs/src/alcs/tag/tag.service.ts b/services/apps/alcs/src/alcs/tag/tag.service.ts index ff3f13c2cb..5563c58789 100644 --- a/services/apps/alcs/src/alcs/tag/tag.service.ts +++ b/services/apps/alcs/src/alcs/tag/tag.service.ts @@ -38,6 +38,9 @@ export class TagService { } async create(dto: TagDto) { + if (await this.hasName(dto)) { + throw new ServiceConflictException('There is already a tag with this name. Unable to create.'); + } const category = dto.category ? await this.categoryRepository.findOne({ where: { @@ -60,6 +63,10 @@ export class TagService { } async update(uuid: string, updateDto: TagDto) { + updateDto.uuid = uuid; + if (await this.hasName(updateDto)) { + throw new ServiceConflictException('There is already a tag with this name. Unable to update.'); + } const category = updateDto.category ? await this.categoryRepository.findOne({ where: { @@ -93,4 +100,14 @@ export class TagService { return (tag.applications && tag.applications.length > 0) || (tag.noticeOfIntents && tag.noticeOfIntents.length > 0); } + + async hasName(tag: TagDto) { + let tags = await this.repository.find({ + where: { name: tag.name }, + }); + if (tag.uuid) { + tags = tags.filter((t) => t.uuid !== tag.uuid); + } + return tags.length > 0; + } } diff --git a/services/apps/alcs/src/providers/typeorm/migrations/1730856145155-remove_unique_constraints.ts b/services/apps/alcs/src/providers/typeorm/migrations/1730856145155-remove_unique_constraints.ts new file mode 100644 index 0000000000..5a0326f70c --- /dev/null +++ b/services/apps/alcs/src/providers/typeorm/migrations/1730856145155-remove_unique_constraints.ts @@ -0,0 +1,16 @@ +import { MigrationInterface, QueryRunner } from "typeorm"; + +export class RemoveUniqueConstraints1730856145155 implements MigrationInterface { + name = 'RemoveUniqueConstraints1730856145155' + + public async up(queryRunner: QueryRunner): Promise { + await queryRunner.query(`ALTER TABLE "alcs"."tag_category" DROP CONSTRAINT "UQ_f48a9fe1f705a7c2a60856d395a"`); + await queryRunner.query(`ALTER TABLE "alcs"."tag" DROP CONSTRAINT "UQ_6a9775008add570dc3e5a0bab7b"`); + } + + public async down(queryRunner: QueryRunner): Promise { + await queryRunner.query(`ALTER TABLE "alcs"."tag" ADD CONSTRAINT "UQ_6a9775008add570dc3e5a0bab7b" UNIQUE ("name")`); + await queryRunner.query(`ALTER TABLE "alcs"."tag_category" ADD CONSTRAINT "UQ_f48a9fe1f705a7c2a60856d395a" UNIQUE ("name")`); + } + +}