diff --git a/src/app/product-store/app-delete/app-delete.component.spec.ts b/src/app/product-store/app-delete/app-delete.component.spec.ts index a92556d..c17bbec 100644 --- a/src/app/product-store/app-delete/app-delete.component.spec.ts +++ b/src/app/product-store/app-delete/app-delete.component.spec.ts @@ -78,7 +78,7 @@ describe('AppDeleteComponent', () => { expect(component.appDeleted.emit).toHaveBeenCalledWith(true) }) - it('should display error if api all fails onConfirmDeletion: mfe', () => { + it('should display error if api call fails onConfirmDeletion: mfe', () => { apiMfeServiceSpy.deleteMicrofrontend.and.returnValue(throwError(() => new Error())) component.appAbstract = appMfe @@ -104,7 +104,7 @@ describe('AppDeleteComponent', () => { expect(msgServiceSpy.success).toHaveBeenCalledWith({ summaryKey: 'ACTIONS.DELETE.APP.OK' }) }) - xit('should display error if api all fails onConfirmDeletion: mfe', () => { + xit('should display error if api call fails onConfirmDeletion: ms', () => { apiMfeServiceSpy.deleteMicroservice.and.returnValue(throwError(() => new Error())) const appMs: AppAbstract = { id: 'id', diff --git a/src/app/product-store/app-delete/app-delete.component.ts b/src/app/product-store/app-delete/app-delete.component.ts index 6427195..bc74c29 100644 --- a/src/app/product-store/app-delete/app-delete.component.ts +++ b/src/app/product-store/app-delete/app-delete.component.ts @@ -27,7 +27,7 @@ export class AppDeleteComponent { public onConfirmDeletion(): void { if (this.appAbstract?.id) { - if (this.appAbstract?.appType === 'MFE') + if (this.appAbstract?.appType === 'MFE') { this.mfeApi.deleteMicrofrontend({ id: this.appAbstract?.id }).subscribe({ next: () => { this.msgService.success({ summaryKey: 'ACTIONS.DELETE.APP.OK' }) @@ -35,7 +35,8 @@ export class AppDeleteComponent { }, error: () => this.msgService.error({ summaryKey: 'ACTIONS.DELETE.APP.NOK' }) }) - if (this.appAbstract?.appType === 'MS') + } + if (this.appAbstract?.appType === 'MS') { this.msApi.deleteMicroservice({ id: this.appAbstract?.id }).subscribe({ next: () => { this.msgService.success({ summaryKey: 'ACTIONS.DELETE.APP.OK' }) @@ -43,6 +44,7 @@ export class AppDeleteComponent { }, error: () => this.msgService.error({ summaryKey: 'ACTIONS.DELETE.APP.NOK' }) }) + } } } } diff --git a/src/app/product-store/app-detail/app-detail.component.ts b/src/app/product-store/app-detail/app-detail.component.ts index ec1465b..41ccbd8 100644 --- a/src/app/product-store/app-detail/app-detail.component.ts +++ b/src/app/product-store/app-detail/app-detail.component.ts @@ -120,12 +120,12 @@ export class AppDetailComponent implements OnChanges { if (this.appAbstract.appType === 'MS') this.getMs() } } - private log(text: string, obj?: object): void { - if (this.debug) { - if (obj) console.log('app detail: ' + text, obj) - else console.log('app detail: ' + text) - } - } + // private log(text: string, obj?: object): void { + // if (this.debug) { + // if (obj) console.log('app detail: ' + text, obj) + // else console.log('app detail: ' + text) + // } + // } public allowEditing(): boolean { return ( diff --git a/src/app/product-store/app-search/app-search.component.spec.ts b/src/app/product-store/app-search/app-search.component.spec.ts index 617974b..2b82a05 100644 --- a/src/app/product-store/app-search/app-search.component.spec.ts +++ b/src/app/product-store/app-search/app-search.component.spec.ts @@ -172,7 +172,51 @@ describe('AppSearchComponent', () => { expect(component.searchApps).toHaveBeenCalled() }) + it('should search mfes: one mfe', (done) => { + component.appSearchCriteriaGroup.controls['appType'].setValue('MFE') + component.mfes$ = of({ + stream: [ + { + id: 'mfe1', + appId: 'appId1', + appName: 'Microfrontend 1', + productName: 'p1', + remoteBaseUrl: 'url' + } + ] + }) + + component.searchApps() + + component.apps$.subscribe({ + next: (apps) => { + expect(apps.length).toBe(1) + apps.forEach((app) => { + expect(app.appType).toEqual('MFE') + }) + done() + }, + error: done.fail + }) + }) + + it('should search mfes: empty', (done) => { + component.appSearchCriteriaGroup.controls['appType'].setValue('MFE') + component.mfes$ = of({}) + + component.searchApps() + + component.apps$.subscribe({ + next: (apps) => { + expect(apps.length).toBe(0) + done() + }, + error: done.fail + }) + }) + it('should catch error on searchApps: mfes', () => { + component.appSearchCriteriaGroup.controls['appType'].setValue('MFE') const err = { status: 404 } @@ -182,8 +226,45 @@ describe('AppSearchComponent', () => { expect(component.exceptionKey).toEqual('') }) + // expect(component.exceptionKey).toEqual('EXCEPTIONS.HTTP_STATUS_404.APPS') + + it('should search mss: one ms', (done) => { + component.appSearchCriteriaGroup.controls['appType'].setValue('MS') + component.mss$ = of({ + stream: [{ id: 'ms1', appId: 'appId3', appName: 'Microservice 1', productName: 'p1' }] + }) + + component.searchApps() + + component.apps$.subscribe({ + next: (apps) => { + expect(apps.length).toBe(1) + apps.forEach((app) => { + expect(app.appType).toEqual('MS') + }) + done() + }, + error: done.fail + }) + }) + + it('should search mss: empty', (done) => { + component.appSearchCriteriaGroup.controls['appType'].setValue('MS') + component.mss$ = of({}) + + component.searchApps() + + component.apps$.subscribe({ + next: (apps) => { + expect(apps.length).toBe(0) + done() + }, + error: done.fail + }) + }) it('should catch error on searchApps: mss', () => { + component.appSearchCriteriaGroup.controls['appType'].setValue('MS') const err = { status: 404 } @@ -194,37 +275,28 @@ describe('AppSearchComponent', () => { expect(component.exceptionKey).toEqual('EXCEPTIONS.HTTP_STATUS_404.APPS') }) - it('should combine mfe and ms streams into apps$ with appType: only mfes', (done: DoneFn) => { - apiServiceSpy.searchMicrofrontends.and.returnValue( - of({ - stream: [ - { id: 'mfe1', name: 'Microfrontend 1', productName: 'p1' }, - { id: 'mfe2', name: 'Microfrontend 2', productName: 'p1' } - ] - }) - ) - apiServiceSpy.searchMicroservice.and.returnValue( - of({ - stream: [ - { id: 'ms1', name: 'Microservice 1', productName: 'p1' }, - { id: 'ms2', name: 'Microservice 2', productName: 'p1' } - ] - }) - ) + it('should combine mfe and ms streams into apps$ with appType', (done: DoneFn) => { + component.appSearchCriteriaGroup.controls['appType'].setValue('ALL') + component.mfes$ = of({ + stream: [ + { + id: 'mfe1', + appId: 'appId1', + appName: 'Microfrontend 1', + productName: 'p1', + remoteBaseUrl: 'url' + } + ] + }) + component.mss$ = of({ + stream: [{ id: 'ms1', appId: 'appId3', appName: 'Microservice 1', productName: 'p1' }] + }) component.searchApps() component.apps$.subscribe({ next: (result) => { - expect(result.length).toBe(0) // should be 4 - /* expect(result).toEqual( - jasmine.arrayContaining([ - jasmine.objectContaining({ id: 'mfe1', name: 'Microfrontend 1', appType: 'MFE' }), - jasmine.objectContaining({ id: 'mfe2', name: 'Microfrontend 2', appType: 'MFE' }) - // jasmine.objectContaining({ id: 'ms1', name: 'Microservice 1', appType: 'MS' }), - // jasmine.objectContaining({ id: 'ms2', name: 'Microservice 2', appType: 'MS' }) - ]) - ) */ + expect(result.length).toBe(2) done() }, error: done.fail diff --git a/src/app/product-store/product-detail/product-apps/product-apps.component.spec.ts b/src/app/product-store/product-detail/product-apps/product-apps.component.spec.ts index 532286e..d4aca88 100644 --- a/src/app/product-store/product-detail/product-apps/product-apps.component.spec.ts +++ b/src/app/product-store/product-detail/product-apps/product-apps.component.spec.ts @@ -1,8 +1,8 @@ import { NO_ERRORS_SCHEMA } from '@angular/core' -import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing' +import { ComponentFixture, TestBed, fakeAsync, waitForAsync } from '@angular/core/testing' import { HttpClientTestingModule } from '@angular/common/http/testing' import { RouterTestingModule } from '@angular/router/testing' -import { of } from 'rxjs' +import { of, throwError } from 'rxjs' import { TranslateTestingModule } from 'ngx-translate-testing' import { PortalMessageService } from '@onecx/portal-integration-angular' @@ -101,6 +101,63 @@ describe('ProductAppsComponent', () => { }) }) + xit('should display console error msg on searchApps', fakeAsync((done: DoneFn) => { + const searchSpy = spyOn((component as any).mfeApi, 'searchMicrofrontends').and.returnValue( + throwError(() => new Error()) + ) + // apiServiceSpy.searchMicrofrontends.and.callFake(() => { + // throw error + // }) + spyOn(console, 'error') + + component.searchApps() + + component.mfes$.subscribe({ + next: (obj) => { + console.log('NEXT', obj) + done() + }, + error: (err) => { + console.log('ERROR', err) + // expect(console.error).toHaveBeenCalled() + } + }) + + expect(searchSpy).toHaveBeenCalledWith({ + mfeAndMsSearchCriteria: { productName: component.product?.name } + }) + expect(console.error).toHaveBeenCalled() + + // tick() + })) + + xit('should combine mfe and ms streams into apps$ with appType', (done: DoneFn) => { + component.mfes$ = of({ + stream: [ + { + id: 'mfe1', + appId: 'appId1', + appName: 'Microfrontend 1', + productName: 'p1', + remoteBaseUrl: 'url' + } + ] + }) + component.mss$ = of({ + stream: [{ id: 'ms1', appId: 'appId3', appName: 'Microservice 1', productName: 'p1' }] + }) + + component.searchApps() + + component.apps$.subscribe({ + next: (result) => { + expect(result.length).toBe(2) + done() + }, + error: done.fail + }) + }) + it('should set correct value onLayoutChange', () => { const viewMode = 'EDIT' @@ -171,4 +228,22 @@ describe('ProductAppsComponent', () => { expect(component.app).toEqual(mockApp) expect(component.displayDeleteDialog).toBeTrue() }) + + it('should call searchApps if app changed', () => { + spyOn(component, 'searchApps') + + component.appChanged(true) + + expect(component.searchApps).toHaveBeenCalled() + expect(component.displayDetailDialog).toBeFalse() + }) + + it('should call searchApps if app deleted', () => { + spyOn(component, 'searchApps') + + component.appDeleted(true) + + expect(component.searchApps).toHaveBeenCalled() + expect(component.displayDetailDialog).toBeFalse() + }) }) diff --git a/src/app/product-store/product-detail/product-apps/product-apps.component.ts b/src/app/product-store/product-detail/product-apps/product-apps.component.ts index f8c1c56..489d7d9 100644 --- a/src/app/product-store/product-detail/product-apps/product-apps.component.ts +++ b/src/app/product-store/product-detail/product-apps/product-apps.component.ts @@ -64,12 +64,12 @@ export class ProductAppsComponent implements OnChanges { if (this.product) this.searchApps() } - private log(text: string, obj?: object): void { - if (this.debug) { - if (obj) console.log('app search: ' + text, obj) - else console.log('app search: ' + text) - } - } + // private log(text: string, obj?: object): void { + // if (this.debug) { + // if (obj) console.log('app search: ' + text, obj) + // else console.log('app search: ' + text) + // } + // } public searchApps(): void { this.searchInProgress = true diff --git a/src/app/product-store/product-detail/product-props/product-props.component.spec.ts b/src/app/product-store/product-detail/product-props/product-props.component.spec.ts index 381641e..46d3e67 100644 --- a/src/app/product-store/product-detail/product-props/product-props.component.spec.ts +++ b/src/app/product-store/product-detail/product-props/product-props.component.spec.ts @@ -1,5 +1,5 @@ import { NO_ERRORS_SCHEMA } from '@angular/core' -import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing' +import { ComponentFixture, TestBed, fakeAsync, tick, waitForAsync } from '@angular/core/testing' import { HttpClientTestingModule } from '@angular/common/http/testing' import { RouterTestingModule } from '@angular/router/testing' import { of, throwError } from 'rxjs' @@ -7,8 +7,30 @@ import { FormControl, FormGroup, Validators } from '@angular/forms' import { TranslateTestingModule } from 'ngx-translate-testing' import { PortalMessageService } from '@onecx/portal-integration-angular' -import { ProductPropertyComponent, ProductDetailForm } from './product-props.component' -import { ProductsAPIService } from 'src/app/shared/generated' +import { ProductPropertyComponent, ProductDetailForm, productNameValidator } from './product-props.component' +import { ProductsAPIService, ImagesInternalAPIService } from 'src/app/shared/generated' + +const mockForm = new FormGroup({ + id: new FormControl(null), // Assuming ID might not be needed or can be null for new entries + name: new FormControl(null, [ + Validators.required, + Validators.minLength(2), + Validators.maxLength(255), + productNameValidator() + ]), + operator: new FormControl(null), + version: new FormControl(null, [Validators.required, Validators.maxLength(255)]), + description: new FormControl(null, [Validators.maxLength(255)]), + imageUrl: new FormControl(null, [Validators.maxLength(255)]), + basePath: new FormControl(null, [Validators.required, Validators.maxLength(255)]), + displayName: new FormControl(null, [ + Validators.required, + Validators.minLength(2), + Validators.maxLength(255) + ]), + iconName: new FormControl(null, [Validators.maxLength(255)]), + classifications: new FormControl(null, [Validators.maxLength(255)]) // Assuming this validation makes sense for your use case +}) describe('ProductPropertyComponent', () => { let component: ProductPropertyComponent @@ -19,6 +41,14 @@ describe('ProductPropertyComponent', () => { updateProduct: jasmine.createSpy('updateProduct').and.returnValue(of({})) } const msgServiceSpy = jasmine.createSpyObj('PortalMessageService', ['success', 'error', 'info']) + const imgServiceSpy = { + getImage: jasmine.createSpy('getImage').and.returnValue(of({})), + updateImage: jasmine.createSpy('updateImage').and.returnValue(of({})), + uploadImage: jasmine.createSpy('uploadImage').and.returnValue(of({})), + configuration: { + basePath: 'basepath' + } + } beforeEach(waitForAsync(() => { TestBed.configureTestingModule({ @@ -33,7 +63,8 @@ describe('ProductPropertyComponent', () => { ], providers: [ { provide: ProductsAPIService, useValue: apiServiceSpy }, - { provide: PortalMessageService, useValue: msgServiceSpy } + { provide: PortalMessageService, useValue: msgServiceSpy }, + { provide: ImagesInternalAPIService, useValue: imgServiceSpy } ], schemas: [NO_ERRORS_SCHEMA] }).compileComponents() @@ -51,12 +82,38 @@ describe('ProductPropertyComponent', () => { msgServiceSpy.info.calls.reset() apiServiceSpy.createProduct.calls.reset() apiServiceSpy.updateProduct.calls.reset() + imgServiceSpy.getImage.calls.reset() + imgServiceSpy.uploadImage.calls.reset() + imgServiceSpy.updateImage.calls.reset() }) it('should create', () => { expect(component).toBeTruthy() }) + it('should validate a product name', () => { + const control = new FormControl('', [productNameValidator()]) + + control.setValue('new') + expect(control.errors).toEqual({ invalidProductName: true }) + + control.setValue('apps') + expect(control.errors).toEqual({ invalidProductName: true }) + + control.setValue('validName') + expect(control.errors).toBeNull() + }) + + it('should getImage onInit', () => { + imgServiceSpy.getImage.and.returnValue(of({})) + component.formGroup = mockForm + component.formGroup.controls['name'].setValue('name') + + component.ngOnInit() + + expect(component.logoImageWasUploaded).toBeTrue() + }) + it('should patchValue in formGroup onChanges if product', () => { const product = { id: 'id', @@ -72,6 +129,21 @@ describe('ProductPropertyComponent', () => { expect(component.product.name).toEqual(product.name) }) + it('should set product.id to undefined onChanges if product and changeMode is COPY', () => { + const product = { + id: 'id', + name: 'name', + basePath: 'path' + } + component.product = product + spyOn(component.formGroup, 'patchValue') + component.changeMode = 'COPY' + + component.ngOnChanges() + + expect(component.productId).toBeUndefined() + }) + it('should reset formGroup onChanges if no product', () => { spyOn(component.formGroup, 'reset') @@ -141,6 +213,7 @@ describe('ProductPropertyComponent', () => { classifications: new FormControl(null) }) component.formGroup = formGroup as FormGroup + component.formGroup.controls['name'].setValue('') component.changeMode = 'EDIT' component.onSubmit() @@ -151,6 +224,51 @@ describe('ProductPropertyComponent', () => { }) }) + it('should display unique constraint error if error code points to it', () => { + const error = { + error: { + errorCode: 'PERSIST_ENTITY_FAILED' + } + } + apiServiceSpy.updateProduct.and.returnValue(throwError(() => error)) + const formGroup = new FormGroup({ + id: new FormControl('id'), + name: new FormControl('name'), + operator: new FormControl(null), + version: new FormControl('version'), + description: new FormControl(null), + imageUrl: new FormControl(null), + basePath: new FormControl('path'), + displayName: new FormControl('display'), + iconName: new FormControl('icon'), + classifications: new FormControl(null) + }) + component.formGroup = formGroup as FormGroup + component.formGroup.controls['name'].setValue('') + component.changeMode = 'EDIT' + + component.onSubmit() + + expect(component.formGroup.valid).toBeTrue() + expect(msgServiceSpy.error).toHaveBeenCalledWith({ + summaryKey: 'ACTIONS.EDIT.PRODUCT.NOK', + detailKey: 'VALIDATION.PRODUCT.UNIQUE_CONSTRAINT' + }) + }) + + it('should display error and set focus to first invalid field if form is invalid', () => { + component.formGroup.controls['name'].setValue('') + + const focusSpy = jasmine.createSpy('focus') + spyOn((component as any).elements.nativeElement, 'querySelector').and.returnValue({ focus: focusSpy }) + + component.onSubmit() + + expect(component.formGroup.valid).toBeFalse() + expect(msgServiceSpy.error).toHaveBeenCalledWith({ summaryKey: 'VALIDATION.FORM_INVALID' }) + expect(focusSpy).toHaveBeenCalled() + }) + it('should display error if createProduct fails', () => { apiServiceSpy.createProduct.and.returnValue(throwError(() => new Error())) const formGroup = new FormGroup({ @@ -199,15 +317,132 @@ describe('ProductPropertyComponent', () => { }) }) - it('should display error onSubmit if formGroup invalid', () => { + it('should not upload a file if productName is empty', () => { + const event = { + target: { + files: ['file'] + } + } + component.formGroup.controls['name'].setValue('') + + component.onFileUpload(event as any, 'logo') + + expect(msgServiceSpy.error).toHaveBeenCalledWith({ + summaryKey: 'LOGO.UPLOAD_FAILED_NAME' + }) + }) + + it('should not upload a file if productName is null', () => { const event = { target: { files: ['file'] } } + component.formGroup.controls['name'].setValue(null) + + component.onFileUpload(event as any, 'logo') + + expect(msgServiceSpy.error).toHaveBeenCalledWith({ + summaryKey: 'LOGO.UPLOAD_FAILED_NAME' + }) + }) + + it('should not upload a file that is too large', () => { + const largeBlob = new Blob(['a'.repeat(120000)], { type: 'image/png' }) + const largeFile = new File([largeBlob], 'test.png', { type: 'image/png' }) + const event = { + target: { + files: [largeFile] + } + } + component.formGroup.controls['name'].setValue('name') + + component.onFileUpload(event as any, 'logo') + + expect(component.formGroup.valid).toBeFalse() + }) + + it('should not upload a file that is too large', () => { + const largeBlob = new Blob(['a'.repeat(120000)], { type: 'image/png' }) + const largeFile = new File([largeBlob], 'test.png', { type: 'image/png' }) + const event = { + target: { + files: [largeFile] + } + } + component.formGroup.controls['name'].setValue('name') component.onFileUpload(event as any, 'logo') expect(component.formGroup.valid).toBeFalse() }) + + it('should upload a file', () => { + imgServiceSpy.updateImage.and.returnValue(of({})) + const blob = new Blob(['a'.repeat(10)], { type: 'image/png' }) + const file = new File([blob], 'test.png', { type: 'image/png' }) + const event = { + target: { + files: [file] + } + } + component.formGroup.controls['name'].setValue('name') + + component.onFileUpload(event as any, 'logo') + + expect(msgServiceSpy.info).toHaveBeenCalledWith({ + summaryKey: 'LOGO.UPLOADED' + }) + }) + + it('should display error if upload fails', () => { + imgServiceSpy.getImage.and.returnValue(throwError(() => new Error())) + const blob = new Blob(['a'.repeat(10)], { type: 'image/png' }) + const file = new File([blob], 'test.png', { type: 'image/png' }) + const event = { + target: { + files: [file] + } + } + component.formGroup.controls['name'].setValue('name') + + component.onFileUpload(event as any, 'logo') + + expect(msgServiceSpy.info).toHaveBeenCalledWith({ + summaryKey: 'LOGO.UPLOADED' + }) + }) + + it('should return an image url', () => { + component.formGroup.controls['imageUrl'].setValue('url') + + const result = component.getImageUrl() + + expect(result).toEqual('url') + }) + + it('should change fetchingLogoUrl on inputChange: valid value', fakeAsync(() => { + const event = { + target: { value: 'newLogoValue' } + } as unknown as Event + + component.inputChange(event) + + tick(1000) + + expect(component.fetchingLogoUrl).toBe('newLogoValue') + })) + + it('should change fetchingLogoUrl on inputChange: empty value', fakeAsync(() => { + const event = { + target: { value: '' } + } as unknown as Event + component.formGroup.controls['name'].setValue('name') + + component.inputChange(event) + + tick(1000) + + expect(component.fetchingLogoUrl).toBe('basepath/images/name/logo') + })) }) diff --git a/src/app/product-store/product-search/product-search.component.spec.ts b/src/app/product-store/product-search/product-search.component.spec.ts index 67d7440..b705aee 100644 --- a/src/app/product-store/product-search/product-search.component.spec.ts +++ b/src/app/product-store/product-search/product-search.component.spec.ts @@ -125,4 +125,24 @@ describe('ProductSearchComponent', () => { expect(routerSpy).toHaveBeenCalledWith(['./apps'], jasmine.any(Object)) }) + + it('should getImageUrl from existing product', () => { + const product = { + imageUrl: 'url' + } + + const result = component.getImageUrl(product) + + expect(result).toEqual(product.imageUrl) + }) + + it('should getImageUrl from image api if not from existing product', () => { + const product = { + id: 'id' + } + + const result = component.getImageUrl(product) + + expect(result).toEqual('http://onecx-product-store-bff:8080/images/undefined/logo') + }) })