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">
|
<section [formGroup]="persistenceForm" class="tb-form-panel no-border no-padding">
|
||||||
<tb-example-hint
|
<tb-example-hint
|
||||||
[hintText]="'rule-node-config.save-time-series.advanced-settings-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-example-hint>
|
||||||
<tb-advanced-persistence-setting-row
|
<tb-advanced-persistence-setting-row *ngIf="timeseries"
|
||||||
formControlName="timeseries"
|
formControlName="timeseries"
|
||||||
title="{{ 'rule-node-config.save-time-series.time-series' | translate }}"
|
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
|
<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"
|
formControlName="latest"
|
||||||
title="{{ 'rule-node-config.save-time-series.latest' | translate }}"
|
title="{{ 'rule-node-config.save-time-series.latest' | translate }}"
|
||||||
></tb-advanced-persistence-setting-row>
|
></tb-advanced-persistence-setting-row>
|
||||||
<tb-advanced-persistence-setting-row
|
<tb-advanced-persistence-setting-row *ngIf="webSockets"
|
||||||
formControlName="webSockets"
|
formControlName="webSockets"
|
||||||
title="{{ 'rule-node-config.save-time-series.web-sockets' | translate }}"
|
title="{{ 'rule-node-config.save-time-series.web-sockets' | translate }}"
|
||||||
></tb-advanced-persistence-setting-row>
|
></tb-advanced-persistence-setting-row>
|
||||||
|
|||||||
@ -19,12 +19,15 @@ import {
|
|||||||
FormBuilder,
|
FormBuilder,
|
||||||
NG_VALIDATORS,
|
NG_VALIDATORS,
|
||||||
NG_VALUE_ACCESSOR,
|
NG_VALUE_ACCESSOR,
|
||||||
|
UntypedFormGroup,
|
||||||
ValidationErrors,
|
ValidationErrors,
|
||||||
Validator
|
Validator
|
||||||
} from '@angular/forms';
|
} 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 { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
||||||
import { AdvancedProcessingStrategy } from '@home/components/rule-node/action/timeseries-config.models';
|
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({
|
@Component({
|
||||||
selector: 'tb-advanced-persistence-settings',
|
selector: 'tb-advanced-persistence-settings',
|
||||||
@ -39,19 +42,48 @@ import { AdvancedProcessingStrategy } from '@home/components/rule-node/action/ti
|
|||||||
multi: true
|
multi: true
|
||||||
}]
|
}]
|
||||||
})
|
})
|
||||||
export class AdvancedPersistenceSettingComponent implements ControlValueAccessor, Validator {
|
export class AdvancedPersistenceSettingComponent implements OnInit, ControlValueAccessor, Validator {
|
||||||
|
|
||||||
persistenceForm = this.fb.group({
|
@Input()
|
||||||
timeseries: [null],
|
@coerceBoolean()
|
||||||
latest: [null],
|
timeseries = false;
|
||||||
webSockets: [null]
|
|
||||||
});
|
@Input()
|
||||||
|
@coerceBoolean()
|
||||||
|
attribute = false;
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
@coerceBoolean()
|
||||||
|
latest = false;
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
@coerceBoolean()
|
||||||
|
webSockets = false;
|
||||||
|
|
||||||
|
persistenceForm: UntypedFormGroup;
|
||||||
|
|
||||||
private propagateChange: (value: any) => void = () => {};
|
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(
|
this.persistenceForm.valueChanges.pipe(
|
||||||
takeUntilDestroyed()
|
takeUntilDestroyed(this.destroyRef)
|
||||||
).subscribe(value => this.propagateChange(value));
|
).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});
|
this.persistenceForm.patchValue(value, {emitEvent: false});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -16,6 +16,47 @@
|
|||||||
|
|
||||||
-->
|
-->
|
||||||
<section [formGroup]="attributesConfigForm" class="tb-form-panel no-border no-padding">
|
<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">
|
<div class="tb-form-panel stroked">
|
||||||
<tb-example-hint [hintText]="'rule-node-config.attributes-scope-hint'">
|
<tb-example-hint [hintText]="'rule-node-config.attributes-scope-hint'">
|
||||||
</tb-example-hint>
|
</tb-example-hint>
|
||||||
|
|||||||
@ -15,10 +15,22 @@
|
|||||||
///
|
///
|
||||||
|
|
||||||
import { Component } from '@angular/core';
|
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 { RuleNodeConfiguration, RuleNodeConfigurationComponent } from '@shared/models/rule-node.models';
|
||||||
import { AttributeScope, telemetryTypeTranslations } from '@app/shared/models/telemetry/telemetry.models';
|
import { AttributeScope, telemetryTypeTranslations } from '@app/shared/models/telemetry/telemetry.models';
|
||||||
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
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({
|
@Component({
|
||||||
selector: 'tb-action-node-attributes-config',
|
selector: 'tb-action-node-attributes-config',
|
||||||
@ -31,6 +43,12 @@ export class AttributesConfigComponent extends RuleNodeConfigurationComponent {
|
|||||||
attributeScopes = Object.keys(AttributeScope);
|
attributeScopes = Object.keys(AttributeScope);
|
||||||
telemetryTypeTranslationsMap = telemetryTypeTranslations;
|
telemetryTypeTranslationsMap = telemetryTypeTranslations;
|
||||||
|
|
||||||
|
PersistenceType = ProcessingType;
|
||||||
|
persistenceStrategies = [ProcessingType.ON_EVERY_MESSAGE, ProcessingType.DEDUPLICATE, ProcessingType.WEBSOCKETS_ONLY];
|
||||||
|
PersistenceTypeTranslationMap = ProcessingTypeTranslationMap;
|
||||||
|
|
||||||
|
maxDeduplicateTime = maxDeduplicateTimeSecs;
|
||||||
|
|
||||||
attributesConfigForm: UntypedFormGroup;
|
attributesConfigForm: UntypedFormGroup;
|
||||||
|
|
||||||
constructor(private fb: UntypedFormBuilder) {
|
constructor(private fb: UntypedFormBuilder) {
|
||||||
@ -41,8 +59,64 @@ export class AttributesConfigComponent extends RuleNodeConfigurationComponent {
|
|||||||
return this.attributesConfigForm;
|
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) {
|
protected onConfigurationSet(configuration: RuleNodeConfiguration) {
|
||||||
this.attributesConfigForm = this.fb.group({
|
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]],
|
scope: [configuration ? configuration.scope : null, [Validators.required]],
|
||||||
notifyDevice: [configuration ? configuration.notifyDevice : true, []],
|
notifyDevice: [configuration ? configuration.notifyDevice : true, []],
|
||||||
sendAttributesUpdatedNotification: [configuration ? configuration.sendAttributesUpdatedNotification : false, []],
|
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
|
<tb-advanced-persistence-settings
|
||||||
class="mb-4"
|
class="mb-4"
|
||||||
formControlName="advanced"
|
formControlName="advanced"
|
||||||
|
timeseries latest webSockets
|
||||||
></tb-advanced-persistence-settings>
|
></tb-advanced-persistence-settings>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -5155,6 +5155,22 @@
|
|||||||
"latest": "Latest values",
|
"latest": "Latest values",
|
||||||
"web-sockets": "WebSockets"
|
"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-val": {
|
||||||
"key": "Key",
|
"key": "Key",
|
||||||
"value": "Value",
|
"value": "Value",
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user