UI: Add new rule node and refactoring code

This commit is contained in:
Vladyslav_Prykhodko 2023-04-04 18:08:44 +03:00
parent a4ad8feb7f
commit ef94a8c198
24 changed files with 42 additions and 22 deletions

View File

@ -48,7 +48,7 @@ import static org.thingsboard.common.util.DonAsynchron.withCallback;
"(<code>messageId</code>, <code>requestId</code>) in the Message Metadata from the AWS SNS. " + "(<code>messageId</code>, <code>requestId</code>) in the Message Metadata from the AWS SNS. " +
"For example <b>requestId</b> field can be accessed with <code>metadata.requestId</code>.", "For example <b>requestId</b> field can be accessed with <code>metadata.requestId</code>.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeSnsConfig", configDirective = "tbExternalNodeSnsConfig",
iconUrl = "" iconUrl = ""
) )
public class TbSnsNode implements TbNode { public class TbSnsNode implements TbNode {

View File

@ -52,7 +52,7 @@ import static org.thingsboard.common.util.DonAsynchron.withCallback;
", <code>sequenceNumber</code>) in the Message Metadata from the AWS SQS." + ", <code>sequenceNumber</code>) in the Message Metadata from the AWS SQS." +
" For example <b>requestId</b> field can be accessed with <code>metadata.requestId</code>.", " For example <b>requestId</b> field can be accessed with <code>metadata.requestId</code>.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeSqsConfig", configDirective = "tbExternalNodeSqsConfig",
iconUrl = "" iconUrl = ""
) )
public class TbSqsNode implements TbNode { public class TbSqsNode implements TbNode {

View File

@ -50,7 +50,7 @@ import java.util.concurrent.TimeUnit;
"(<code>messageId</code> in the Message Metadata from the GCP PubSub. " + "(<code>messageId</code> in the Message Metadata from the GCP PubSub. " +
"<b>messageId</b> field can be accessed with <code>metadata.messageId</code>.", "<b>messageId</b> field can be accessed with <code>metadata.messageId</code>.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodePubSubConfig", configDirective = "tbExternalNodePubSubConfig",
iconUrl = "" iconUrl = ""
) )
public class TbPubSubNode implements TbNode { public class TbPubSubNode implements TbNode {

View File

@ -53,7 +53,7 @@ import java.util.Properties;
"Outbound message will contain response fields (<code>offset</code>, <code>partition</code>, <code>topic</code>)" + "Outbound message will contain response fields (<code>offset</code>, <code>partition</code>, <code>topic</code>)" +
" from the Kafka in the Message Metadata. For example <b>partition</b> field can be accessed with <code>metadata.partition</code>.", " from the Kafka in the Message Metadata. For example <b>partition</b> field can be accessed with <code>metadata.partition</code>.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeKafkaConfig", configDirective = "tbExternalNodeKafkaConfig",
iconUrl = "" iconUrl = ""
) )
public class TbKafkaNode implements TbNode { public class TbKafkaNode implements TbNode {

View File

@ -44,7 +44,7 @@ import static org.thingsboard.common.util.DonAsynchron.withCallback;
" where created using <code>to Email</code> transformation Node, please connect this Node " + " where created using <code>to Email</code> transformation Node, please connect this Node " +
"with <code>to Email</code> Node using <code>Successful</code> chain.", "with <code>to Email</code> Node using <code>Successful</code> chain.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeSendEmailConfig", configDirective = "tbExternalNodeSendEmailConfig",
icon = "send" icon = "send"
) )
public class TbSendEmailNode implements TbNode { public class TbSendEmailNode implements TbNode {

View File

@ -50,7 +50,7 @@ import java.util.concurrent.TimeoutException;
nodeDescription = "Publish messages to the MQTT broker", nodeDescription = "Publish messages to the MQTT broker",
nodeDetails = "Will publish message payload to the MQTT broker with QoS <b>AT_LEAST_ONCE</b>.", nodeDetails = "Will publish message payload to the MQTT broker with QoS <b>AT_LEAST_ONCE</b>.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeMqttConfig", configDirective = "tbExternalNodeMqttConfig",
icon = "call_split" icon = "call_split"
) )
public class TbMqttNode implements TbNode { public class TbMqttNode implements TbNode {

View File

@ -44,7 +44,7 @@ import javax.net.ssl.SSLException;
nodeDescription = "Publish messages to the Azure IoT Hub", nodeDescription = "Publish messages to the Azure IoT Hub",
nodeDetails = "Will publish message payload to the Azure IoT Hub with QoS <b>AT_LEAST_ONCE</b>.", nodeDetails = "Will publish message payload to the Azure IoT Hub with QoS <b>AT_LEAST_ONCE</b>.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeAzureIotHubConfig" configDirective = "tbExternalNodeAzureIotHubConfig"
) )
public class TbAzureIotHubNode extends TbMqttNode { public class TbAzureIotHubNode extends TbMqttNode {
@Override @Override

View File

@ -39,7 +39,9 @@ import java.util.concurrent.ExecutionException;
configClazz = TbNotificationNodeConfiguration.class, configClazz = TbNotificationNodeConfiguration.class,
nodeDescription = "Sends notification to targets using the template", nodeDescription = "Sends notification to targets using the template",
nodeDetails = "Will send notification to the specified targets using the template", nodeDetails = "Will send notification to the specified targets using the template",
uiResources = {"static/rulenode/rulenode-core-config.js"} uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbExternalNodeNotificationConfig",
icon = "notifications"
) )
public class TbNotificationNode implements TbNode { public class TbNotificationNode implements TbNode {

View File

@ -33,7 +33,9 @@ import java.util.concurrent.ExecutionException;
configClazz = TbSlackNodeConfiguration.class, configClazz = TbSlackNodeConfiguration.class,
nodeDescription = "Send message via Slack", nodeDescription = "Send message via Slack",
nodeDetails = "Sends message to a Slack channel or user", nodeDetails = "Sends message to a Slack channel or user",
uiResources = {"static/rulenode/rulenode-core-config.js"} uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbExternalNodeSlackConfig",
iconUrl = ""
) )
public class TbSlackNode implements TbNode { public class TbSlackNode implements TbNode {

View File

@ -40,6 +40,7 @@ public class TbSlackNodeConfiguration implements NodeConfiguration<TbSlackNodeCo
@Override @Override
public TbSlackNodeConfiguration defaultConfiguration() { public TbSlackNodeConfiguration defaultConfiguration() {
TbSlackNodeConfiguration config = new TbSlackNodeConfiguration(); TbSlackNodeConfiguration config = new TbSlackNodeConfiguration();
config.setUseSystemSettings(true);
config.setBotToken("xoxb-"); config.setBotToken("xoxb-");
config.setMessageTemplate("Device ${deviceId}: temperature is $[temperature]"); config.setMessageTemplate("Device ${deviceId}: temperature is $[temperature]");
config.setConversationType(SlackConversationType.PUBLIC_CHANNEL); config.setConversationType(SlackConversationType.PUBLIC_CHANNEL);

View File

@ -45,7 +45,7 @@ import static org.thingsboard.common.util.DonAsynchron.withCallback;
nodeDescription = "Publish messages to the RabbitMQ", nodeDescription = "Publish messages to the RabbitMQ",
nodeDetails = "Will publish message payload to RabbitMQ queue.", nodeDetails = "Will publish message payload to RabbitMQ queue.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeRabbitMqConfig", configDirective = "tbExternalNodeRabbitMqConfig",
iconUrl = "" iconUrl = ""
) )
public class TbRabbitMqNode implements TbNode { public class TbRabbitMqNode implements TbNode {

View File

@ -40,7 +40,7 @@ import org.thingsboard.server.common.msg.TbMsg;
"<br/><b>Note-</b> if you use system proxy properties, the next system proxy properties should be added: \"http.proxyHost\" and \"http.proxyPort\" or \"https.proxyHost\" and \"https.proxyPort\" or \"socksProxyHost\" and \"socksProxyPort\"," + "<br/><b>Note-</b> if you use system proxy properties, the next system proxy properties should be added: \"http.proxyHost\" and \"http.proxyPort\" or \"https.proxyHost\" and \"https.proxyPort\" or \"socksProxyHost\" and \"socksProxyPort\"," +
"and if your proxy with auth, the next ones should be added: \"tb.proxy.user\" and \"tb.proxy.password\" to the thingsboard.conf file.", "and if your proxy with auth, the next ones should be added: \"tb.proxy.user\" and \"tb.proxy.password\" to the thingsboard.conf file.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeRestApiCallConfig", configDirective = "tbExternalNodeRestApiCallConfig",
iconUrl = "" iconUrl = ""
) )
public class TbRestApiCallNode implements TbNode { public class TbRestApiCallNode implements TbNode {

View File

@ -37,7 +37,7 @@ import static org.thingsboard.common.util.DonAsynchron.withCallback;
nodeDescription = "Sends SMS message via SMS provider.", nodeDescription = "Sends SMS message via SMS provider.",
nodeDetails = "Will send SMS message by populating target phone numbers and sms message fields using values derived from message metadata.", nodeDetails = "Will send SMS message by populating target phone numbers and sms message fields using values derived from message metadata.",
uiResources = {"static/rulenode/rulenode-core-config.js"}, uiResources = {"static/rulenode/rulenode-core-config.js"},
configDirective = "tbActionNodeSendSmsConfig", configDirective = "tbExternalNodeSendSmsConfig",
icon = "sms" icon = "sms"
) )
public class TbSendSmsNode implements TbNode { public class TbSendSmsNode implements TbNode {

View File

@ -93,9 +93,12 @@ export class NotificationService {
return this.http.post<NotificationSettings>('/api/notification/settings', notificationSettings, defaultHttpOptionsFromConfig(config)); return this.http.post<NotificationSettings>('/api/notification/settings', notificationSettings, defaultHttpOptionsFromConfig(config));
} }
public listSlackConversations(type: SlackChanelType, config?: RequestConfig): Observable<Array<SlackConversation>> { public listSlackConversations(type: SlackChanelType, token?: string, config?: RequestConfig): Observable<Array<SlackConversation>> {
return this.http.get<Array<SlackConversation>>(`/api/notification/slack/conversations?type=${type}`, let url = `/api/notification/slack/conversations?type=${type}`;
defaultHttpOptionsFromConfig(config)); if (isNotEmptyStr(token)) {
url += `&token=${token}`;
}
return this.http.get<Array<SlackConversation>>(url, defaultHttpOptionsFromConfig(config));
} }
public saveNotificationRule(notificationRule: NotificationRule, config?: RequestConfig): Observable<NotificationRule> { public saveNotificationRule(notificationRule: NotificationRule, config?: RequestConfig): Observable<NotificationRule> {

View File

@ -153,6 +153,8 @@ import * as ColorInputComponent from '@shared/components/color-input.component';
import * as MaterialIconSelectComponent from '@shared/components/material-icon-select.component'; import * as MaterialIconSelectComponent from '@shared/components/material-icon-select.component';
import * as NodeScriptTestDialogComponent from '@shared/components/dialog/node-script-test-dialog.component'; import * as NodeScriptTestDialogComponent from '@shared/components/dialog/node-script-test-dialog.component';
import * as JsonFormComponent from '@shared/components/json-form/json-form.component'; import * as JsonFormComponent from '@shared/components/json-form/json-form.component';
import * as NotificationComponent from '@shared/components/notification/notification.component';
import * as TemplateAutocompleteComponent from '@shared/components/notification/template-autocomplete.component';
import * as ImageInputComponent from '@shared/components/image-input.component'; import * as ImageInputComponent from '@shared/components/image-input.component';
import * as FileInputComponent from '@shared/components/file-input.component'; import * as FileInputComponent from '@shared/components/file-input.component';
import * as MessageTypeAutocompleteComponent from '@shared/components/message-type-autocomplete.component'; import * as MessageTypeAutocompleteComponent from '@shared/components/message-type-autocomplete.component';
@ -170,6 +172,7 @@ import * as WidgetsBundleSearchComponent from '@shared/components/widgets-bundle
import * as CopyButtonComponent from '@shared/components/button/copy-button.component'; import * as CopyButtonComponent from '@shared/components/button/copy-button.component';
import * as TogglePasswordComponent from '@shared/components/button/toggle-password.component'; import * as TogglePasswordComponent from '@shared/components/button/toggle-password.component';
import * as ProtobufContentComponent from '@shared/components/protobuf-content.component'; import * as ProtobufContentComponent from '@shared/components/protobuf-content.component';
import * as SlackConversationAutocompleteComponent from '@shared/components/slack-conversation-autocomplete.component';
import * as AddEntityDialogComponent from '@home/components/entity/add-entity-dialog.component'; import * as AddEntityDialogComponent from '@home/components/entity/add-entity-dialog.component';
import * as EntitiesTableComponent from '@home/components/entity/entities-table.component'; import * as EntitiesTableComponent from '@home/components/entity/entities-table.component';
@ -441,6 +444,8 @@ class ModulesMap implements IModulesMap {
'@shared/components/material-icon-select.component': MaterialIconSelectComponent, '@shared/components/material-icon-select.component': MaterialIconSelectComponent,
'@shared/components/dialog/node-script-test-dialog.component': NodeScriptTestDialogComponent, '@shared/components/dialog/node-script-test-dialog.component': NodeScriptTestDialogComponent,
'@shared/components/json-form/json-form.component': JsonFormComponent, '@shared/components/json-form/json-form.component': JsonFormComponent,
'@shared/components/notification/notification.component': NotificationComponent,
'@shared/components/notification/template-autocomplete.component': TemplateAutocompleteComponent,
'@shared/components/image-input.component': ImageInputComponent, '@shared/components/image-input.component': ImageInputComponent,
'@shared/components/file-input.component': FileInputComponent, '@shared/components/file-input.component': FileInputComponent,
'@shared/components/message-type-autocomplete.component': MessageTypeAutocompleteComponent, '@shared/components/message-type-autocomplete.component': MessageTypeAutocompleteComponent,
@ -458,6 +463,7 @@ class ModulesMap implements IModulesMap {
'@shared/components/button/copy-button.component': CopyButtonComponent, '@shared/components/button/copy-button.component': CopyButtonComponent,
'@shared/components/button/toggle-password.component': TogglePasswordComponent, '@shared/components/button/toggle-password.component': TogglePasswordComponent,
'@shared/components/protobuf-content.component': ProtobufContentComponent, '@shared/components/protobuf-content.component': ProtobufContentComponent,
'@shared/components/slack-conversation-autocomplete.component': SlackConversationAutocompleteComponent,
'@home/components/entity/add-entity-dialog.component': AddEntityDialogComponent, '@home/components/entity/add-entity-dialog.component': AddEntityDialogComponent,
'@home/components/entity/entities-table.component': EntitiesTableComponent, '@home/components/entity/entities-table.component': EntitiesTableComponent,

View File

@ -27,7 +27,6 @@ import {
RecipientNotificationDialogComponent RecipientNotificationDialogComponent
} from '@home/pages/notification/recipient/recipient-notification-dialog.component'; } from '@home/pages/notification/recipient/recipient-notification-dialog.component';
import { RecipientTableHeaderComponent } from '@home/pages/notification/recipient/recipient-table-header.component'; import { RecipientTableHeaderComponent } from '@home/pages/notification/recipient/recipient-table-header.component';
import { TemplateAutocompleteComponent } from '@home/pages/notification/template/template-autocomplete.component';
import { import {
TemplateNotificationDialogComponent TemplateNotificationDialogComponent
} from '@home/pages/notification/template/template-notification-dialog.component'; } from '@home/pages/notification/template/template-notification-dialog.component';
@ -45,7 +44,6 @@ import { RuleTableHeaderComponent } from '@home/pages/notification/rule/rule-tab
SentNotificationDialogComponent, SentNotificationDialogComponent,
RecipientNotificationDialogComponent, RecipientNotificationDialogComponent,
RecipientTableHeaderComponent, RecipientTableHeaderComponent,
TemplateAutocompleteComponent,
TemplateNotificationDialogComponent, TemplateNotificationDialogComponent,
TemplateTableHeaderComponent, TemplateTableHeaderComponent,
EscalationFormComponent, EscalationFormComponent,

View File

@ -20,3 +20,5 @@ export * from './page.component';
export * from './dialog.component'; export * from './dialog.component';
export * from './js-func.component'; export * from './js-func.component';
export * from './script-lang.component'; export * from './script-lang.component';
export * from './slack-conversation-autocomplete.component';
export * from './notification/template-autocomplete.component';

View File

@ -68,6 +68,9 @@ export class SlackConversationAutocompleteComponent implements ControlValueAcces
@Input() @Input()
slackChanelType: SlackChanelType; slackChanelType: SlackChanelType;
@Input()
token: string;
@ViewChild('slackInput', {static: true}) slackInput: ElementRef; @ViewChild('slackInput', {static: true}) slackInput: ElementRef;
slackConversation$: Observable<Array<SlackConversation>>; slackConversation$: Observable<Array<SlackConversation>>;
@ -119,7 +122,7 @@ export class SlackConversationAutocompleteComponent implements ControlValueAcces
this.clear(); this.clear();
} }
}), }),
map(value => value ? (typeof value === 'string' ? value : value.name) : ''), map(value => value ? (typeof value === 'string' ? value : value.title) : ''),
switchMap(name => this.fetchSlackConversation(name)), switchMap(name => this.fetchSlackConversation(name)),
share() share()
); );
@ -129,7 +132,7 @@ export class SlackConversationAutocompleteComponent implements ControlValueAcces
for (const propName of Object.keys(changes)) { for (const propName of Object.keys(changes)) {
const change = changes[propName]; const change = changes[propName];
if (!change.firstChange && change.currentValue !== change.previousValue) { if (!change.firstChange && change.currentValue !== change.previousValue) {
if (propName === 'slackChanelType') { if (propName === 'slackChanelType' || propName === 'token') {
this.clearSlackCache(); this.clearSlackCache();
this.conversationSlackFormGroup.get('conversation').patchValue(''); this.conversationSlackFormGroup.get('conversation').patchValue('');
} }
@ -196,7 +199,7 @@ export class SlackConversationAutocompleteComponent implements ControlValueAcces
if (this.slackConversetionFetchObservable$ === null) { if (this.slackConversetionFetchObservable$ === null) {
let fetchObservable: Observable<Array<SlackConversation>>; let fetchObservable: Observable<Array<SlackConversation>>;
if (this.slackChanelType) { if (this.slackChanelType) {
fetchObservable = this.notificationService.listSlackConversations(this.slackChanelType, {ignoreLoading: true}); fetchObservable = this.notificationService.listSlackConversations(this.slackChanelType, this.token, {ignoreLoading: true});
} else { } else {
fetchObservable = of([]); fetchObservable = of([]);
} }

View File

@ -173,6 +173,7 @@ import { CustomDateAdapter } from '@shared/adapter/custom-datatime-adapter';
import { CustomPaginatorIntl } from '@shared/services/custom-paginator-intl'; import { CustomPaginatorIntl } from '@shared/services/custom-paginator-intl';
import { TbScriptLangComponent } from '@shared/components/script-lang.component'; import { TbScriptLangComponent } from '@shared/components/script-lang.component';
import { NotificationComponent } from '@shared/components/notification/notification.component'; import { NotificationComponent } from '@shared/components/notification/notification.component';
import { TemplateAutocompleteComponent } from '@shared/components/notification/template-autocomplete.component';
import { SlackConversationAutocompleteComponent } from '@shared/components/slack-conversation-autocomplete.component'; import { SlackConversationAutocompleteComponent } from '@shared/components/slack-conversation-autocomplete.component';
import { DateAgoPipe } from '@shared/pipe/date-ago.pipe'; import { DateAgoPipe } from '@shared/pipe/date-ago.pipe';
@ -324,6 +325,7 @@ export function MarkedOptionsFactory(markedOptionsService: MarkedOptionsService)
PhoneInputComponent, PhoneInputComponent,
TbScriptLangComponent, TbScriptLangComponent,
NotificationComponent, NotificationComponent,
TemplateAutocompleteComponent,
SlackConversationAutocompleteComponent, SlackConversationAutocompleteComponent,
DateAgoPipe DateAgoPipe
], ],
@ -537,6 +539,7 @@ export function MarkedOptionsFactory(markedOptionsService: MarkedOptionsService)
PhoneInputComponent, PhoneInputComponent,
TbScriptLangComponent, TbScriptLangComponent,
NotificationComponent, NotificationComponent,
TemplateAutocompleteComponent,
SlackConversationAutocompleteComponent, SlackConversationAutocompleteComponent,
DateAgoPipe DateAgoPipe
] ]

View File

@ -416,7 +416,7 @@
"resources": "Resources", "resources": "Resources",
"notifications": "Notifications", "notifications": "Notifications",
"notifications-settings": "Notifications settings", "notifications-settings": "Notifications settings",
"slack-api-token": "Slack api token", "slack-api-token": "Slack API token",
"slack": "Slack", "slack": "Slack",
"slack-settings": "Slack settings" "slack-settings": "Slack settings"
}, },