mirror of https://github.com/apache/nifi.git
NIFI-12401: Allow combo editor to reference parameters (#8068)
* NIFI-12401: - Allow combo editor to reference parameters. * NIFI-12401: - Addressing review feedback. - Handling corner cases where there is no parameter context and where there are no parameters in a bound parameter context. * NIFI-12401: - Fixing formatting issues. This closes #8068
This commit is contained in:
parent
9d50c6dd53
commit
ebfb5bc12e
|
@ -44,10 +44,7 @@ module.exports = function (config) {
|
||||||
coverageReporter: {
|
coverageReporter: {
|
||||||
dir: require('path').join(__dirname, './coverage/nifi'),
|
dir: require('path').join(__dirname, './coverage/nifi'),
|
||||||
subdir: '.',
|
subdir: '.',
|
||||||
reporters: [
|
reporters: [{ type: 'html' }, { type: 'text-summary' }]
|
||||||
{ type: 'html' },
|
|
||||||
{ type: 'text-summary' }
|
|
||||||
]
|
|
||||||
},
|
},
|
||||||
reporters: ['progress', 'kjhtml'],
|
reporters: ['progress', 'kjhtml'],
|
||||||
browsers: ['ChromeHeadless'],
|
browsers: ['ChromeHeadless'],
|
||||||
|
|
|
@ -43,11 +43,11 @@
|
||||||
"@angular-devkit/build-angular": "^16.2.0",
|
"@angular-devkit/build-angular": "^16.2.0",
|
||||||
"@angular/cli": "~16.2.0",
|
"@angular/cli": "~16.2.0",
|
||||||
"@angular/compiler-cli": "^16.2.0",
|
"@angular/compiler-cli": "^16.2.0",
|
||||||
|
"@types/codemirror": "^5.60.13",
|
||||||
"@types/d3": "^7.4.0",
|
"@types/d3": "^7.4.0",
|
||||||
"@types/humanize-duration": "^3.27.1",
|
"@types/humanize-duration": "^3.27.1",
|
||||||
"@types/jasmine": "~4.3.0",
|
"@types/jasmine": "~4.3.0",
|
||||||
"@types/webfontloader": "^1.6.35",
|
"@types/webfontloader": "^1.6.35",
|
||||||
"@types/codemirror": "^5.60.13",
|
|
||||||
"autoprefixer": "^10.4.15",
|
"autoprefixer": "^10.4.15",
|
||||||
"eslint-config-prettier": "^9.0.0",
|
"eslint-config-prettier": "^9.0.0",
|
||||||
"eslint-plugin-prettier": "^5.0.0",
|
"eslint-plugin-prettier": "^5.0.0",
|
||||||
|
|
|
@ -190,7 +190,7 @@
|
||||||
</a>
|
</a>
|
||||||
<ng-template #resultLink>
|
<ng-template #resultLink>
|
||||||
<a [routerLink]="['/process-groups', result.parentGroup.id, path, result.id]">
|
<a [routerLink]="['/process-groups', result.parentGroup.id, path, result.id]">
|
||||||
{{ result.name }}
|
{{ result.name ? result.name : result.id }}
|
||||||
</a>
|
</a>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
|
|
|
@ -171,6 +171,8 @@ export interface ParameterContextReference {
|
||||||
|
|
||||||
export interface AffectedComponentEntity {
|
export interface AffectedComponentEntity {
|
||||||
permissions: Permissions;
|
permissions: Permissions;
|
||||||
|
id: string;
|
||||||
|
revision: Revision;
|
||||||
bulletins: BulletinEntity[];
|
bulletins: BulletinEntity[];
|
||||||
component: AffectedComponent;
|
component: AffectedComponent;
|
||||||
processGroup: ProcessGroupName;
|
processGroup: ProcessGroupName;
|
||||||
|
@ -289,8 +291,8 @@ export interface PropertyDescriptor {
|
||||||
name: string;
|
name: string;
|
||||||
displayName: string;
|
displayName: string;
|
||||||
description: string;
|
description: string;
|
||||||
defaultValue: string;
|
defaultValue?: string;
|
||||||
allowableValues: AllowableValueEntity[];
|
allowableValues?: AllowableValueEntity[];
|
||||||
required: boolean;
|
required: boolean;
|
||||||
sensitive: boolean;
|
sensitive: boolean;
|
||||||
dynamic: boolean;
|
dynamic: boolean;
|
||||||
|
|
|
@ -24,7 +24,8 @@
|
||||||
formControlName="value"
|
formControlName="value"
|
||||||
[placeholder]="getComboPlaceholder()"
|
[placeholder]="getComboPlaceholder()"
|
||||||
[panelClass]="'combo-panel'"
|
[panelClass]="'combo-panel'"
|
||||||
(mousedown)="preventDrag($event)">
|
(mousedown)="preventDrag($event)"
|
||||||
|
(selectionChange)="allowableValueChanged($event.value)">
|
||||||
<ng-container *ngFor="let allowableValue of allowableValues">
|
<ng-container *ngFor="let allowableValue of allowableValues">
|
||||||
<ng-container *ngIf="allowableValue.description; else noDescription">
|
<ng-container *ngIf="allowableValue.description; else noDescription">
|
||||||
<mat-option
|
<mat-option
|
||||||
|
@ -49,6 +50,41 @@
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</mat-select>
|
</mat-select>
|
||||||
</div>
|
</div>
|
||||||
|
<div *ngIf="showParameterAllowableValues">
|
||||||
|
<div *ngIf="!parametersLoaded; else showParameters">
|
||||||
|
<ngx-skeleton-loader count="1"></ngx-skeleton-loader>
|
||||||
|
</div>
|
||||||
|
<ng-template #showParameters>
|
||||||
|
<mat-select
|
||||||
|
class="combo"
|
||||||
|
formControlName="parameterReference"
|
||||||
|
[panelClass]="'combo-panel'"
|
||||||
|
(mousedown)="preventDrag($event)">
|
||||||
|
<ng-container *ngFor="let parameterAllowableValue of parameterAllowableValues">
|
||||||
|
<ng-container *ngIf="parameterAllowableValue.description; else noDescription">
|
||||||
|
<mat-option
|
||||||
|
[value]="parameterAllowableValue.id"
|
||||||
|
(mousedown)="preventDrag($event)"
|
||||||
|
nifiTooltip
|
||||||
|
[tooltipComponentType]="TextTip"
|
||||||
|
[tooltipInputData]="getAllowableValueOptionTipData(parameterAllowableValue)"
|
||||||
|
[delayClose]="false">
|
||||||
|
<span class="option-text" [class.unset]="parameterAllowableValue.value == null">{{
|
||||||
|
parameterAllowableValue.displayName
|
||||||
|
}}</span>
|
||||||
|
</mat-option>
|
||||||
|
</ng-container>
|
||||||
|
<ng-template #noDescription>
|
||||||
|
<mat-option [value]="parameterAllowableValue.id" (mousedown)="preventDrag($event)">
|
||||||
|
<span class="option-text" [class.unset]="parameterAllowableValue.value == null">{{
|
||||||
|
parameterAllowableValue.displayName
|
||||||
|
}}</span>
|
||||||
|
</mat-option>
|
||||||
|
</ng-template>
|
||||||
|
</ng-container>
|
||||||
|
</mat-select>
|
||||||
|
</ng-template>
|
||||||
|
</div>
|
||||||
<div class="flex justify-end items-center gap-x-2">
|
<div class="flex justify-end items-center gap-x-2">
|
||||||
<button
|
<button
|
||||||
color="accent"
|
color="accent"
|
||||||
|
|
|
@ -18,19 +18,67 @@
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
|
|
||||||
import { ComboEditor } from './combo-editor.component';
|
import { ComboEditor } from './combo-editor.component';
|
||||||
|
import { PropertyItem } from '../../property-table.component';
|
||||||
|
import { Parameter } from '../../../../../state/shared';
|
||||||
|
import { of } from 'rxjs';
|
||||||
|
|
||||||
describe('ComboEditor', () => {
|
describe('ComboEditor', () => {
|
||||||
let component: ComboEditor;
|
let component: ComboEditor;
|
||||||
let fixture: ComponentFixture<ComboEditor>;
|
let fixture: ComponentFixture<ComboEditor>;
|
||||||
|
|
||||||
|
let item: PropertyItem | null = null;
|
||||||
|
let parameters: Parameter[] = [
|
||||||
|
{
|
||||||
|
name: 'one',
|
||||||
|
description: 'Description for one.',
|
||||||
|
sensitive: false,
|
||||||
|
value: 'value',
|
||||||
|
provided: false,
|
||||||
|
referencingComponents: [],
|
||||||
|
parameterContext: {
|
||||||
|
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
|
||||||
|
permissions: {
|
||||||
|
canRead: true,
|
||||||
|
canWrite: true
|
||||||
|
},
|
||||||
|
component: {
|
||||||
|
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
|
||||||
|
name: 'params 1'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
inherited: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'two',
|
||||||
|
description: 'Description for two.',
|
||||||
|
sensitive: false,
|
||||||
|
value: 'value',
|
||||||
|
provided: false,
|
||||||
|
referencingComponents: [],
|
||||||
|
parameterContext: {
|
||||||
|
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
|
||||||
|
permissions: {
|
||||||
|
canRead: true,
|
||||||
|
canWrite: true
|
||||||
|
},
|
||||||
|
component: {
|
||||||
|
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
|
||||||
|
name: 'params 1'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
inherited: false
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
imports: [ComboEditor]
|
imports: [ComboEditor]
|
||||||
});
|
});
|
||||||
fixture = TestBed.createComponent(ComboEditor);
|
fixture = TestBed.createComponent(ComboEditor);
|
||||||
component = fixture.componentInstance;
|
component = fixture.componentInstance;
|
||||||
component.supportsParameters = false;
|
|
||||||
component.item = {
|
// re-establish the item before each test execution
|
||||||
|
item = {
|
||||||
property: 'Destination',
|
property: 'Destination',
|
||||||
value: 'flowfile-attribute',
|
value: 'flowfile-attribute',
|
||||||
descriptor: {
|
descriptor: {
|
||||||
|
@ -69,10 +117,114 @@ describe('ComboEditor', () => {
|
||||||
dirty: false,
|
dirty: false,
|
||||||
type: 'required'
|
type: 'required'
|
||||||
};
|
};
|
||||||
fixture.detectChanges();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should create', () => {
|
it('should create', () => {
|
||||||
|
if (item) {
|
||||||
|
component.item = item;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
expect(component).toBeTruthy();
|
expect(component).toBeTruthy();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('verify combo value', () => {
|
||||||
|
if (item) {
|
||||||
|
component.item = item;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
const formValue = component.comboEditorForm.get('value')?.value;
|
||||||
|
expect(component.itemLookup.get(formValue)?.value).toEqual(item.value);
|
||||||
|
expect(component.comboEditorForm.get('parameterReference')).toBeNull();
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith(item.value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('verify combo not required with null value and default', () => {
|
||||||
|
if (item) {
|
||||||
|
item.value = null;
|
||||||
|
item.descriptor.required = false;
|
||||||
|
|
||||||
|
component.item = item;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
const formValue = component.comboEditorForm.get('value')?.value;
|
||||||
|
expect(component.itemLookup.get(formValue)?.value).toEqual(item.descriptor.defaultValue);
|
||||||
|
expect(component.comboEditorForm.get('parameterReference')).toBeNull();
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith(item.descriptor.defaultValue);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('verify combo not required with null value and no default', () => {
|
||||||
|
if (item) {
|
||||||
|
item.value = null;
|
||||||
|
item.descriptor.required = false;
|
||||||
|
item.descriptor.defaultValue = undefined;
|
||||||
|
|
||||||
|
component.item = item;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
const formValue = component.comboEditorForm.get('value')?.value;
|
||||||
|
expect(component.itemLookup.get(formValue)?.value).toEqual(item.value);
|
||||||
|
expect(component.comboEditorForm.get('parameterReference')).toBeNull();
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith(item.value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('verify combo with parameter reference', () => {
|
||||||
|
if (item) {
|
||||||
|
item.value = '#{one}';
|
||||||
|
|
||||||
|
component.item = item;
|
||||||
|
component.getParameters = (sensitive: boolean) => {
|
||||||
|
return of(parameters);
|
||||||
|
};
|
||||||
|
fixture.detectChanges();
|
||||||
|
fixture.whenStable().then(() => {
|
||||||
|
const formValue = component.comboEditorForm.get('value')?.value;
|
||||||
|
expect(component.itemLookup.get(formValue)?.value).toEqual(item?.value);
|
||||||
|
expect(component.comboEditorForm.get('parameterReference')).toBeDefined();
|
||||||
|
|
||||||
|
const parameterReferenceValue = component.comboEditorForm.get('parameterReference')?.value;
|
||||||
|
expect(component.itemLookup.get(parameterReferenceValue)?.value).toEqual(item?.value);
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith(item?.value);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('verify combo with missing parameter reference', () => {
|
||||||
|
if (item) {
|
||||||
|
item.value = '#{three}';
|
||||||
|
|
||||||
|
component.item = item;
|
||||||
|
component.getParameters = (sensitive: boolean) => {
|
||||||
|
return of(parameters);
|
||||||
|
};
|
||||||
|
fixture.detectChanges();
|
||||||
|
fixture.whenStable().then(() => {
|
||||||
|
const formValue = component.comboEditorForm.get('value')?.value;
|
||||||
|
expect(component.itemLookup.get(formValue)?.value).toEqual('#{' + parameters[0].value + '}');
|
||||||
|
expect(component.comboEditorForm.get('parameterReference')).toBeDefined();
|
||||||
|
|
||||||
|
const parameterReferenceValue = component.comboEditorForm.get('parameterReference')?.value;
|
||||||
|
expect(component.itemLookup.get(parameterReferenceValue)?.value).toEqual(item?.value);
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith('#{' + parameters[0].value + '}');
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -25,12 +25,14 @@ import { MatButtonModule } from '@angular/material/button';
|
||||||
import { MatCheckboxModule } from '@angular/material/checkbox';
|
import { MatCheckboxModule } from '@angular/material/checkbox';
|
||||||
import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common';
|
import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common';
|
||||||
import { NifiTooltipDirective } from '../../../tooltips/nifi-tooltip.directive';
|
import { NifiTooltipDirective } from '../../../tooltips/nifi-tooltip.directive';
|
||||||
import { PropertyDescriptor, AllowableValue, TextTipInput } from '../../../../../state/shared';
|
import { AllowableValue, Parameter, PropertyDescriptor, TextTipInput } from '../../../../../state/shared';
|
||||||
import { MatOptionModule } from '@angular/material/core';
|
import { MatOptionModule } from '@angular/material/core';
|
||||||
import { MatSelectModule } from '@angular/material/select';
|
import { MatSelectModule } from '@angular/material/select';
|
||||||
import { MatTooltipModule } from '@angular/material/tooltip';
|
import { MatTooltipModule } from '@angular/material/tooltip';
|
||||||
import { TextTip } from '../../../tooltips/text-tip/text-tip.component';
|
import { TextTip } from '../../../tooltips/text-tip/text-tip.component';
|
||||||
import { A11yModule } from '@angular/cdk/a11y';
|
import { A11yModule } from '@angular/cdk/a11y';
|
||||||
|
import { Observable, take } from 'rxjs';
|
||||||
|
import { NgxSkeletonLoaderModule } from 'ngx-skeleton-loader';
|
||||||
|
|
||||||
export interface AllowableValueItem extends AllowableValue {
|
export interface AllowableValueItem extends AllowableValue {
|
||||||
id: number;
|
id: number;
|
||||||
|
@ -55,15 +57,68 @@ export interface AllowableValueItem extends AllowableValue {
|
||||||
NgForOf,
|
NgForOf,
|
||||||
MatTooltipModule,
|
MatTooltipModule,
|
||||||
NgIf,
|
NgIf,
|
||||||
A11yModule
|
A11yModule,
|
||||||
|
NgxSkeletonLoaderModule
|
||||||
],
|
],
|
||||||
styleUrls: ['./combo-editor.component.scss']
|
styleUrls: ['./combo-editor.component.scss']
|
||||||
})
|
})
|
||||||
export class ComboEditor {
|
export class ComboEditor {
|
||||||
@Input() set item(item: PropertyItem) {
|
@Input() set item(item: PropertyItem) {
|
||||||
this.itemLookup.clear();
|
if (item.value != null) {
|
||||||
|
this.configuredValue = item.value;
|
||||||
|
} else if (item.descriptor.defaultValue != null) {
|
||||||
|
this.configuredValue = item.descriptor.defaultValue;
|
||||||
|
}
|
||||||
|
|
||||||
this.descriptor = item.descriptor;
|
this.descriptor = item.descriptor;
|
||||||
|
this.sensitive = item.descriptor.sensitive;
|
||||||
|
|
||||||
|
this.itemSet = true;
|
||||||
|
this.initialAllowableValues();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Input() set getParameters(getParameters: (sensitive: boolean) => Observable<Parameter[]>) {
|
||||||
|
this._getParameters = getParameters;
|
||||||
|
|
||||||
|
this.supportsParameters = getParameters != null;
|
||||||
|
this.initialAllowableValues();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Output() ok: EventEmitter<any> = new EventEmitter<any>();
|
||||||
|
@Output() cancel: EventEmitter<void> = new EventEmitter<void>();
|
||||||
|
|
||||||
|
protected readonly TextTip = TextTip;
|
||||||
|
|
||||||
|
itemLookup: Map<number, AllowableValueItem> = new Map<number, AllowableValueItem>();
|
||||||
|
referencesParametersId: number = -1;
|
||||||
|
configuredParameterId: number = -1;
|
||||||
|
|
||||||
|
comboEditorForm: FormGroup;
|
||||||
|
descriptor!: PropertyDescriptor;
|
||||||
|
allowableValues!: AllowableValueItem[];
|
||||||
|
|
||||||
|
showParameterAllowableValues: boolean = false;
|
||||||
|
parameterAllowableValues!: AllowableValueItem[];
|
||||||
|
|
||||||
|
sensitive: boolean = false;
|
||||||
|
supportsParameters: boolean = false;
|
||||||
|
parametersLoaded: boolean = false;
|
||||||
|
|
||||||
|
itemSet: boolean = false;
|
||||||
|
configuredValue: string | null = null;
|
||||||
|
_getParameters!: (sensitive: boolean) => Observable<Parameter[]>;
|
||||||
|
|
||||||
|
constructor(private formBuilder: FormBuilder) {
|
||||||
|
this.comboEditorForm = this.formBuilder.group({
|
||||||
|
value: new FormControl(null, Validators.required)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
initialAllowableValues(): void {
|
||||||
|
if (this.itemSet) {
|
||||||
|
this.itemLookup.clear();
|
||||||
this.allowableValues = [];
|
this.allowableValues = [];
|
||||||
|
this.referencesParametersId = -1;
|
||||||
|
|
||||||
let i: number = 0;
|
let i: number = 0;
|
||||||
let selectedItem: AllowableValueItem | null = null;
|
let selectedItem: AllowableValueItem | null = null;
|
||||||
|
@ -77,11 +132,12 @@ export class ComboEditor {
|
||||||
this.itemLookup.set(noValue.id, noValue);
|
this.itemLookup.set(noValue.id, noValue);
|
||||||
this.allowableValues.push(noValue);
|
this.allowableValues.push(noValue);
|
||||||
|
|
||||||
if (noValue.value == item.value) {
|
if (noValue.value == this.configuredValue) {
|
||||||
selectedItem = noValue;
|
selectedItem = noValue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.descriptor.allowableValues) {
|
||||||
const allowableValueItems: AllowableValueItem[] = this.descriptor.allowableValues.map(
|
const allowableValueItems: AllowableValueItem[] = this.descriptor.allowableValues.map(
|
||||||
(allowableValueEntity) => {
|
(allowableValueEntity) => {
|
||||||
const allowableValue: AllowableValueItem = {
|
const allowableValue: AllowableValueItem = {
|
||||||
|
@ -90,7 +146,7 @@ export class ComboEditor {
|
||||||
};
|
};
|
||||||
this.itemLookup.set(allowableValue.id, allowableValue);
|
this.itemLookup.set(allowableValue.id, allowableValue);
|
||||||
|
|
||||||
if (allowableValue.value == item.value) {
|
if (allowableValue.value == this.configuredValue) {
|
||||||
selectedItem = allowableValue;
|
selectedItem = allowableValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -98,6 +154,70 @@ export class ComboEditor {
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
this.allowableValues.push(...allowableValueItems);
|
this.allowableValues.push(...allowableValueItems);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.supportsParameters) {
|
||||||
|
this.parametersLoaded = false;
|
||||||
|
|
||||||
|
// parameters are supported so add the item to support showing
|
||||||
|
// and hiding the parameter options select
|
||||||
|
const referencesParameterOption: AllowableValueItem = {
|
||||||
|
id: i++,
|
||||||
|
displayName: 'Reference Parameter...',
|
||||||
|
value: null
|
||||||
|
};
|
||||||
|
this.allowableValues.push(referencesParameterOption);
|
||||||
|
this.itemLookup.set(referencesParameterOption.id, referencesParameterOption);
|
||||||
|
|
||||||
|
// record the item of the item to more easily identify this item
|
||||||
|
this.referencesParametersId = referencesParameterOption.id;
|
||||||
|
|
||||||
|
// if the current value references a parameter auto select the
|
||||||
|
// references parameter item
|
||||||
|
if (this.referencesParameter(this.configuredValue)) {
|
||||||
|
selectedItem = referencesParameterOption;
|
||||||
|
|
||||||
|
// trigger allowable value changed to show the parameters
|
||||||
|
this.allowableValueChanged(this.referencesParametersId);
|
||||||
|
}
|
||||||
|
|
||||||
|
this._getParameters(this.sensitive)
|
||||||
|
.pipe(take(1))
|
||||||
|
.subscribe((parameters) => {
|
||||||
|
if (parameters.length > 0) {
|
||||||
|
// capture the value of i which will be the id of the first
|
||||||
|
// parameter
|
||||||
|
this.configuredParameterId = i;
|
||||||
|
|
||||||
|
// create allowable values for each parameter
|
||||||
|
parameters.forEach((parameter) => {
|
||||||
|
const parameterItem: AllowableValueItem = {
|
||||||
|
id: i++,
|
||||||
|
displayName: parameter.name,
|
||||||
|
value: '#{' + parameter.name + '}',
|
||||||
|
description: parameter.description
|
||||||
|
};
|
||||||
|
this.parameterAllowableValues.push(parameterItem);
|
||||||
|
this.itemLookup.set(parameterItem.id, parameterItem);
|
||||||
|
|
||||||
|
// if the configured parameter is still available,
|
||||||
|
// capture the id, so we can auto select it
|
||||||
|
if (parameterItem.value === this.configuredValue) {
|
||||||
|
this.configuredParameterId = parameterItem.id;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// if combo still set to reference a parameter, set the default value
|
||||||
|
if (this.comboEditorForm.get('value')?.value == this.referencesParametersId) {
|
||||||
|
this.comboEditorForm.get('parameterReference')?.setValue(this.configuredParameterId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.parametersLoaded = true;
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.parameterAllowableValues = [];
|
||||||
|
}
|
||||||
|
|
||||||
if (selectedItem) {
|
if (selectedItem) {
|
||||||
// mat-select does not have good support for options with null value so we've
|
// mat-select does not have good support for options with null value so we've
|
||||||
|
@ -105,29 +225,37 @@ export class ComboEditor {
|
||||||
this.comboEditorForm.get('value')?.setValue(selectedItem.id);
|
this.comboEditorForm.get('value')?.setValue(selectedItem.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@Input() supportsParameters: boolean = false;
|
}
|
||||||
|
|
||||||
@Output() ok: EventEmitter<any> = new EventEmitter<any>();
|
referencesParameter(value: string | null): boolean {
|
||||||
@Output() cancel: EventEmitter<void> = new EventEmitter<void>();
|
if (value) {
|
||||||
|
return value.startsWith('#{') && value.endsWith('}');
|
||||||
|
}
|
||||||
|
|
||||||
protected readonly TextTip = TextTip;
|
return false;
|
||||||
|
|
||||||
itemLookup: Map<number, AllowableValueItem> = new Map<number, AllowableValueItem>();
|
|
||||||
|
|
||||||
comboEditorForm: FormGroup;
|
|
||||||
descriptor!: PropertyDescriptor;
|
|
||||||
allowableValues!: AllowableValueItem[];
|
|
||||||
|
|
||||||
constructor(private formBuilder: FormBuilder) {
|
|
||||||
this.comboEditorForm = this.formBuilder.group({
|
|
||||||
value: new FormControl(null, Validators.required)
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
preventDrag(event: MouseEvent): void {
|
preventDrag(event: MouseEvent): void {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
allowableValueChanged(value: number): void {
|
||||||
|
this.showParameterAllowableValues = value === this.referencesParametersId;
|
||||||
|
|
||||||
|
if (this.showParameterAllowableValues) {
|
||||||
|
if (this.configuredParameterId === -1) {
|
||||||
|
this.comboEditorForm.addControl('parameterReference', new FormControl(null, Validators.required));
|
||||||
|
} else {
|
||||||
|
this.comboEditorForm.addControl(
|
||||||
|
'parameterReference',
|
||||||
|
new FormControl(this.configuredParameterId, Validators.required)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.comboEditorForm.removeControl('parameterReference');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
getComboPlaceholder(): string {
|
getComboPlaceholder(): string {
|
||||||
const valueControl: AbstractControl | null = this.comboEditorForm.get('value');
|
const valueControl: AbstractControl | null = this.comboEditorForm.get('value');
|
||||||
if (valueControl) {
|
if (valueControl) {
|
||||||
|
@ -150,10 +278,24 @@ export class ComboEditor {
|
||||||
if (valueControl) {
|
if (valueControl) {
|
||||||
const selectedItem: AllowableValueItem | undefined = this.itemLookup.get(valueControl.value);
|
const selectedItem: AllowableValueItem | undefined = this.itemLookup.get(valueControl.value);
|
||||||
if (selectedItem) {
|
if (selectedItem) {
|
||||||
|
// if the value currently references a parameter emit the parameter, get the parameter reference control and emit that value
|
||||||
|
if (selectedItem.id == this.referencesParametersId) {
|
||||||
|
const parameterReferenceControl: AbstractControl | null =
|
||||||
|
this.comboEditorForm.get('parameterReference');
|
||||||
|
if (parameterReferenceControl) {
|
||||||
|
const selectedParameterItem: AllowableValueItem | undefined = this.itemLookup.get(
|
||||||
|
parameterReferenceControl.value
|
||||||
|
);
|
||||||
|
if (selectedParameterItem) {
|
||||||
|
this.ok.next(selectedParameterItem.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
this.ok.next(selectedItem.value);
|
this.ok.next(selectedItem.value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
cancelClicked(): void {
|
cancelClicked(): void {
|
||||||
this.cancel.next();
|
this.cancel.next();
|
||||||
|
|
|
@ -19,6 +19,7 @@ import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
|
|
||||||
import { NfEditor } from './nf-editor.component';
|
import { NfEditor } from './nf-editor.component';
|
||||||
import { HttpClientTestingModule } from '@angular/common/http/testing';
|
import { HttpClientTestingModule } from '@angular/common/http/testing';
|
||||||
|
import { PropertyItem } from '../../property-table.component';
|
||||||
|
|
||||||
describe('NfEditor', () => {
|
describe('NfEditor', () => {
|
||||||
let component: NfEditor;
|
let component: NfEditor;
|
||||||
|
@ -30,10 +31,84 @@ describe('NfEditor', () => {
|
||||||
});
|
});
|
||||||
fixture = TestBed.createComponent(NfEditor);
|
fixture = TestBed.createComponent(NfEditor);
|
||||||
component = fixture.componentInstance;
|
component = fixture.componentInstance;
|
||||||
fixture.detectChanges();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should create', () => {
|
it('should create', () => {
|
||||||
|
fixture.detectChanges();
|
||||||
expect(component).toBeTruthy();
|
expect(component).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('verify value set', () => {
|
||||||
|
const value: string = 'my-group-id';
|
||||||
|
const item: PropertyItem = {
|
||||||
|
property: 'group.id',
|
||||||
|
value,
|
||||||
|
descriptor: {
|
||||||
|
name: 'group.id',
|
||||||
|
displayName: 'Group ID',
|
||||||
|
description:
|
||||||
|
"A Group ID is used to identify consumers that are within the same consumer group. Corresponds to Kafka's 'group.id' property.",
|
||||||
|
required: true,
|
||||||
|
sensitive: false,
|
||||||
|
dynamic: false,
|
||||||
|
supportsEl: true,
|
||||||
|
expressionLanguageScope: 'Environment variables defined at JVM level and system properties',
|
||||||
|
dependencies: []
|
||||||
|
},
|
||||||
|
id: 3,
|
||||||
|
triggerEdit: false,
|
||||||
|
deleted: false,
|
||||||
|
added: false,
|
||||||
|
dirty: false,
|
||||||
|
type: 'required'
|
||||||
|
};
|
||||||
|
|
||||||
|
component.item = item;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(component.nfEditorForm.get('value')?.value).toEqual(value);
|
||||||
|
expect(component.nfEditorForm.get('value')?.disabled).toBeFalse();
|
||||||
|
expect(component.nfEditorForm.get('setEmptyString')?.value).toBeFalse();
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith(value);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('verify empty value set', () => {
|
||||||
|
const value: string = '';
|
||||||
|
const item: PropertyItem = {
|
||||||
|
property: 'group.id',
|
||||||
|
value,
|
||||||
|
descriptor: {
|
||||||
|
name: 'group.id',
|
||||||
|
displayName: 'Group ID',
|
||||||
|
description:
|
||||||
|
"A Group ID is used to identify consumers that are within the same consumer group. Corresponds to Kafka's 'group.id' property.",
|
||||||
|
required: true,
|
||||||
|
sensitive: false,
|
||||||
|
dynamic: false,
|
||||||
|
supportsEl: true,
|
||||||
|
expressionLanguageScope: 'Environment variables defined at JVM level and system properties',
|
||||||
|
dependencies: []
|
||||||
|
},
|
||||||
|
id: 3,
|
||||||
|
triggerEdit: false,
|
||||||
|
deleted: false,
|
||||||
|
added: false,
|
||||||
|
dirty: false,
|
||||||
|
type: 'required'
|
||||||
|
};
|
||||||
|
|
||||||
|
component.item = item;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(component.nfEditorForm.get('value')?.value).toEqual(value);
|
||||||
|
expect(component.nfEditorForm.get('value')?.disabled).toBeTruthy();
|
||||||
|
expect(component.nfEditorForm.get('setEmptyString')?.value).toBeTruthy();
|
||||||
|
|
||||||
|
spyOn(component.ok, 'next');
|
||||||
|
component.okClicked();
|
||||||
|
expect(component.ok.next).toHaveBeenCalledWith(value);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -15,18 +15,7 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {
|
import { Component, EventEmitter, Input, OnDestroy, Output, Renderer2, ViewContainerRef } from '@angular/core';
|
||||||
AfterViewInit,
|
|
||||||
Component,
|
|
||||||
ElementRef,
|
|
||||||
EventEmitter,
|
|
||||||
Input,
|
|
||||||
OnDestroy,
|
|
||||||
Output,
|
|
||||||
Renderer2,
|
|
||||||
ViewChild,
|
|
||||||
ViewContainerRef
|
|
||||||
} from '@angular/core';
|
|
||||||
import { PropertyItem } from '../../property-table.component';
|
import { PropertyItem } from '../../property-table.component';
|
||||||
import { CdkDrag, CdkDragHandle } from '@angular/cdk/drag-drop';
|
import { CdkDrag, CdkDragHandle } from '@angular/cdk/drag-drop';
|
||||||
import { AbstractControl, FormBuilder, FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';
|
import { AbstractControl, FormBuilder, FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';
|
||||||
|
|
|
@ -145,6 +145,7 @@
|
||||||
<combo-editor
|
<combo-editor
|
||||||
*ngIf="hasAllowableValues(editorItem); else nfEditor"
|
*ngIf="hasAllowableValues(editorItem); else nfEditor"
|
||||||
[item]="editorItem"
|
[item]="editorItem"
|
||||||
|
[getParameters]="getParameters"
|
||||||
(ok)="savePropertyValue(editorItem, $event)"
|
(ok)="savePropertyValue(editorItem, $event)"
|
||||||
(cancel)="closeEditor()"></combo-editor>
|
(cancel)="closeEditor()"></combo-editor>
|
||||||
<ng-template #nfEditor>
|
<ng-template #nfEditor>
|
||||||
|
|
|
@ -347,8 +347,8 @@ export class PropertyTable implements AfterViewInit, ControlValueAccessor {
|
||||||
}
|
}
|
||||||
|
|
||||||
resolvePropertyValue(property: Property): string | null {
|
resolvePropertyValue(property: Property): string | null {
|
||||||
const allowableValues: AllowableValueEntity[] = property.descriptor.allowableValues;
|
const allowableValues: AllowableValueEntity[] | undefined = property.descriptor.allowableValues;
|
||||||
if (this.nifiCommon.isEmpty(allowableValues)) {
|
if (allowableValues == null || this.nifiCommon.isEmpty(allowableValues)) {
|
||||||
return property.value;
|
return property.value;
|
||||||
} else {
|
} else {
|
||||||
const allowableValue: AllowableValueEntity | undefined = allowableValues.find(
|
const allowableValue: AllowableValueEntity | undefined = allowableValues.find(
|
||||||
|
@ -392,7 +392,7 @@ export class PropertyTable implements AfterViewInit, ControlValueAccessor {
|
||||||
// TODO - add Input() for supportsGoTo? currently only false in summary table
|
// TODO - add Input() for supportsGoTo? currently only false in summary table
|
||||||
|
|
||||||
const descriptor: PropertyDescriptor = item.descriptor;
|
const descriptor: PropertyDescriptor = item.descriptor;
|
||||||
if (item.value && descriptor.identifiesControllerService) {
|
if (item.value && descriptor.identifiesControllerService && descriptor.allowableValues) {
|
||||||
return descriptor.allowableValues.some(
|
return descriptor.allowableValues.some(
|
||||||
(entity: AllowableValueEntity) => entity.allowableValue.value == item.value
|
(entity: AllowableValueEntity) => entity.allowableValue.value == item.value
|
||||||
);
|
);
|
||||||
|
|
Loading…
Reference in New Issue