Merge pull request #1052 from ShvaykaD/master
add get originator telemetry node
This commit is contained in:
commit
a1abbd239a
@ -0,0 +1,164 @@
|
|||||||
|
/**
|
||||||
|
* Copyright © 2016-2018 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.
|
||||||
|
*/
|
||||||
|
package org.thingsboard.rule.engine.metadata;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.JsonGenerator;
|
||||||
|
import com.fasterxml.jackson.core.JsonParser;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import com.fasterxml.jackson.databind.node.ArrayNode;
|
||||||
|
import com.fasterxml.jackson.databind.node.ObjectNode;
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.thingsboard.rule.engine.api.*;
|
||||||
|
import org.thingsboard.rule.engine.api.util.DonAsynchron;
|
||||||
|
import org.thingsboard.rule.engine.api.util.TbNodeUtils;
|
||||||
|
import org.thingsboard.server.common.data.kv.BaseTsKvQuery;
|
||||||
|
import org.thingsboard.server.common.data.kv.TsKvEntry;
|
||||||
|
import org.thingsboard.server.common.data.kv.TsKvQuery;
|
||||||
|
import org.thingsboard.server.common.data.plugin.ComponentType;
|
||||||
|
import org.thingsboard.server.common.msg.TbMsg;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import static org.thingsboard.rule.engine.api.TbRelationTypes.SUCCESS;
|
||||||
|
import static org.thingsboard.rule.engine.metadata.TbGetTelemetryNodeConfiguration.FETCH_MODE_ALL;
|
||||||
|
import static org.thingsboard.rule.engine.metadata.TbGetTelemetryNodeConfiguration.MAX_FETCH_SIZE;
|
||||||
|
import static org.thingsboard.server.common.data.kv.Aggregation.NONE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by mshvayka on 04.09.18.
|
||||||
|
*/
|
||||||
|
@Slf4j
|
||||||
|
@RuleNode(type = ComponentType.ENRICHMENT,
|
||||||
|
name = "originator telemetry",
|
||||||
|
configClazz = TbGetTelemetryNodeConfiguration.class,
|
||||||
|
nodeDescription = "Add Message Originator Telemetry for selected time range into Message Metadata\n",
|
||||||
|
nodeDetails = "The node allows you to select fetch mode <b>FIRST/LAST/ALL</b> to fetch telemetry of certain time range that are added into Message metadata without any prefix. " +
|
||||||
|
"If selected fetch mode <b>ALL</b> Telemetry will be added like array into Message Metadata where <b>key</b> is Timestamp and <b>value</b> is value of Telemetry. " +
|
||||||
|
"If selected fetch mode <b>FIRST</b> or <b>LAST</b> Telemetry will be added like string without Timestamp",
|
||||||
|
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||||
|
configDirective = "tbEnrichmentNodeGetTelemetryFromDatabase")
|
||||||
|
public class TbGetTelemetryNode implements TbNode {
|
||||||
|
|
||||||
|
private TbGetTelemetryNodeConfiguration config;
|
||||||
|
private List<String> tsKeyNames;
|
||||||
|
private long startTsOffset;
|
||||||
|
private long endTsOffset;
|
||||||
|
private int limit;
|
||||||
|
private ObjectMapper mapper;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void init(TbContext ctx, TbNodeConfiguration configuration) throws TbNodeException {
|
||||||
|
this.config = TbNodeUtils.convert(configuration, TbGetTelemetryNodeConfiguration.class);
|
||||||
|
tsKeyNames = config.getLatestTsKeyNames();
|
||||||
|
startTsOffset = TimeUnit.valueOf(config.getStartIntervalTimeUnit()).toMillis(config.getStartInterval());
|
||||||
|
endTsOffset = TimeUnit.valueOf(config.getEndIntervalTimeUnit()).toMillis(config.getEndInterval());
|
||||||
|
limit = config.getFetchMode().equals(FETCH_MODE_ALL) ? MAX_FETCH_SIZE : 1;
|
||||||
|
mapper = new ObjectMapper();
|
||||||
|
mapper.configure(JsonGenerator.Feature.QUOTE_FIELD_NAMES, false);
|
||||||
|
mapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMsg(TbContext ctx, TbMsg msg) throws ExecutionException, InterruptedException, TbNodeException {
|
||||||
|
if (tsKeyNames.isEmpty()) {
|
||||||
|
ctx.tellFailure(msg, new IllegalStateException("Telemetry is not selected!"));
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
List<TsKvQuery> queries = buildQueries();
|
||||||
|
ListenableFuture<List<TsKvEntry>> list = ctx.getTimeseriesService().findAll(msg.getOriginator(), queries);
|
||||||
|
DonAsynchron.withCallback(list, data -> {
|
||||||
|
process(data, msg);
|
||||||
|
TbMsg newMsg = ctx.newMsg(msg.getType(), msg.getOriginator(), msg.getMetaData(), msg.getData());
|
||||||
|
ctx.tellNext(newMsg, SUCCESS);
|
||||||
|
}, error -> ctx.tellFailure(msg, error), ctx.getDbCallbackExecutor());
|
||||||
|
} catch (Exception e) {
|
||||||
|
ctx.tellFailure(msg, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO: handle direction;
|
||||||
|
private List<TsKvQuery> buildQueries() {
|
||||||
|
long ts = System.currentTimeMillis();
|
||||||
|
long startTs = ts - startTsOffset;
|
||||||
|
long endTs = ts - endTsOffset;
|
||||||
|
|
||||||
|
return tsKeyNames.stream()
|
||||||
|
.map(key -> new BaseTsKvQuery(key, startTs, endTs, 1, limit, NONE))
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
private void process(List<TsKvEntry> entries, TbMsg msg) {
|
||||||
|
ObjectNode resultNode = mapper.createObjectNode();
|
||||||
|
if (limit == MAX_FETCH_SIZE) {
|
||||||
|
entries.forEach(entry -> processArray(resultNode, entry));
|
||||||
|
} else {
|
||||||
|
entries.forEach(entry -> processSingle(resultNode, entry));
|
||||||
|
}
|
||||||
|
|
||||||
|
for (String key : tsKeyNames) {
|
||||||
|
if(resultNode.has(key)){
|
||||||
|
msg.getMetaData().putValue(key, resultNode.get(key).toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void processSingle(ObjectNode node, TsKvEntry entry) {
|
||||||
|
node.put(entry.getKey(), entry.getValueAsString());
|
||||||
|
}
|
||||||
|
|
||||||
|
private void processArray(ObjectNode node, TsKvEntry entry) {
|
||||||
|
if(node.has(entry.getKey())){
|
||||||
|
ArrayNode arrayNode = (ArrayNode) node.get(entry.getKey());
|
||||||
|
ObjectNode obj = buildNode(entry);
|
||||||
|
arrayNode.add(obj);
|
||||||
|
}else {
|
||||||
|
ArrayNode arrayNode = mapper.createArrayNode();
|
||||||
|
ObjectNode obj = buildNode(entry);
|
||||||
|
arrayNode.add(obj);
|
||||||
|
node.set(entry.getKey(), arrayNode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ObjectNode buildNode(TsKvEntry entry) {
|
||||||
|
ObjectNode obj = mapper.createObjectNode()
|
||||||
|
.put("ts", entry.getTs());
|
||||||
|
switch (entry.getDataType()) {
|
||||||
|
case STRING:
|
||||||
|
obj.put("value", entry.getValueAsString());
|
||||||
|
break;
|
||||||
|
case LONG:
|
||||||
|
obj.put("value", entry.getLongValue().get());
|
||||||
|
break;
|
||||||
|
case BOOLEAN:
|
||||||
|
obj.put("value", entry.getBooleanValue().get());
|
||||||
|
break;
|
||||||
|
case DOUBLE:
|
||||||
|
obj.put("value", entry.getDoubleValue().get());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void destroy() {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,57 @@
|
|||||||
|
/**
|
||||||
|
* Copyright © 2016-2018 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.
|
||||||
|
*/
|
||||||
|
package org.thingsboard.rule.engine.metadata;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
import org.thingsboard.rule.engine.api.NodeConfiguration;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by mshvayka on 04.09.18.
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class TbGetTelemetryNodeConfiguration implements NodeConfiguration<TbGetTelemetryNodeConfiguration> {
|
||||||
|
|
||||||
|
public static final String FETCH_MODE_FIRST = "FIRST";
|
||||||
|
public static final String FETCH_MODE_LAST = "LAST";
|
||||||
|
public static final String FETCH_MODE_ALL = "ALL";
|
||||||
|
public static final int MAX_FETCH_SIZE = 1000;
|
||||||
|
|
||||||
|
private int startInterval;
|
||||||
|
private int endInterval;
|
||||||
|
private String startIntervalTimeUnit;
|
||||||
|
private String endIntervalTimeUnit;
|
||||||
|
private String fetchMode; //FIRST, LAST, LATEST
|
||||||
|
|
||||||
|
private List<String> latestTsKeyNames;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TbGetTelemetryNodeConfiguration defaultConfiguration() {
|
||||||
|
TbGetTelemetryNodeConfiguration configuration = new TbGetTelemetryNodeConfiguration();
|
||||||
|
configuration.setLatestTsKeyNames(Collections.emptyList());
|
||||||
|
configuration.setFetchMode("FIRST");
|
||||||
|
configuration.setStartIntervalTimeUnit(TimeUnit.MINUTES.name());
|
||||||
|
configuration.setStartInterval(2);
|
||||||
|
configuration.setEndIntervalTimeUnit(TimeUnit.MINUTES.name());
|
||||||
|
configuration.setEndInterval(1);
|
||||||
|
return configuration;
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because one or more lines are too long
Loading…
x
Reference in New Issue
Block a user