Save attributes strategies: merge UI
This commit is contained in:
commit
5e9d79a7f8
@ -18,18 +18,22 @@
|
||||
<section [formGroup]="persistenceForm" class="tb-form-panel no-border no-padding">
|
||||
<tb-example-hint
|
||||
[hintText]="'rule-node-config.save-time-series.advanced-settings-hint'"
|
||||
[popupHelpLink]="'rulenode/save_timeseries_node_advanced'"
|
||||
[popupHelpLink]="timeseries ? 'rulenode/save_timeseries_node_advanced' : null"
|
||||
>
|
||||
</tb-example-hint>
|
||||
<tb-advanced-persistence-setting-row
|
||||
<tb-advanced-persistence-setting-row *ngIf="timeseries"
|
||||
formControlName="timeseries"
|
||||
title="{{ 'rule-node-config.save-time-series.time-series' | translate }}"
|
||||
></tb-advanced-persistence-setting-row>
|
||||
<tb-advanced-persistence-setting-row
|
||||
<tb-advanced-persistence-setting-row *ngIf="attribute"
|
||||
formControlName="attribute"
|
||||
title="{{ 'rule-node-config.save-attribute.attribute' | translate }}"
|
||||
></tb-advanced-persistence-setting-row>
|
||||
<tb-advanced-persistence-setting-row *ngIf="latest"
|
||||
formControlName="latest"
|
||||
title="{{ 'rule-node-config.save-time-series.latest' | translate }}"
|
||||
></tb-advanced-persistence-setting-row>
|
||||
<tb-advanced-persistence-setting-row
|
||||
<tb-advanced-persistence-setting-row *ngIf="webSockets"
|
||||
formControlName="webSockets"
|
||||
title="{{ 'rule-node-config.save-time-series.web-sockets' | translate }}"
|
||||
></tb-advanced-persistence-setting-row>
|
||||
|
||||
@ -19,12 +19,15 @@ import {
|
||||
FormBuilder,
|
||||
NG_VALIDATORS,
|
||||
NG_VALUE_ACCESSOR,
|
||||
UntypedFormGroup,
|
||||
ValidationErrors,
|
||||
Validator
|
||||
} from '@angular/forms';
|
||||
import { Component, forwardRef } from '@angular/core';
|
||||
import { Component, DestroyRef, forwardRef, Input, OnInit } from '@angular/core';
|
||||
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
||||
import { AdvancedProcessingStrategy } from '@home/components/rule-node/action/timeseries-config.models';
|
||||
import { coerceBoolean } from '@shared/decorators/coercion';
|
||||
import { AttributeAdvancedProcessingStrategy } from '@home/components/rule-node/action/attributes-config.model';
|
||||
|
||||
@Component({
|
||||
selector: 'tb-advanced-persistence-settings',
|
||||
@ -39,19 +42,48 @@ import { AdvancedProcessingStrategy } from '@home/components/rule-node/action/ti
|
||||
multi: true
|
||||
}]
|
||||
})
|
||||
export class AdvancedPersistenceSettingComponent implements ControlValueAccessor, Validator {
|
||||
export class AdvancedPersistenceSettingComponent implements OnInit, ControlValueAccessor, Validator {
|
||||
|
||||
persistenceForm = this.fb.group({
|
||||
timeseries: [null],
|
||||
latest: [null],
|
||||
webSockets: [null]
|
||||
});
|
||||
@Input()
|
||||
@coerceBoolean()
|
||||
timeseries = false;
|
||||
|
||||
@Input()
|
||||
@coerceBoolean()
|
||||
attribute = false;
|
||||
|
||||
@Input()
|
||||
@coerceBoolean()
|
||||
latest = false;
|
||||
|
||||
@Input()
|
||||
@coerceBoolean()
|
||||
webSockets = false;
|
||||
|
||||
persistenceForm: UntypedFormGroup;
|
||||
|
||||
private propagateChange: (value: any) => void = () => {};
|
||||
|
||||
constructor(private fb: FormBuilder) {
|
||||
constructor(private fb: FormBuilder,
|
||||
private destroyRef: DestroyRef) {
|
||||
}
|
||||
|
||||
ngOnInit() {
|
||||
this.persistenceForm = this.fb.group({});
|
||||
if (this.timeseries) {
|
||||
this.persistenceForm.addControl('timeseries', this.fb.control(null, []));
|
||||
}
|
||||
if (this.attribute) {
|
||||
this.persistenceForm.addControl('attribute', this.fb.control(null, []));
|
||||
}
|
||||
if (this.attribute) {
|
||||
this.persistenceForm.addControl('latest', this.fb.control(null, []));
|
||||
}
|
||||
if (this.attribute) {
|
||||
this.persistenceForm.addControl('webSockets', this.fb.control(null, []));
|
||||
}
|
||||
this.persistenceForm.valueChanges.pipe(
|
||||
takeUntilDestroyed()
|
||||
takeUntilDestroyed(this.destroyRef)
|
||||
).subscribe(value => this.propagateChange(value));
|
||||
}
|
||||
|
||||
@ -76,7 +108,7 @@ export class AdvancedPersistenceSettingComponent implements ControlValueAccessor
|
||||
};
|
||||
}
|
||||
|
||||
writeValue(value: AdvancedProcessingStrategy) {
|
||||
writeValue(value: AdvancedProcessingStrategy | AttributeAdvancedProcessingStrategy) {
|
||||
this.persistenceForm.patchValue(value, {emitEvent: false});
|
||||
}
|
||||
}
|
||||
|
||||
@ -16,6 +16,47 @@
|
||||
|
||||
-->
|
||||
<section [formGroup]="attributesConfigForm" class="tb-form-panel no-border no-padding">
|
||||
<div class="tb-form-panel stroked no-padding-bottom no-gap" formGroupName="processingSettings">
|
||||
<div class="mb-4 flex flex-row items-center justify-between">
|
||||
<div class="tb-form-panel-title" tb-hint-tooltip-icon="{{ 'rule-node-config.save-attribute.processing-settings-hint' | translate}}" translate>
|
||||
rule-node-config.save-attribute.processing-settings
|
||||
</div>
|
||||
<tb-toggle-select appearance="fill" selectMediaBreakpoint="xs"
|
||||
formControlName="isAdvanced">
|
||||
<tb-toggle-option [value]=false>{{ 'rule-node-config.basic-mode' | translate}}</tb-toggle-option>
|
||||
<tb-toggle-option [value]=true>{{ 'rule-node-config.advanced-mode' | translate }}</tb-toggle-option>
|
||||
</tb-toggle-select>
|
||||
</div>
|
||||
@if(!attributesConfigForm.get('processingSettings.isAdvanced').value) {
|
||||
<mat-form-field>
|
||||
<mat-label translate>rule-node-config.save-attribute.strategy</mat-label>
|
||||
<mat-select formControlName="type">
|
||||
@for (strategy of persistenceStrategies; track strategy) {
|
||||
<mat-option [value]="strategy">{{ PersistenceTypeTranslationMap.get(strategy) | translate }}</mat-option>
|
||||
}
|
||||
</mat-select>
|
||||
</mat-form-field>
|
||||
|
||||
@if(attributesConfigForm.get('processingSettings.type').value === PersistenceType.DEDUPLICATE) {
|
||||
<tb-time-unit-input
|
||||
required
|
||||
labelText="{{ 'rule-node-config.save-attribute.deduplication-interval' | translate }}"
|
||||
requiredText="{{ 'rule-node-config.save-attribute.deduplication-interval-required' | translate }}"
|
||||
minErrorText="{{ 'rule-node-config.save-attribute.deduplication-interval-min-max-range' | translate }}"
|
||||
maxErrorText="{{ 'rule-node-config.save-attribute.deduplication-interval-min-max-range' | translate }}"
|
||||
[maxTime]="maxDeduplicateTime"
|
||||
[minTime]="1"
|
||||
formControlName="deduplicationIntervalSecs">
|
||||
</tb-time-unit-input>
|
||||
}
|
||||
} @else {
|
||||
<tb-advanced-persistence-settings
|
||||
class="mb-4"
|
||||
formControlName="advanced"
|
||||
attribute webSockets
|
||||
></tb-advanced-persistence-settings>
|
||||
}
|
||||
</div>
|
||||
<div class="tb-form-panel stroked">
|
||||
<tb-example-hint [hintText]="'rule-node-config.attributes-scope-hint'">
|
||||
</tb-example-hint>
|
||||
|
||||
@ -15,10 +15,22 @@
|
||||
///
|
||||
|
||||
import { Component } from '@angular/core';
|
||||
import { UntypedFormBuilder, UntypedFormGroup, Validators } from '@angular/forms';
|
||||
import { FormGroup, UntypedFormBuilder, UntypedFormGroup, Validators } from '@angular/forms';
|
||||
import { RuleNodeConfiguration, RuleNodeConfigurationComponent } from '@shared/models/rule-node.models';
|
||||
import { AttributeScope, telemetryTypeTranslations } from '@app/shared/models/telemetry/telemetry.models';
|
||||
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
||||
import {
|
||||
maxDeduplicateTimeSecs,
|
||||
ProcessingSettings,
|
||||
ProcessingSettingsForm,
|
||||
ProcessingType,
|
||||
ProcessingTypeTranslationMap
|
||||
} from '@home/components/rule-node/action/timeseries-config.models';
|
||||
import {
|
||||
AttributeNodeConfiguration,
|
||||
AttributeNodeConfigurationForm,
|
||||
defaultAttributeAdvancedPersistenceStrategy
|
||||
} from '@home/components/rule-node/action/attributes-config.model';
|
||||
|
||||
@Component({
|
||||
selector: 'tb-action-node-attributes-config',
|
||||
@ -31,6 +43,12 @@ export class AttributesConfigComponent extends RuleNodeConfigurationComponent {
|
||||
attributeScopes = Object.keys(AttributeScope);
|
||||
telemetryTypeTranslationsMap = telemetryTypeTranslations;
|
||||
|
||||
PersistenceType = ProcessingType;
|
||||
persistenceStrategies = [ProcessingType.ON_EVERY_MESSAGE, ProcessingType.DEDUPLICATE, ProcessingType.WEBSOCKETS_ONLY];
|
||||
PersistenceTypeTranslationMap = ProcessingTypeTranslationMap;
|
||||
|
||||
maxDeduplicateTime = maxDeduplicateTimeSecs;
|
||||
|
||||
attributesConfigForm: UntypedFormGroup;
|
||||
|
||||
constructor(private fb: UntypedFormBuilder) {
|
||||
@ -41,8 +59,64 @@ export class AttributesConfigComponent extends RuleNodeConfigurationComponent {
|
||||
return this.attributesConfigForm;
|
||||
}
|
||||
|
||||
protected validatorTriggers(): string[] {
|
||||
return ['processingSettings.isAdvanced', 'processingSettings.type'];
|
||||
}
|
||||
|
||||
protected prepareInputConfig(config: AttributeNodeConfiguration): AttributeNodeConfigurationForm {
|
||||
let processingSettings: ProcessingSettingsForm;
|
||||
if (config?.processingSettings) {
|
||||
const isAdvanced = config?.processingSettings?.type === ProcessingType.ADVANCED;
|
||||
processingSettings = {
|
||||
type: isAdvanced ? ProcessingType.ON_EVERY_MESSAGE : config.processingSettings.type,
|
||||
isAdvanced: isAdvanced,
|
||||
deduplicationIntervalSecs: config.processingSettings?.deduplicationIntervalSecs ?? 60,
|
||||
advanced: isAdvanced ? config.processingSettings : defaultAttributeAdvancedPersistenceStrategy
|
||||
}
|
||||
} else {
|
||||
processingSettings = {
|
||||
type: ProcessingType.ON_EVERY_MESSAGE,
|
||||
isAdvanced: false,
|
||||
deduplicationIntervalSecs: 60,
|
||||
advanced: defaultAttributeAdvancedPersistenceStrategy
|
||||
};
|
||||
}
|
||||
return {
|
||||
...config,
|
||||
processingSettings: processingSettings
|
||||
}
|
||||
}
|
||||
|
||||
protected prepareOutputConfig(config: AttributeNodeConfigurationForm): AttributeNodeConfiguration {
|
||||
let processingSettings: ProcessingSettings;
|
||||
if (config.processingSettings.isAdvanced) {
|
||||
processingSettings = {
|
||||
...config.processingSettings.advanced,
|
||||
type: ProcessingType.ADVANCED
|
||||
};
|
||||
} else {
|
||||
processingSettings = {
|
||||
type: config.processingSettings.type,
|
||||
deduplicationIntervalSecs: config.processingSettings?.deduplicationIntervalSecs
|
||||
};
|
||||
}
|
||||
return {
|
||||
...config,
|
||||
processingSettings
|
||||
};
|
||||
}
|
||||
|
||||
protected onConfigurationSet(configuration: RuleNodeConfiguration) {
|
||||
this.attributesConfigForm = this.fb.group({
|
||||
processingSettings: this.fb.group({
|
||||
isAdvanced: [configuration?.processingSettings?.isAdvanced ?? false],
|
||||
type: [configuration?.processingSettings?.type ?? ProcessingType.ON_EVERY_MESSAGE],
|
||||
deduplicationIntervalSecs: [
|
||||
{value: configuration?.processingSettings?.deduplicationIntervalSecs ?? 60, disabled: true},
|
||||
[Validators.required, Validators.max(maxDeduplicateTimeSecs)]
|
||||
],
|
||||
advanced: [{value: null, disabled: true}]
|
||||
}),
|
||||
scope: [configuration ? configuration.scope : null, [Validators.required]],
|
||||
notifyDevice: [configuration ? configuration.notifyDevice : true, []],
|
||||
sendAttributesUpdatedNotification: [configuration ? configuration.sendAttributesUpdatedNotification : false, []],
|
||||
@ -62,4 +136,19 @@ export class AttributesConfigComponent extends RuleNodeConfigurationComponent {
|
||||
});
|
||||
}
|
||||
|
||||
protected updateValidators(emitEvent: boolean, _trigger?: string) {
|
||||
const processingForm = this.attributesConfigForm.get('processingSettings') as FormGroup;
|
||||
const isAdvanced: boolean = processingForm.get('isAdvanced').value;
|
||||
const type: ProcessingType = processingForm.get('type').value;
|
||||
if (!isAdvanced && type === ProcessingType.DEDUPLICATE) {
|
||||
processingForm.get('deduplicationIntervalSecs').enable({emitEvent});
|
||||
} else {
|
||||
processingForm.get('deduplicationIntervalSecs').disable({emitEvent});
|
||||
}
|
||||
if (isAdvanced) {
|
||||
processingForm.get('advanced').enable({emitEvent});
|
||||
} else {
|
||||
processingForm.get('advanced').disable({emitEvent});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -0,0 +1,59 @@
|
||||
///
|
||||
/// Copyright © 2016-2025 The Thingsboard Authors
|
||||
///
|
||||
/// Licensed 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 { DAY, SECOND } from '@shared/models/time/time.models';
|
||||
import { AttributeScope } from '@shared/models/telemetry/telemetry.models';
|
||||
import { BasicProcessingSettings, ProcessingType } from '@home/components/rule-node/action/timeseries-config.models';
|
||||
|
||||
export interface AttributeNodeConfiguration {
|
||||
processingSettings: AttributeProcessingSettings;
|
||||
scope: AttributeScope;
|
||||
notifyDevice: boolean;
|
||||
sendAttributesUpdatedNotification: boolean;
|
||||
updateAttributesOnlyOnValueChange: boolean;
|
||||
}
|
||||
|
||||
export interface AttributeNodeConfigurationForm extends Omit<AttributeNodeConfiguration, 'processingSettings'> {
|
||||
processingSettings: AttributeProcessingSettingsForm
|
||||
}
|
||||
|
||||
export type AttributeProcessingSettings = BasicProcessingSettings & Partial<AttributeDeduplicateProcessingStrategy> & Partial<AttributeAdvancedProcessingStrategy>;
|
||||
|
||||
export type AttributeProcessingSettingsForm = Omit<AttributeProcessingSettings, keyof AttributeAdvancedProcessingStrategy> & {
|
||||
isAdvanced: boolean;
|
||||
advanced?: Partial<AttributeAdvancedProcessingStrategy>;
|
||||
type: ProcessingType;
|
||||
};
|
||||
|
||||
export interface AttributeDeduplicateProcessingStrategy extends BasicProcessingSettings{
|
||||
deduplicationIntervalSecs: number;
|
||||
}
|
||||
|
||||
export interface AttributeAdvancedProcessingStrategy extends BasicProcessingSettings{
|
||||
attribute: AttributeAdvancedProcessingConfig;
|
||||
webSockets: AttributeAdvancedProcessingConfig;
|
||||
}
|
||||
|
||||
export type AttributeAdvancedProcessingConfig = WithOptional<AttributeDeduplicateProcessingStrategy, 'deduplicationIntervalSecs'>;
|
||||
|
||||
export const defaultAdvancedProcessingConfig: AttributeAdvancedProcessingConfig = {
|
||||
type: ProcessingType.ON_EVERY_MESSAGE
|
||||
}
|
||||
|
||||
export const defaultAttributeAdvancedPersistenceStrategy: Omit<AttributeAdvancedProcessingStrategy, 'type'> = {
|
||||
attribute: defaultAdvancedProcessingConfig,
|
||||
webSockets: defaultAdvancedProcessingConfig,
|
||||
}
|
||||
@ -53,6 +53,7 @@
|
||||
<tb-advanced-persistence-settings
|
||||
class="mb-4"
|
||||
formControlName="advanced"
|
||||
timeseries latest webSockets
|
||||
></tb-advanced-persistence-settings>
|
||||
}
|
||||
</div>
|
||||
|
||||
@ -5155,6 +5155,22 @@
|
||||
"latest": "Latest values",
|
||||
"web-sockets": "WebSockets"
|
||||
},
|
||||
"save-attribute": {
|
||||
"processing-settings": "Processing settings",
|
||||
"processing-settings-hint": "Define how incoming messages are processed. In Basic mode, select a preconfigured processing strategy or enable only WebSocket updates. Advanced mode allows you to select individual processing strategies for each action.",
|
||||
"advanced-settings-hint": "Be cautious when configuring processing strategies. Certain combinations can lead to unexpected behavior.",
|
||||
"strategy": "Strategy",
|
||||
"deduplication-interval": "Deduplication interval",
|
||||
"deduplication-interval-required": "Deduplication interval is required",
|
||||
"deduplication-interval-min-max-range": "Deduplication interval should be at least 1 second and at most 1 day",
|
||||
"strategy-type": {
|
||||
"every-message": "On every message",
|
||||
"skip": "Skip",
|
||||
"deduplicate": "Deduplicate",
|
||||
"web-sockets-only": "WebSockets only"
|
||||
},
|
||||
"attribute": "Attribute"
|
||||
},
|
||||
"key-val": {
|
||||
"key": "Key",
|
||||
"value": "Value",
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user