Save attributes strategies: merge UI

This commit is contained in:
Dmytro Skarzhynets 2025-02-28 15:31:40 +02:00 committed by GitHub
commit 5e9d79a7f8
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 257 additions and 15 deletions

View File

@ -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>

View File

@ -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});
}
}

View File

@ -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>

View File

@ -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});
}
}
}

View File

@ -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,
}

View File

@ -53,6 +53,7 @@
<tb-advanced-persistence-settings
class="mb-4"
formControlName="advanced"
timeseries latest webSockets
></tb-advanced-persistence-settings>
}
</div>

View File

@ -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",