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.spec.ts b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.spec.ts index 6025f707aa..c0910e17cc 100644 --- a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.spec.ts +++ b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.controller.spec.ts @@ -7,6 +7,7 @@ import { initTagCategoryMockEntity } from '../../../../test/mocks/mockEntities'; import { mockKeyCloakProviders } from '../../../../test/mocks/mockTypes'; import { TagCategory } from './tag-category.entity'; import { TagCategoryDto } from './tag-category.dto'; +import { UpdateResult } from 'typeorm'; describe('TagCategoryController', () => { let controller: TagCategoryController; @@ -62,20 +63,17 @@ describe('TagCategoryController', () => { name: mockCategoryTag.name, } as TagCategoryDto; - const result = await controller.update( - mockCategoryTag.uuid, - categoryToUpdate, - ); + const result = await controller.update(mockCategoryTag.uuid, categoryToUpdate); expect(tagCategoryService.update).toHaveBeenCalledTimes(1); expect(result).toEqual(mockCategoryTag); }); it('should delete a tag category', async () => { - tagCategoryService.delete.mockResolvedValue(mockCategoryTag); + tagCategoryService.delete.mockResolvedValue({} as UpdateResult); const result = await controller.delete(mockCategoryTag.uuid); expect(tagCategoryService.delete).toHaveBeenCalledTimes(1); - expect(result).toEqual(mockCategoryTag); + expect(result).toBeDefined(); }); }); 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.spec.ts b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.spec.ts index 9616a274e6..fb0b496d26 100644 --- a/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.spec.ts +++ b/services/apps/alcs/src/alcs/tag/tag-category/tag-category.service.spec.ts @@ -48,6 +48,7 @@ describe('TagCategoryService', () => { }).compile(); service = module.get(TagCategoryService); + tagCategoryRepositoryMock.find.mockResolvedValue([]); tagCategoryRepositoryMock.findOne.mockResolvedValue(mockTagCategoryEntity); tagCategoryRepositoryMock.findOneOrFail.mockResolvedValue(mockTagCategoryEntity); tagCategoryRepositoryMock.save.mockResolvedValue(mockTagCategoryEntity); 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.spec.ts b/services/apps/alcs/src/alcs/tag/tag.controller.spec.ts index 8042769a45..7191b00adb 100644 --- a/services/apps/alcs/src/alcs/tag/tag.controller.spec.ts +++ b/services/apps/alcs/src/alcs/tag/tag.controller.spec.ts @@ -48,6 +48,7 @@ describe('TagController', () => { it('should create a tag', async () => { const dto: TagDto = { + uuid: mockTag.uuid, name: mockTag.name, category: mockTag.category ? { 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.spec.ts b/services/apps/alcs/src/alcs/tag/tag.service.spec.ts index b853208c7f..60a49469e7 100644 --- a/services/apps/alcs/src/alcs/tag/tag.service.spec.ts +++ b/services/apps/alcs/src/alcs/tag/tag.service.spec.ts @@ -52,6 +52,7 @@ describe('TagCategoryService', () => { tagRepositoryMock.findOneOrFail.mockResolvedValue(mockTagEntity); tagRepositoryMock.findOne.mockResolvedValue(mockTagEntity); + tagRepositoryMock.find.mockResolvedValue([]); tagRepositoryMock.save.mockResolvedValue(mockTagEntity); tagCategoryRepositoryMock.findOne.mockResolvedValue(mockTagCategoryEntity); tagCategoryRepositoryMock = module.get(getRepositoryToken(TagCategory)); @@ -67,6 +68,7 @@ describe('TagCategoryService', () => { it('should call save when an Tag is updated', async () => { const payload: TagDto = { + uuid: mockTagEntity.uuid, name: mockTagEntity.name, isActive: mockTagEntity.isActive, category: mockTagCategoryEntity, @@ -80,6 +82,7 @@ describe('TagCategoryService', () => { it('should call save when tag successfully create', async () => { const payload: TagDto = { + uuid: mockTagEntity.uuid, name: mockTagEntity.name, isActive: mockTagEntity.isActive, category: mockTagCategoryEntity, 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")`); + } + +}