From cb0881a4c952a89a3e85717c34a1666a2fcc4839 Mon Sep 17 00:00:00 2001 From: Enno Gotthold Date: Sat, 20 Jul 2024 12:17:27 +0200 Subject: [PATCH 1/2] Cobbler-API: Enable test for "background_buildiso" --- .../src/lib/cobbler-api.service.spec.ts | 29 ++++++++++++++++--- 1 file changed, 25 insertions(+), 4 deletions(-) diff --git a/projects/cobbler-api/src/lib/cobbler-api.service.spec.ts b/projects/cobbler-api/src/lib/cobbler-api.service.spec.ts index 48a656a1..5bdbed4d 100644 --- a/projects/cobbler-api/src/lib/cobbler-api.service.spec.ts +++ b/projects/cobbler-api/src/lib/cobbler-api.service.spec.ts @@ -1,6 +1,6 @@ import {TestBed} from '@angular/core/testing'; import {HttpClientTestingModule, HttpTestingController} from '@angular/common/http/testing'; -import {BackgroundImportOptions, BackgroundReplicateOptions} from 'cobbler-api'; +import {BackgroundBuildisoOptions, BackgroundImportOptions, BackgroundReplicateOptions} from 'cobbler-api'; import {Event, ExtendedVersion, InstallationStatus} from './custom-types/misc'; import {COBBLER_URL} from './lib.config'; import {AngularXmlrpcService} from 'typescript-xmlrpc'; @@ -72,9 +72,30 @@ describe('CobblerApiService', () => { mockRequest.flush(methodResponse); }); - xit('should execute the background_buildiso action on the Cobbler Server', () => { - service.background_buildiso(undefined, ''); - expect(service).toBeFalsy(); + it('should execute the background_buildiso action on the Cobbler Server', (done: DoneFn) => { + // eslint-disable-next-line max-len + const methodResponse = `2023-01-24_083001_Build Iso_20fa7d4256fc4f61a2b9c2237c80fb41` + const result = "2023-01-24_083001_Build Iso_20fa7d4256fc4f61a2b9c2237c80fb41" + const buildisoOptions: BackgroundBuildisoOptions = { + iso: "", + profiles: "", + systems: "", + buildisodir: "", + distro: "", + standalone: false, + airgapped: false, + source: "", + excludeDNS: false, + xorrisofsOpts: "", + } + + service.background_buildiso(buildisoOptions, '').subscribe( + value => { + expect(value).toEqual(result); + done(); + }); + const mockRequest = httpTestingController.expectOne('http://localhost/cobbler_api'); + mockRequest.flush(methodResponse); }); xit('should execute the background_aclsetup action on the Cobbler Server', () => { From b133c64a819c3a799a01bc2b0b2f042e7efb2728 Mon Sep 17 00:00:00 2001 From: Enno Gotthold Date: Sat, 20 Jul 2024 12:17:45 +0200 Subject: [PATCH 2/2] Cobbler-Frontend: Implement UI for "cobbler buildiso" --- .../actions/build-iso/build-iso.component.css | 0 .../build-iso/build-iso.component.html | 49 +++++++++--- .../build-iso/build-iso.component.scss | 9 +++ .../build-iso/build-iso.component.spec.ts | 16 +++- .../actions/build-iso/build-iso.component.ts | 79 +++++++++++++++++-- 5 files changed, 132 insertions(+), 21 deletions(-) delete mode 100644 projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.css create mode 100644 projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.scss diff --git a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.css b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.css deleted file mode 100644 index e69de29b..00000000 diff --git a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.html b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.html index 2f0bf9e5..e6f1e850 100644 --- a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.html +++ b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.html @@ -1,13 +1,36 @@ -
- -
-

BUILD

-
- - data 01 - data 02 - data 03 - -
-
-
+

BUILD ISO IMAGES

+ +
+ + Distro + + + + ISO + + + + Profiles + + + + Systems + + + + Buildiso directory + + + + Source + + + + xorrisofs Options + + + Standalone + Airgapped + Exclude DNS + +
diff --git a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.scss b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.scss new file mode 100644 index 00000000..e6b6c704 --- /dev/null +++ b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.scss @@ -0,0 +1,9 @@ +.form-replicate { + min-width: 150px; + max-width: 600px; + width: 100%; +} + +.form-field-full-width { + width: 100%; +} diff --git a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.spec.ts b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.spec.ts index e592e9fe..5282107a 100644 --- a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.spec.ts +++ b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.spec.ts @@ -1,6 +1,10 @@ +import {provideHttpClient} from '@angular/common/http'; +import {provideHttpClientTesting} from '@angular/common/http/testing'; import { ComponentFixture, TestBed } from '@angular/core/testing'; import { MatListModule } from '@angular/material/list'; +import {NoopAnimationsModule} from '@angular/platform-browser/animations'; import {provideRouter} from '@angular/router'; +import {COBBLER_URL} from 'cobbler-api'; import { BuildISOComponent } from './build-iso.component'; @@ -11,9 +15,17 @@ describe('BuildISOComponent', () => { beforeEach(async () => { await TestBed.configureTestingModule({ - imports: [MatListModule, BuildISOComponent], + imports: [ + BuildISOComponent, + NoopAnimationsModule, + ], providers: [ - provideRouter([]), + provideHttpClient(), + provideHttpClientTesting(), + { + provide: COBBLER_URL, + useValue: new URL('http://localhost/cobbler_api') + }, ] }).compileComponents(); }); diff --git a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.ts b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.ts index b063d016..2a7b6bbf 100644 --- a/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.ts +++ b/projects/cobbler-frontend/src/app/actions/build-iso/build-iso.component.ts @@ -1,14 +1,81 @@ -import { Component } from '@angular/core'; -import { MatListModule } from '@angular/material/list'; -import { RouterOutlet } from '@angular/router'; +import {Component, inject} from '@angular/core'; +import {FormBuilder, FormsModule, ReactiveFormsModule} from '@angular/forms'; +import {MatButton} from '@angular/material/button'; +import {MatCheckbox} from '@angular/material/checkbox'; +import {MatFormField, MatLabel} from '@angular/material/form-field'; +import {MatInput} from '@angular/material/input'; +import {MatListModule} from '@angular/material/list'; +import {MatSnackBar} from '@angular/material/snack-bar'; +import {BackgroundBuildisoOptions, CobblerApiService} from 'cobbler-api'; +import {UserService} from '../../services/user.service'; @Component({ selector: 'cobbler-build-iso', templateUrl: './build-iso.component.html', - styleUrls: ['./build-iso.component.css'], + styleUrls: ['./build-iso.component.scss'], standalone: true, - imports: [RouterOutlet, MatListModule], + imports: [ + MatListModule, + FormsModule, + MatButton, + MatFormField, + MatInput, + MatLabel, + ReactiveFormsModule, + MatCheckbox + ], }) export class BuildISOComponent { - constructor() {} + private readonly _formBuilder = inject(FormBuilder); + buildisoFormGroup = this._formBuilder.group({ + iso: '', + profiles: '', + systems: '', + buildisodir: '', + distro: '', + standalone: false, + airgapped: false, + source: '', + excludeDNS: false, + xorrisofsOpts: '', + }) + + constructor( + public userService: UserService, + private cobblerApiService: CobblerApiService, + private _snackBar: MatSnackBar + ) { + } + + runBuildiso(): void { + const buildisoOptions: BackgroundBuildisoOptions = { + iso: this.buildisoFormGroup.controls.iso.value, + profiles: this.buildisoFormGroup.controls.profiles.value, + systems: this.buildisoFormGroup.controls.systems.value, + buildisodir: this.buildisoFormGroup.controls.buildisodir.value, + distro: this.buildisoFormGroup.controls.distro.value, + standalone: this.buildisoFormGroup.controls.standalone.value, + airgapped: this.buildisoFormGroup.controls.airgapped.value, + source: this.buildisoFormGroup.controls.source.value, + excludeDNS: this.buildisoFormGroup.controls.excludeDNS.value, + xorrisofsOpts: this.buildisoFormGroup.controls.xorrisofsOpts.value, + }; + if (this.buildisoFormGroup.invalid) { + this._snackBar.open('Please fill out all required inputs!', 'Close', {duration: 2000}); + return; + } + this.cobblerApiService.background_buildiso(buildisoOptions, this.userService.token).subscribe( + value => { + // TODO + }, + error => { + // HTML encode the error message since it originates from XML + this._snackBar.open(this.toHTML(error.message), 'Close'); + }); + } + + toHTML(input: string): any { + // FIXME: Deduplicate method + return new DOMParser().parseFromString(input, 'text/html').documentElement.textContent; + } }