NIFI-12417: Process Group Configuration (#8075)

* NIFI-12417:
- Process Group Configuration.
- Removed unnecessary module imports.

* NIFI-12417:
- Addressing review feedback.

This closes #8075
This commit is contained in:
Matt Gilman 2023-11-29 17:21:35 -05:00 committed by GitHub
parent ebfb5bc12e
commit 05575364a3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 671 additions and 29 deletions

View File

@ -45,7 +45,8 @@
"vendorChunk": true,
"extractLicenses": false,
"sourceMap": true,
"namedChunks": true
"namedChunks": true,
"outputHashing": "none"
}
},
"defaultConfiguration": "production"

View File

@ -44,8 +44,6 @@ import { MAT_FORM_FIELD_DEFAULT_OPTIONS } from '@angular/material/form-field';
BrowserModule,
AppRoutingModule,
BrowserAnimationsModule,
FlowDesignerModule,
LoginModule,
HttpClientModule,
HttpClientXsrfModule.withOptions({
cookieName: '__Secure-Request-Token',

View File

@ -307,14 +307,15 @@ export class CanvasUtils {
* @param selection
*/
public isConfigurable(selection: any): boolean {
// ensure the correct number of components are selected
if (selection.size() !== 1) {
return selection.empty();
if (selection.empty()) {
return this.canvasPermissions.canRead && this.canvasPermissions.canWrite;
}
if (this.isProcessGroup(selection)) {
return true;
// disable configuration if there are multiple components selected
if (selection.size() > 1) {
return false;
}
if (!this.canRead(selection) || !this.canModify(selection)) {
return false;
}

View File

@ -258,6 +258,11 @@ export const openEditConnectionDialog = createAction(
props<{ request: EditConnectionDialogRequest }>()
);
export const openEditProcessGroupDialog = createAction(
'[Canvas] Open Edit Process Group Dialog',
props<{ request: EditComponentDialogRequest }>()
);
export const updateComponent = createAction('[Canvas] Update Component', props<{ request: UpdateComponentRequest }>());
export const updateComponentSuccess = createAction(

View File

@ -85,6 +85,7 @@ import { CreateConnection } from '../../ui/connection/create-connection/create-c
import { EditConnectionComponent } from '../../ui/connection/edit-connection/edit-connection.component';
import { OkDialog } from '../../../../ui/common/ok-dialog/ok-dialog.component';
import { GroupComponents } from '../../ui/process-group/group-components/group-components.component';
import { EditProcessGroup } from '../../ui/process-group/edit-process-group/edit-process-group.component';
@Injectable()
export class FlowEffects {
@ -651,6 +652,8 @@ export class FlowEffects {
return of(FlowActions.openEditProcessorDialog({ request }));
case ComponentType.Connection:
return of(FlowActions.openEditConnectionDialog({ request }));
case ComponentType.ProcessGroup:
return of(FlowActions.openEditProcessGroupDialog({ request }));
case ComponentType.InputPort:
case ComponentType.OutputPort:
return of(FlowActions.openEditPortDialog({ request }));
@ -773,18 +776,20 @@ export class FlowEffects {
// TODO - inline service creation...
editDialogReference.componentInstance.editProcessor.pipe(take(1)).subscribe((payload: any) => {
this.store.dispatch(
FlowActions.updateProcessor({
request: {
id: request.entity.id,
uri: request.uri,
type: request.type,
payload
}
})
);
});
editDialogReference.componentInstance.editProcessor
.pipe(takeUntil(editDialogReference.afterClosed()))
.subscribe((payload: any) => {
this.store.dispatch(
FlowActions.updateProcessor({
request: {
id: request.entity.id,
uri: request.uri,
type: request.type,
payload
}
})
);
});
editDialogReference.afterClosed().subscribe(() => {
this.store.dispatch(FlowActions.clearFlowApiError());
@ -869,6 +874,61 @@ export class FlowEffects {
{ dispatch: false }
);
openEditProcessGroupDialog$ = createEffect(
() =>
this.actions$.pipe(
ofType(FlowActions.openEditProcessGroupDialog),
map((action) => action.request),
switchMap((action) =>
this.flowService.getParameterContexts().pipe(
take(1),
map((response) => [action, response.parameterContexts])
)
),
tap(([request, parameterContexts]) => {
const editDialogReference = this.dialog.open(EditProcessGroup, {
data: request,
panelClass: 'large-dialog'
});
editDialogReference.componentInstance.saving$ = this.store.select(selectSaving);
editDialogReference.componentInstance.parameterContexts = parameterContexts;
editDialogReference.componentInstance.editProcessGroup
.pipe(takeUntil(editDialogReference.afterClosed()))
.subscribe((payload: any) => {
this.store.dispatch(
FlowActions.updateComponent({
request: {
id: request.entity.id,
uri: request.uri,
type: request.type,
payload
}
})
);
});
editDialogReference.afterClosed().subscribe(() => {
this.store.dispatch(FlowActions.clearFlowApiError());
this.store.dispatch(
FlowActions.selectComponents({
request: {
components: [
{
id: request.entity.id,
componentType: request.type
}
]
}
})
);
});
})
),
{ dispatch: false }
);
updateComponent$ = createEffect(() =>
this.actions$.pipe(
ofType(FlowActions.updateComponent),

View File

@ -286,15 +286,18 @@ export class ContextMenu implements OnInit {
clazz: 'fa fa-gear',
text: 'Configure',
action: function (store: Store<CanvasState>, selection: any) {
const selectionData = selection.datum();
store.dispatch(
navigateToEditComponent({
request: {
type: selectionData.type,
id: selectionData.id
}
})
);
// TODO - when selection is empty support configuring the current Process Group
if (!selection.empty()) {
const selectionData = selection.datum();
store.dispatch(
navigateToEditComponent({
request: {
type: selectionData.type,
id: selectionData.id
}
})
);
}
}
},
{

View File

@ -0,0 +1,162 @@
<!--
~ Licensed to the Apache Software Foundation (ASF) under one or more
~ contributor license agreements. See the NOTICE file distributed with
~ this work for additional information regarding copyright ownership.
~ The ASF licenses this file to You under the Apache License, Version 2.0
~ (the "License"); you may not use this file except in compliance with
~ the License. You may obtain a copy of the License at
~
~ http://www.apache.org/licenses/LICENSE-2.0
~
~ Unless required by applicable law or agreed to in writing, software
~ distributed under the License is distributed on an "AS IS" BASIS,
~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
~ See the License for the specific language governing permissions and
~ limitations under the License.
-->
<h2 mat-dialog-title>Edit Process Group</h2>
<form class="process-group-edit-form" [formGroup]="editProcessGroupForm">
<mat-dialog-content>
<mat-tab-group>
<mat-tab label="Settings">
<div class="tab-content py-4 flex gap-x-4">
<div class="w-full">
<div>
<mat-form-field>
<mat-label>Name</mat-label>
<input matInput formControlName="name" type="text" />
</mat-form-field>
</div>
<div>
<mat-form-field>
<mat-label>Parameter Context</mat-label>
<mat-select formControlName="parameterContext">
<ng-container *ngFor="let option of parameterContextsOptions">
<ng-container *ngIf="option.description; else noDescription">
<mat-option
[value]="option.value"
nifiTooltip
[tooltipComponentType]="TextTip"
[tooltipInputData]="getSelectOptionTipData(option)"
[delayClose]="false"
>{{ option.text }}</mat-option
>
</ng-container>
<ng-template #noDescription>
<mat-option [value]="option.value">{{ option.text }}</mat-option>
</ng-template>
</ng-container>
</mat-select>
</mat-form-field>
</div>
<div class="-mt-6 mb-4">
<mat-checkbox
formControlName="applyParameterContextRecursively"
name="applyParameterContextRecursively">
Apply Recursively
</mat-checkbox>
</div>
<div>
<mat-form-field>
<mat-label>Execution Engine</mat-label>
<mat-select formControlName="executionEngine">
<ng-container *ngFor="let option of executionEngineOptions">
<mat-option
[value]="option.value"
nifiTooltip
[tooltipComponentType]="TextTip"
[tooltipInputData]="getSelectOptionTipData(option)"
[delayClose]="false">
{{ option.text }}
</mat-option>
</ng-container>
</mat-select>
</mat-form-field>
</div>
<div>
<mat-form-field>
<mat-label>Process Group FlowFile Concurrency</mat-label>
<mat-select formControlName="flowfileConcurrency">
<ng-container *ngFor="let option of flowfileConcurrencyOptions">
<mat-option
[value]="option.value"
nifiTooltip
[tooltipComponentType]="TextTip"
[tooltipInputData]="getSelectOptionTipData(option)"
[delayClose]="false">
{{ option.text }}
</mat-option>
</ng-container>
</mat-select>
</mat-form-field>
</div>
<div>
<mat-form-field>
<mat-label>Process Group Outbound Policy</mat-label>
<mat-select formControlName="flowfileOutboundPolicy">
<ng-container *ngFor="let option of flowfileOutboundPolicyOptions">
<mat-option
[value]="option.value"
nifiTooltip
[tooltipComponentType]="TextTip"
[tooltipInputData]="getSelectOptionTipData(option)"
[delayClose]="false">
{{ option.text }}
</mat-option>
</ng-container>
</mat-select>
</mat-form-field>
</div>
</div>
<div class="w-full">
<div>
<mat-form-field>
<mat-label>Default FlowFile Expiration</mat-label>
<input matInput formControlName="defaultFlowFileExpiration" type="text" />
</mat-form-field>
</div>
<div>
<mat-form-field>
<mat-label>Default Back Pressure Object Threshold</mat-label>
<input matInput formControlName="defaultBackPressureObjectThreshold" type="text" />
</mat-form-field>
</div>
<div>
<mat-form-field>
<mat-label>Default Back Pressure Data Size Threshold</mat-label>
<input matInput formControlName="defaultBackPressureDataSizeThreshold" type="text" />
</mat-form-field>
</div>
<div>
<mat-form-field>
<mat-label>Log File Suffix</mat-label>
<input matInput formControlName="logFileSuffix" type="text" />
</mat-form-field>
</div>
</div>
</div>
</mat-tab>
<mat-tab label="Comments">
<div class="tab-content py-4">
<mat-form-field>
<mat-label>Comments</mat-label>
<textarea matInput formControlName="comments" type="text" rows="15"></textarea>
</mat-form-field>
</div>
</mat-tab>
</mat-tab-group>
</mat-dialog-content>
<mat-dialog-actions align="end" *ngIf="{ value: (saving$ | async)! } as saving">
<button color="accent" mat-raised-button mat-dialog-close>Cancel</button>
<button
[disabled]="!editProcessGroupForm.dirty || editProcessGroupForm.invalid || saving.value"
class="h-8"
type="button"
color="primary"
(click)="submitForm()"
mat-raised-button>
<span *nifiSpinner="saving.value">Apply</span>
</button>
</mat-dialog-actions>
</form>

View File

@ -0,0 +1,36 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
@use '@angular/material' as mat;
.process-group-edit-form {
@include mat.button-density(-1);
.mdc-dialog__content {
padding: 0 16px;
font-size: 14px;
.tab-content {
height: 475px;
overflow-y: auto;
}
}
.mat-mdc-form-field {
width: 100%;
}
}

View File

@ -0,0 +1,149 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { EditProcessGroup } from './edit-process-group.component';
import { EditComponentDialogRequest } from '../../../state/flow';
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
import { ComponentType } from '../../../../../state/shared';
import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
describe('EditProcessGroup', () => {
let component: EditProcessGroup;
let fixture: ComponentFixture<EditProcessGroup>;
const parameterContextId: string = '95d509b9-018b-1000-daff-b7957ea7934f';
const data: any = {
type: 'ProcessGroup',
uri: 'https://localhost:4200/nifi-api/process-groups/162380af-018c-1000-a7eb-f5d06f77168b',
entity: {
revision: {
clientId: 'de5d3be3-05be-4ba5-bc42-729e7a4b00c4',
version: 14
},
id: '162380af-018c-1000-a7eb-f5d06f77168b',
uri: 'https://localhost:4200/nifi-api/process-groups/162380af-018c-1000-a7eb-f5d06f77168b',
position: {
x: 446,
y: 151
},
permissions: {
canRead: true,
canWrite: true
},
bulletins: [],
component: {
id: '162380af-018c-1000-a7eb-f5d06f77168b',
parentGroupId: '1621f9d1-018c-1000-cb13-7eab94ffe23c',
position: {
x: 446,
y: 151
},
name: 'pg2',
comments: '',
flowfileConcurrency: 'UNBOUNDED',
flowfileOutboundPolicy: 'BATCH_OUTPUT',
defaultFlowFileExpiration: '0 sec',
defaultBackPressureObjectThreshold: 10000,
defaultBackPressureDataSizeThreshold: '1 GB',
parameterContext: {
id: parameterContextId
},
executionEngine: 'INHERITED',
maxConcurrentTasks: 1,
statelessFlowTimeout: '1 min',
runningCount: 0,
stoppedCount: 0,
invalidCount: 0,
disabledCount: 0,
activeRemotePortCount: 0,
inactiveRemotePortCount: 0,
upToDateCount: 0,
locallyModifiedCount: 0,
staleCount: 0,
locallyModifiedAndStaleCount: 0,
syncFailureCount: 0,
localInputPortCount: 0,
localOutputPortCount: 0,
publicInputPortCount: 0,
publicOutputPortCount: 0,
statelessGroupScheduledState: 'STOPPED',
inputPortCount: 0,
outputPortCount: 0
},
runningCount: 0,
stoppedCount: 0,
invalidCount: 0,
disabledCount: 0,
activeRemotePortCount: 0,
inactiveRemotePortCount: 0,
upToDateCount: 0,
locallyModifiedCount: 0,
staleCount: 0,
locallyModifiedAndStaleCount: 0,
syncFailureCount: 0,
localInputPortCount: 0,
localOutputPortCount: 0,
publicInputPortCount: 0,
publicOutputPortCount: 0,
inputPortCount: 0,
outputPortCount: 0
}
};
beforeEach(() => {
TestBed.configureTestingModule({
imports: [EditProcessGroup, BrowserAnimationsModule],
providers: [{ provide: MAT_DIALOG_DATA, useValue: data }]
});
fixture = TestBed.createComponent(EditProcessGroup);
component = fixture.componentInstance;
component.parameterContexts = [
{
revision: {
version: 0
},
id: parameterContextId,
uri: '',
permissions: {
canRead: true,
canWrite: true
},
component: {
name: 'params 2',
description: '',
parameters: [],
boundProcessGroups: [],
inheritedParameterContexts: [],
id: '95d509b9-018b-1000-daff-b7957ea7934f'
}
}
];
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
it('verify parameter context value', () => {
expect(component.parameterContextsOptions.length).toEqual(2);
expect(component.editProcessGroupForm.get('parameterContext')?.value).toEqual(parameterContextId);
});
});

View File

@ -0,0 +1,227 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { Component, EventEmitter, Inject, Input, Output } from '@angular/core';
import { MAT_DIALOG_DATA, MatDialogModule } from '@angular/material/dialog';
import { FormBuilder, FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';
import { MatInputModule } from '@angular/material/input';
import { MatCheckboxModule } from '@angular/material/checkbox';
import { MatButtonModule } from '@angular/material/button';
import { AsyncPipe, NgForOf, NgIf } from '@angular/common';
import { MatTabsModule } from '@angular/material/tabs';
import { MatOptionModule } from '@angular/material/core';
import { MatSelectModule } from '@angular/material/select';
import { Observable } from 'rxjs';
import { SelectOption, TextTipInput } from '../../../../../state/shared';
import { Client } from '../../../../../service/client.service';
import { EditComponentDialogRequest } from '../../../state/flow';
import { PropertyTable } from '../../../../../ui/common/property-table/property-table.component';
import { NifiSpinnerDirective } from '../../../../../ui/common/spinner/nifi-spinner.directive';
import { NifiTooltipDirective } from '../../../../../ui/common/tooltips/nifi-tooltip.directive';
import { TextTip } from '../../../../../ui/common/tooltips/text-tip/text-tip.component';
import { ParameterContextEntity } from '../../../../parameter-contexts/state/parameter-context-listing';
import { ControllerServiceTable } from '../../../../../ui/common/controller-service/controller-service-table/controller-service-table.component';
@Component({
selector: 'edit-process-group',
standalone: true,
templateUrl: './edit-process-group.component.html',
imports: [
ReactiveFormsModule,
MatDialogModule,
MatInputModule,
MatCheckboxModule,
MatButtonModule,
NgIf,
MatTabsModule,
MatOptionModule,
MatSelectModule,
NgForOf,
AsyncPipe,
PropertyTable,
NifiSpinnerDirective,
NifiTooltipDirective,
FormsModule,
ControllerServiceTable
],
styleUrls: ['./edit-process-group.component.scss']
})
export class EditProcessGroup {
@Input() set parameterContexts(parameterContexts: ParameterContextEntity[]) {
parameterContexts.forEach((parameterContext) => {
if (parameterContext.permissions.canRead) {
this.parameterContextsOptions.push({
text: parameterContext.component.name,
value: parameterContext.id,
description: parameterContext.component.description
});
}
});
if (this.request.entity.component.parameterContext) {
this.editProcessGroupForm.addControl(
'parameterContext',
new FormControl(this.request.entity.component.parameterContext.id)
);
} else {
this.editProcessGroupForm.addControl('parameterContext', new FormControl(null));
}
}
@Input() saving$!: Observable<boolean>;
@Output() editProcessGroup: EventEmitter<any> = new EventEmitter<any>();
protected readonly TextTip = TextTip;
editProcessGroupForm: FormGroup;
parameterContextsOptions: SelectOption[] = [];
executionEngineOptions: SelectOption[] = [
{
text: 'Inherited',
value: 'INHERITED',
description: 'Use whichever Execution Engine the parent Process Group is configured to use.'
},
{
text: 'Standard',
value: 'STANDARD',
description: 'Use the Standard NiFi Execution Engine. See the User Guide for additional details.'
},
{
text: 'Stateless',
value: 'STATELESS',
description:
'Run the dataflow using the Stateless Execution Engine. See the User Guide for additional details.'
}
];
flowfileConcurrencyOptions: SelectOption[] = [
{
text: 'Single FlowFile Per Node',
value: 'SINGLE_FLOWFILE_PER_NODE',
description:
'Only a single FlowFile is to be allowed to enter the Process Group at a time on each node in the cluster. While that FlowFile may be split into many or ' +
'spawn many children, no additional FlowFiles will be allowed to enter the Process Group through a Local Input Port until the previous FlowFile ' +
'- and all of its child/descendent FlowFiles - have been processed.'
},
{
text: 'Single Batch Per Node',
value: 'SINGLE_BATCH_PER_NODE',
description:
'When an Input Port pulls a FlowFile into the Process Group, FlowFiles will continue to be ingested into the Process Group until all input queues ' +
'have been emptied. At that point, no additional FlowFiles will be allowed to enter the Process Group through a Local Input Port until the entire batch ' +
'of FlowFiles has been processed.'
},
{
text: 'Unbounded',
value: 'UNBOUNDED',
description: 'The number of FlowFiles that can be processed concurrently is unbounded.'
}
];
flowfileOutboundPolicyOptions: SelectOption[] = [
{
text: 'Stream When Available',
value: 'STREAM_WHEN_AVAILABLE',
description:
'FlowFiles that are queued up to be transferred out of a Process Group by an Output Port will be transferred out ' +
'of the Process Group as soon as they are available.'
},
{
text: 'Batch Output',
value: 'BATCH_OUTPUT',
description:
'FlowFiles that are queued up to be transferred out of a Process Group by an Output Port will remain queued until ' +
'all FlowFiles in the Process Group are ready to be transferred out of the group. The FlowFiles will then be transferred ' +
'out of the group. This setting will be ignored if the FlowFile Concurrency is Unbounded.'
}
];
constructor(
@Inject(MAT_DIALOG_DATA) public request: EditComponentDialogRequest,
private formBuilder: FormBuilder,
private client: Client
) {
this.parameterContextsOptions.push({
text: 'No parameter context',
value: null
});
this.editProcessGroupForm = this.formBuilder.group({
name: new FormControl(request.entity.component.name, Validators.required),
applyParameterContextRecursively: new FormControl(false),
executionEngine: new FormControl(request.entity.component.executionEngine, Validators.required),
flowfileConcurrency: new FormControl(request.entity.component.flowfileConcurrency, Validators.required),
flowfileOutboundPolicy: new FormControl(
request.entity.component.flowfileOutboundPolicy,
Validators.required
),
defaultFlowFileExpiration: new FormControl(
request.entity.component.defaultFlowFileExpiration,
Validators.required
),
defaultBackPressureObjectThreshold: new FormControl(
request.entity.component.defaultBackPressureObjectThreshold,
Validators.required
),
defaultBackPressureDataSizeThreshold: new FormControl(
request.entity.component.defaultBackPressureDataSizeThreshold,
Validators.required
),
logFileSuffix: new FormControl(request.entity.component.logFileSuffix),
comments: new FormControl(request.entity.component.comments)
});
}
getSelectOptionTipData(option: SelectOption): TextTipInput {
return {
// @ts-ignore
text: option.description
};
}
submitForm() {
let updateStrategy: string = 'DIRECT_CHILDREN';
if (this.editProcessGroupForm.get('applyParameterContextRecursively')?.value) {
updateStrategy = 'ALL_DESCENDANTS';
}
const payload: any = {
revision: this.client.getRevision(this.request.entity),
processGroupUpdateStrategy: updateStrategy,
component: {
id: this.request.entity.id,
name: this.editProcessGroupForm.get('name')?.value,
executionEngine: this.editProcessGroupForm.get('executionEngine')?.value,
flowfileConcurrency: this.editProcessGroupForm.get('flowfileConcurrency')?.value,
flowfileOutboundPolicy: this.editProcessGroupForm.get('flowfileOutboundPolicy')?.value,
defaultFlowFileExpiration: this.editProcessGroupForm.get('defaultFlowFileExpiration')?.value,
defaultBackPressureObjectThreshold: this.editProcessGroupForm.get('defaultBackPressureObjectThreshold')
?.value,
defaultBackPressureDataSizeThreshold: this.editProcessGroupForm.get(
'defaultBackPressureDataSizeThreshold'
)?.value,
logFileSuffix: this.editProcessGroupForm.get('logFileSuffix')?.value,
parameterContext: {
id: this.editProcessGroupForm.get('parameterContext')?.value
},
comments: this.editProcessGroupForm.get('comments')?.value
}
};
this.editProcessGroup.next(payload);
}
}