Merge branch 'rc' into error-processing-device-profile
This commit is contained in:
commit
ca1c8f71c0
@ -43,19 +43,6 @@ volumes:
|
|||||||
Execute the following command to start upgrade process:
|
Execute the following command to start upgrade process:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker compose -f docker-compose-upgrade.yml up
|
docker compose -f docker-compose-upgrade.yml up --abort-on-container-exit
|
||||||
{:copy-code}
|
{:copy-code}
|
||||||
```
|
```
|
||||||
|
|
||||||
Once upgrade process successfully completed, exit from the docker-compose shell by this combination:
|
|
||||||
|
|
||||||
```text
|
|
||||||
Ctrl + C
|
|
||||||
```
|
|
||||||
|
|
||||||
Execute the following command to stop TB Edge upgrade container:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker compose -f docker-compose-upgrade.yml stop
|
|
||||||
{:copy-code}
|
|
||||||
```
|
|
||||||
@ -865,6 +865,8 @@ public class DeviceActorMessageProcessor extends AbstractContextAwareMsgProcesso
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void notifyTransportAboutClosedSessionMaxSessionsLimit(UUID sessionId, SessionInfoMetaData sessionMd) {
|
private void notifyTransportAboutClosedSessionMaxSessionsLimit(UUID sessionId, SessionInfoMetaData sessionMd) {
|
||||||
|
attributeSubscriptions.remove(sessionId);
|
||||||
|
rpcSubscriptions.remove(sessionId);
|
||||||
notifyTransportAboutClosedSession(sessionId, sessionMd, TransportSessionCloseReason.MAX_CONCURRENT_SESSIONS_LIMIT_REACHED);
|
notifyTransportAboutClosedSession(sessionId, sessionMd, TransportSessionCloseReason.MAX_CONCURRENT_SESSIONS_LIMIT_REACHED);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -404,6 +404,9 @@ public class EdgeGrpcService extends EdgeRpcServiceGrpc.EdgeRpcServiceImplBase i
|
|||||||
private void scheduleEdgeEventsCheck(EdgeGrpcSession session) {
|
private void scheduleEdgeEventsCheck(EdgeGrpcSession session) {
|
||||||
EdgeId edgeId = session.getEdge().getId();
|
EdgeId edgeId = session.getEdge().getId();
|
||||||
TenantId tenantId = session.getEdge().getTenantId();
|
TenantId tenantId = session.getEdge().getTenantId();
|
||||||
|
|
||||||
|
cancelScheduleEdgeEventsCheck(edgeId);
|
||||||
|
|
||||||
if (sessions.containsKey(edgeId)) {
|
if (sessions.containsKey(edgeId)) {
|
||||||
ScheduledFuture<?> edgeEventCheckTask = edgeEventProcessingExecutorService.schedule(() -> {
|
ScheduledFuture<?> edgeEventCheckTask = edgeEventProcessingExecutorService.schedule(() -> {
|
||||||
try {
|
try {
|
||||||
|
|||||||
@ -448,7 +448,11 @@ public abstract class EdgeGrpcSession implements Closeable {
|
|||||||
private void scheduleDownlinkMsgsPackSend(int attempt) {
|
private void scheduleDownlinkMsgsPackSend(int attempt) {
|
||||||
Runnable sendDownlinkMsgsTask = () -> {
|
Runnable sendDownlinkMsgsTask = () -> {
|
||||||
try {
|
try {
|
||||||
if (isConnected() && !sessionState.getPendingMsgsMap().values().isEmpty()) {
|
if (!isConnected()) {
|
||||||
|
stopCurrentSendDownlinkMsgsTask(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!sessionState.getPendingMsgsMap().values().isEmpty()) {
|
||||||
List<DownlinkMsg> copy = new ArrayList<>(sessionState.getPendingMsgsMap().values());
|
List<DownlinkMsg> copy = new ArrayList<>(sessionState.getPendingMsgsMap().values());
|
||||||
if (attempt > 1) {
|
if (attempt > 1) {
|
||||||
String error = "Failed to deliver the batch";
|
String error = "Failed to deliver the batch";
|
||||||
@ -529,6 +533,11 @@ public abstract class EdgeGrpcSession implements Closeable {
|
|||||||
log.debug("[{}][{}][{}] Msg has been processed successfully! Msg Id: [{}], Msg: {}", tenantId, edge.getId(), sessionId, msg.getDownlinkMsgId(), msg);
|
log.debug("[{}][{}][{}] Msg has been processed successfully! Msg Id: [{}], Msg: {}", tenantId, edge.getId(), sessionId, msg.getDownlinkMsgId(), msg);
|
||||||
} else {
|
} else {
|
||||||
log.error("[{}][{}][{}] Msg processing failed! Msg Id: [{}], Error msg: {}", tenantId, edge.getId(), sessionId, msg.getDownlinkMsgId(), msg.getErrorMsg());
|
log.error("[{}][{}][{}] Msg processing failed! Msg Id: [{}], Error msg: {}", tenantId, edge.getId(), sessionId, msg.getDownlinkMsgId(), msg.getErrorMsg());
|
||||||
|
DownlinkMsg downlinkMsg = sessionState.getPendingMsgsMap().get(msg.getDownlinkMsgId());
|
||||||
|
// if NOT timeseries or attributes failures - ack failed downlink
|
||||||
|
if (downlinkMsg.getEntityDataCount() == 0) {
|
||||||
|
sessionState.getPendingMsgsMap().remove(msg.getDownlinkMsgId());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (sessionState.getPendingMsgsMap().isEmpty()) {
|
if (sessionState.getPendingMsgsMap().isEmpty()) {
|
||||||
log.debug("[{}][{}][{}] Pending msgs map is empty. Stopping current iteration", tenantId, edge.getId(), sessionId);
|
log.debug("[{}][{}][{}] Pending msgs map is empty. Stopping current iteration", tenantId, edge.getId(), sessionId);
|
||||||
|
|||||||
@ -54,23 +54,25 @@ public class GeneralEdgeEventFetcher implements EdgeEventFetcher {
|
|||||||
log.trace("[{}] Finding general edge events [{}], seqIdStart = {}, pageLink = {}",
|
log.trace("[{}] Finding general edge events [{}], seqIdStart = {}, pageLink = {}",
|
||||||
tenantId, edge.getId(), seqIdStart, pageLink);
|
tenantId, edge.getId(), seqIdStart, pageLink);
|
||||||
PageData<EdgeEvent> edgeEvents = edgeEventService.findEdgeEvents(tenantId, edge.getId(), seqIdStart, null, (TimePageLink) pageLink);
|
PageData<EdgeEvent> edgeEvents = edgeEventService.findEdgeEvents(tenantId, edge.getId(), seqIdStart, null, (TimePageLink) pageLink);
|
||||||
if (edgeEvents.getData().isEmpty()) {
|
if (!edgeEvents.getData().isEmpty()) {
|
||||||
|
return edgeEvents;
|
||||||
|
}
|
||||||
|
if (seqIdStart > this.maxReadRecordsCount) {
|
||||||
edgeEvents = edgeEventService.findEdgeEvents(tenantId, edge.getId(), 0L, Math.max(this.maxReadRecordsCount, seqIdStart - this.maxReadRecordsCount), (TimePageLink) pageLink);
|
edgeEvents = edgeEventService.findEdgeEvents(tenantId, edge.getId(), 0L, Math.max(this.maxReadRecordsCount, seqIdStart - this.maxReadRecordsCount), (TimePageLink) pageLink);
|
||||||
if (edgeEvents.getData().stream().anyMatch(ee -> ee.getSeqId() < seqIdStart)) {
|
if (edgeEvents.getData().stream().anyMatch(ee -> ee.getSeqId() < seqIdStart)) {
|
||||||
log.info("[{}] seqId column of edge_event table started new cycle [{}]", tenantId, edge.getId());
|
log.info("[{}] seqId column of edge_event table started new cycle [{}]", tenantId, edge.getId());
|
||||||
this.seqIdNewCycleStarted = true;
|
this.seqIdNewCycleStarted = true;
|
||||||
this.seqIdStart = 0L;
|
this.seqIdStart = 0L;
|
||||||
} else {
|
return edgeEvents;
|
||||||
edgeEvents = new PageData<>();
|
|
||||||
log.warn("[{}] unexpected edge notification message received. " +
|
|
||||||
"no new events found and seqId column of edge_event table doesn't started new cycle [{}]", tenantId, edge.getId());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return edgeEvents;
|
log.info("[{}] Unexpected edge notification message received. " +
|
||||||
|
"No new events found, and the seqId column of the edge_event table has not started a new cycle [{}].", tenantId, edge.getId());
|
||||||
|
return new PageData<>();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.error("[{}] failed to find edge events [{}]", tenantId, edge.getId());
|
log.error("[{}] Failed to find edge events [{}]", tenantId, edge.getId(), e);
|
||||||
|
return new PageData<>();
|
||||||
}
|
}
|
||||||
return new PageData<>();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -26,6 +26,7 @@ import org.thingsboard.server.dao.service.DataValidator;
|
|||||||
import org.thingsboard.server.gen.edge.v1.DashboardUpdateMsg;
|
import org.thingsboard.server.gen.edge.v1.DashboardUpdateMsg;
|
||||||
import org.thingsboard.server.service.edge.rpc.processor.BaseEdgeProcessor;
|
import org.thingsboard.server.service.edge.rpc.processor.BaseEdgeProcessor;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
@Slf4j
|
@Slf4j
|
||||||
@ -40,54 +41,66 @@ public abstract class BaseDashboardProcessor extends BaseEdgeProcessor {
|
|||||||
if (dashboard == null) {
|
if (dashboard == null) {
|
||||||
throw new RuntimeException("[{" + tenantId + "}] dashboardUpdateMsg {" + dashboardUpdateMsg + "} cannot be converted to dashboard");
|
throw new RuntimeException("[{" + tenantId + "}] dashboardUpdateMsg {" + dashboardUpdateMsg + "} cannot be converted to dashboard");
|
||||||
}
|
}
|
||||||
Set<ShortCustomerInfo> assignedCustomers = null;
|
Set<ShortCustomerInfo> newAssignedCustomers = new HashSet<>();
|
||||||
|
if (dashboard.getAssignedCustomers() != null && !dashboard.getAssignedCustomers().isEmpty()) {
|
||||||
|
newAssignedCustomers.addAll(dashboard.getAssignedCustomers());
|
||||||
|
}
|
||||||
Dashboard dashboardById = edgeCtx.getDashboardService().findDashboardById(tenantId, dashboardId);
|
Dashboard dashboardById = edgeCtx.getDashboardService().findDashboardById(tenantId, dashboardId);
|
||||||
if (dashboardById == null) {
|
if (dashboardById == null) {
|
||||||
created = true;
|
created = true;
|
||||||
dashboard.setId(null);
|
dashboard.setId(null);
|
||||||
|
dashboard.setAssignedCustomers(null);
|
||||||
} else {
|
} else {
|
||||||
dashboard.setId(dashboardId);
|
dashboard.setId(dashboardId);
|
||||||
assignedCustomers = filterNonExistingCustomers(tenantId, dashboardById.getAssignedCustomers());
|
dashboard.setAssignedCustomers(dashboardById.getAssignedCustomers());
|
||||||
}
|
}
|
||||||
|
|
||||||
dashboardValidator.validate(dashboard, Dashboard::getTenantId);
|
dashboardValidator.validate(dashboard, Dashboard::getTenantId);
|
||||||
if (created) {
|
if (created) {
|
||||||
dashboard.setId(dashboardId);
|
dashboard.setId(dashboardId);
|
||||||
}
|
}
|
||||||
Set<ShortCustomerInfo> msgAssignedCustomers = filterNonExistingCustomers(tenantId, dashboard.getAssignedCustomers());
|
|
||||||
if (msgAssignedCustomers != null) {
|
|
||||||
if (assignedCustomers == null) {
|
|
||||||
assignedCustomers = msgAssignedCustomers;
|
|
||||||
} else {
|
|
||||||
assignedCustomers.addAll(msgAssignedCustomers);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
dashboard.setAssignedCustomers(assignedCustomers);
|
|
||||||
Dashboard savedDashboard = edgeCtx.getDashboardService().saveDashboard(dashboard, false);
|
Dashboard savedDashboard = edgeCtx.getDashboardService().saveDashboard(dashboard, false);
|
||||||
if (msgAssignedCustomers != null && !msgAssignedCustomers.isEmpty()) {
|
|
||||||
for (ShortCustomerInfo assignedCustomer : msgAssignedCustomers) {
|
updateDashboardAssignments(tenantId, dashboardById, savedDashboard, newAssignedCustomers);
|
||||||
if (assignedCustomer.getCustomerId().equals(customerId)) {
|
|
||||||
edgeCtx.getDashboardService().assignDashboardToCustomer(tenantId, savedDashboard.getId(), assignedCustomer.getCustomerId());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
unassignCustomersFromDashboard(tenantId, savedDashboard, customerId);
|
|
||||||
}
|
|
||||||
return created;
|
return created;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void unassignCustomersFromDashboard(TenantId tenantId, Dashboard dashboard, CustomerId customerId) {
|
private void updateDashboardAssignments(TenantId tenantId, Dashboard dashboardById, Dashboard savedDashboard, Set<ShortCustomerInfo> newAssignedCustomers) {
|
||||||
if (dashboard.getAssignedCustomers() != null && !dashboard.getAssignedCustomers().isEmpty()) {
|
Set<ShortCustomerInfo> currentAssignedCustomers = new HashSet<>();
|
||||||
for (ShortCustomerInfo assignedCustomer : dashboard.getAssignedCustomers()) {
|
if (dashboardById != null) {
|
||||||
if (assignedCustomer.getCustomerId().equals(customerId)) {
|
if (dashboardById.getAssignedCustomers() != null) {
|
||||||
edgeCtx.getDashboardService().unassignDashboardFromCustomer(tenantId, dashboard.getId(), assignedCustomer.getCustomerId());
|
currentAssignedCustomers.addAll(dashboardById.getAssignedCustomers());
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
newAssignedCustomers = filterNonExistingCustomers(tenantId, currentAssignedCustomers, newAssignedCustomers);
|
||||||
|
|
||||||
|
Set<CustomerId> addedCustomerIds = new HashSet<>();
|
||||||
|
Set<CustomerId> removedCustomerIds = new HashSet<>();
|
||||||
|
for (ShortCustomerInfo newAssignedCustomer : newAssignedCustomers) {
|
||||||
|
if (!savedDashboard.isAssignedToCustomer(newAssignedCustomer.getCustomerId())) {
|
||||||
|
addedCustomerIds.add(newAssignedCustomer.getCustomerId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ShortCustomerInfo currentAssignedCustomer : currentAssignedCustomers) {
|
||||||
|
if (!newAssignedCustomers.contains(currentAssignedCustomer)) {
|
||||||
|
removedCustomerIds.add(currentAssignedCustomer.getCustomerId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (CustomerId customerIdToAdd : addedCustomerIds) {
|
||||||
|
edgeCtx.getDashboardService().assignDashboardToCustomer(tenantId, savedDashboard.getId(), customerIdToAdd);
|
||||||
|
}
|
||||||
|
for (CustomerId customerIdToRemove : removedCustomerIds) {
|
||||||
|
edgeCtx.getDashboardService().unassignDashboardFromCustomer(tenantId, savedDashboard.getId(), customerIdToRemove);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract Dashboard constructDashboardFromUpdateMsg(TenantId tenantId, DashboardId dashboardId, DashboardUpdateMsg dashboardUpdateMsg);
|
protected abstract Dashboard constructDashboardFromUpdateMsg(TenantId tenantId, DashboardId dashboardId, DashboardUpdateMsg dashboardUpdateMsg);
|
||||||
|
|
||||||
protected abstract Set<ShortCustomerInfo> filterNonExistingCustomers(TenantId tenantId, Set<ShortCustomerInfo> assignedCustomers);
|
protected abstract Set<ShortCustomerInfo> filterNonExistingCustomers(TenantId tenantId, Set<ShortCustomerInfo> currentAssignedCustomers, Set<ShortCustomerInfo> newAssignedCustomers);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -128,9 +128,9 @@ public abstract class DashboardEdgeProcessor extends BaseDashboardProcessor impl
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Set<ShortCustomerInfo> filterNonExistingCustomers(TenantId tenantId, Set<ShortCustomerInfo> assignedCustomers) {
|
protected Set<ShortCustomerInfo> filterNonExistingCustomers(TenantId tenantId, Set<ShortCustomerInfo> currentAssignedCustomers, Set<ShortCustomerInfo> newAssignedCustomers) {
|
||||||
// do nothing on cloud
|
newAssignedCustomers.addAll(currentAssignedCustomers);
|
||||||
return assignedCustomers;
|
return newAssignedCustomers;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -26,6 +26,7 @@ import org.thingsboard.server.common.data.id.TenantId;
|
|||||||
import org.thingsboard.server.gen.edge.v1.DashboardUpdateMsg;
|
import org.thingsboard.server.gen.edge.v1.DashboardUpdateMsg;
|
||||||
import org.thingsboard.server.queue.util.TbCoreComponent;
|
import org.thingsboard.server.queue.util.TbCoreComponent;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
@Component
|
@Component
|
||||||
@ -44,7 +45,7 @@ public class DashboardEdgeProcessorV1 extends DashboardEdgeProcessor {
|
|||||||
Set<ShortCustomerInfo> assignedCustomers;
|
Set<ShortCustomerInfo> assignedCustomers;
|
||||||
if (dashboardUpdateMsg.hasAssignedCustomers()) {
|
if (dashboardUpdateMsg.hasAssignedCustomers()) {
|
||||||
assignedCustomers = JacksonUtil.fromString(dashboardUpdateMsg.getAssignedCustomers(), new TypeReference<>() {});
|
assignedCustomers = JacksonUtil.fromString(dashboardUpdateMsg.getAssignedCustomers(), new TypeReference<>() {});
|
||||||
assignedCustomers = filterNonExistingCustomers(tenantId, assignedCustomers);
|
assignedCustomers = filterNonExistingCustomers(tenantId, new HashSet<>(), assignedCustomers);
|
||||||
dashboard.setAssignedCustomers(assignedCustomers);
|
dashboard.setAssignedCustomers(assignedCustomers);
|
||||||
}
|
}
|
||||||
dashboard.setMobileOrder(dashboardUpdateMsg.hasMobileOrder() ? dashboardUpdateMsg.getMobileOrder() : null);
|
dashboard.setMobileOrder(dashboardUpdateMsg.hasMobileOrder() ? dashboardUpdateMsg.getMobileOrder() : null);
|
||||||
|
|||||||
@ -53,6 +53,9 @@ public abstract class BaseResourceProcessor extends BaseEdgeProcessor {
|
|||||||
}
|
}
|
||||||
String resourceKey = resource.getResourceKey();
|
String resourceKey = resource.getResourceKey();
|
||||||
ResourceType resourceType = resource.getResourceType();
|
ResourceType resourceType = resource.getResourceType();
|
||||||
|
if (!created && !resourceType.isUpdatable()) {
|
||||||
|
resource.setData(null);
|
||||||
|
}
|
||||||
PageDataIterable<TbResource> resourcesIterable = new PageDataIterable<>(
|
PageDataIterable<TbResource> resourcesIterable = new PageDataIterable<>(
|
||||||
link -> edgeCtx.getResourceService().findTenantResourcesByResourceTypeAndPageLink(tenantId, resourceType, link), 1024);
|
link -> edgeCtx.getResourceService().findTenantResourcesByResourceTypeAndPageLink(tenantId, resourceType, link), 1024);
|
||||||
for (TbResource tbResource : resourcesIterable) {
|
for (TbResource tbResource : resourcesIterable) {
|
||||||
|
|||||||
@ -28,17 +28,22 @@ import org.thingsboard.server.gen.edge.v1.WidgetBundleTypesRequestMsg;
|
|||||||
|
|
||||||
public interface EdgeRequestsService {
|
public interface EdgeRequestsService {
|
||||||
|
|
||||||
|
@Deprecated(since = "3.9.1", forRemoval = true)
|
||||||
ListenableFuture<Void> processRuleChainMetadataRequestMsg(TenantId tenantId, Edge edge, RuleChainMetadataRequestMsg ruleChainMetadataRequestMsg);
|
ListenableFuture<Void> processRuleChainMetadataRequestMsg(TenantId tenantId, Edge edge, RuleChainMetadataRequestMsg ruleChainMetadataRequestMsg);
|
||||||
|
|
||||||
ListenableFuture<Void> processAttributesRequestMsg(TenantId tenantId, Edge edge, AttributesRequestMsg attributesRequestMsg);
|
ListenableFuture<Void> processAttributesRequestMsg(TenantId tenantId, Edge edge, AttributesRequestMsg attributesRequestMsg);
|
||||||
|
|
||||||
ListenableFuture<Void> processRelationRequestMsg(TenantId tenantId, Edge edge, RelationRequestMsg relationRequestMsg);
|
ListenableFuture<Void> processRelationRequestMsg(TenantId tenantId, Edge edge, RelationRequestMsg relationRequestMsg);
|
||||||
|
|
||||||
|
@Deprecated(since = "3.9.1", forRemoval = true)
|
||||||
ListenableFuture<Void> processDeviceCredentialsRequestMsg(TenantId tenantId, Edge edge, DeviceCredentialsRequestMsg deviceCredentialsRequestMsg);
|
ListenableFuture<Void> processDeviceCredentialsRequestMsg(TenantId tenantId, Edge edge, DeviceCredentialsRequestMsg deviceCredentialsRequestMsg);
|
||||||
|
|
||||||
|
@Deprecated(since = "3.9.1", forRemoval = true)
|
||||||
ListenableFuture<Void> processUserCredentialsRequestMsg(TenantId tenantId, Edge edge, UserCredentialsRequestMsg userCredentialsRequestMsg);
|
ListenableFuture<Void> processUserCredentialsRequestMsg(TenantId tenantId, Edge edge, UserCredentialsRequestMsg userCredentialsRequestMsg);
|
||||||
|
|
||||||
|
@Deprecated(since = "3.9.1", forRemoval = true)
|
||||||
ListenableFuture<Void> processWidgetBundleTypesRequestMsg(TenantId tenantId, Edge edge, WidgetBundleTypesRequestMsg widgetBundleTypesRequestMsg);
|
ListenableFuture<Void> processWidgetBundleTypesRequestMsg(TenantId tenantId, Edge edge, WidgetBundleTypesRequestMsg widgetBundleTypesRequestMsg);
|
||||||
|
|
||||||
|
@Deprecated(since = "3.9.1", forRemoval = true)
|
||||||
ListenableFuture<Void> processEntityViewsRequestMsg(TenantId tenantId, Edge edge, EntityViewsRequestMsg entityViewsRequestMsg);
|
ListenableFuture<Void> processEntityViewsRequestMsg(TenantId tenantId, Edge edge, EntityViewsRequestMsg entityViewsRequestMsg);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -115,8 +115,9 @@ public class DefaultTbRuleEngineRpcService implements TbRuleEngineDeviceRpcServi
|
|||||||
ToDeviceRpcRequest request = new ToDeviceRpcRequest(src.getRequestUUID(), src.getTenantId(), src.getDeviceId(),
|
ToDeviceRpcRequest request = new ToDeviceRpcRequest(src.getRequestUUID(), src.getTenantId(), src.getDeviceId(),
|
||||||
src.isOneway(), src.getExpirationTime(), new ToDeviceRpcRequestBody(src.getMethod(), src.getBody()), src.isPersisted(), src.getRetries(), src.getAdditionalInfo());
|
src.isOneway(), src.getExpirationTime(), new ToDeviceRpcRequestBody(src.getMethod(), src.getBody()), src.isPersisted(), src.getRetries(), src.getAdditionalInfo());
|
||||||
forwardRpcRequestToDeviceActor(request, response -> {
|
forwardRpcRequestToDeviceActor(request, response -> {
|
||||||
if (src.isRestApiCall()) {
|
String originServiceId = src.getOriginServiceId();
|
||||||
sendRpcResponseToTbCore(src.getOriginServiceId(), response);
|
if (src.isRestApiCall() && originServiceId != null) {
|
||||||
|
sendRpcResponseToTbCore(originServiceId, response);
|
||||||
}
|
}
|
||||||
consumer.accept(RuleEngineDeviceRpcResponse.builder()
|
consumer.accept(RuleEngineDeviceRpcResponse.builder()
|
||||||
.deviceId(src.getDeviceId())
|
.deviceId(src.getDeviceId())
|
||||||
|
|||||||
@ -212,9 +212,9 @@ ui:
|
|||||||
database:
|
database:
|
||||||
ts_max_intervals: "${DATABASE_TS_MAX_INTERVALS:700}" # Max number of DB queries generated by a single API call to fetch telemetry records
|
ts_max_intervals: "${DATABASE_TS_MAX_INTERVALS:700}" # Max number of DB queries generated by a single API call to fetch telemetry records
|
||||||
ts:
|
ts:
|
||||||
type: "${DATABASE_TS_TYPE:sql}" # cassandra or sql. timescale option is deprecated and will no longer be supported in ThingsBoard 4.0
|
type: "${DATABASE_TS_TYPE:sql}" # cassandra, sql, or timescale (for hybrid mode, DATABASE_TS_TYPE value should be cassandra, or timescale)
|
||||||
ts_latest:
|
ts_latest:
|
||||||
type: "${DATABASE_TS_LATEST_TYPE:sql}" # cassandra or sql. timescale option is deprecated and will no longer be supported in ThingsBoard 4.0
|
type: "${DATABASE_TS_LATEST_TYPE:sql}" # cassandra, sql, or timescale (for hybrid mode, DATABASE_TS_TYPE value should be cassandra, or timescale)
|
||||||
|
|
||||||
# Cassandra driver configuration parameters
|
# Cassandra driver configuration parameters
|
||||||
cassandra:
|
cassandra:
|
||||||
|
|||||||
@ -16,6 +16,7 @@
|
|||||||
package org.thingsboard.server.edge;
|
package org.thingsboard.server.edge;
|
||||||
|
|
||||||
import com.fasterxml.jackson.core.type.TypeReference;
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
import com.google.protobuf.AbstractMessage;
|
import com.google.protobuf.AbstractMessage;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
@ -47,12 +48,14 @@ public class DashboardEdgeTest extends AbstractEdgeTest {
|
|||||||
private static final int MOBILE_ORDER = 5;
|
private static final int MOBILE_ORDER = 5;
|
||||||
private static final String IMAGE = "";
|
private static final String IMAGE = "";
|
||||||
|
|
||||||
|
private static final String DASHBOARD_TITLE = "Edge Test Dashboard";
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testDashboards() throws Exception {
|
public void testDashboards() throws Exception {
|
||||||
// create dashboard and assign to edge
|
// create dashboard and assign to edge
|
||||||
edgeImitator.expectMessageAmount(2);
|
edgeImitator.expectMessageAmount(2);
|
||||||
Dashboard dashboard = new Dashboard();
|
Dashboard dashboard = new Dashboard();
|
||||||
dashboard.setTitle("Edge Test Dashboard");
|
dashboard.setTitle(DASHBOARD_TITLE);
|
||||||
dashboard.setMobileHide(true);
|
dashboard.setMobileHide(true);
|
||||||
dashboard.setImage(IMAGE);
|
dashboard.setImage(IMAGE);
|
||||||
dashboard.setMobileOrder(MOBILE_ORDER);
|
dashboard.setMobileOrder(MOBILE_ORDER);
|
||||||
@ -175,7 +178,11 @@ public class DashboardEdgeTest extends AbstractEdgeTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testSendDashboardToCloud() throws Exception {
|
public void testSendDashboardToCloud() throws Exception {
|
||||||
Dashboard dashboard = buildDashboardForUplinkMsg();
|
Customer customer = new Customer();
|
||||||
|
customer.setTitle("Edge Customer");
|
||||||
|
Customer savedCustomer = doPost("/api/customer", customer, Customer.class);
|
||||||
|
|
||||||
|
Dashboard dashboard = buildDashboardForUplinkMsg(savedCustomer);
|
||||||
|
|
||||||
UplinkMsg.Builder uplinkMsgBuilder = UplinkMsg.newBuilder();
|
UplinkMsg.Builder uplinkMsgBuilder = UplinkMsg.newBuilder();
|
||||||
DashboardUpdateMsg.Builder dashboardUpdateMsgBuilder = DashboardUpdateMsg.newBuilder();
|
DashboardUpdateMsg.Builder dashboardUpdateMsgBuilder = DashboardUpdateMsg.newBuilder();
|
||||||
@ -195,7 +202,27 @@ public class DashboardEdgeTest extends AbstractEdgeTest {
|
|||||||
|
|
||||||
Dashboard foundDashboard = doGet("/api/dashboard/" + dashboard.getUuidId(), Dashboard.class);
|
Dashboard foundDashboard = doGet("/api/dashboard/" + dashboard.getUuidId(), Dashboard.class);
|
||||||
Assert.assertNotNull(foundDashboard);
|
Assert.assertNotNull(foundDashboard);
|
||||||
Assert.assertEquals("Edge Test Dashboard", foundDashboard.getName());
|
Assert.assertEquals(DASHBOARD_TITLE, foundDashboard.getName());
|
||||||
|
|
||||||
|
PageData<DashboardInfo> pageData = doGetTypedWithPageLink("/api/customer/" + savedCustomer.getId().toString() + "/dashboards?",
|
||||||
|
new TypeReference<>() {}, new PageLink(100));
|
||||||
|
Assert.assertEquals(1, pageData.getData().size());
|
||||||
|
Assert.assertEquals(DASHBOARD_TITLE, pageData.getData().get(0).getTitle());
|
||||||
|
|
||||||
|
dashboard.setTitle(DASHBOARD_TITLE + " Updated");
|
||||||
|
dashboard.setAssignedCustomers(null);
|
||||||
|
dashboardUpdateMsgBuilder.setEntity(JacksonUtil.toString(dashboard));
|
||||||
|
dashboardUpdateMsgBuilder.setMsgType(UpdateMsgType.ENTITY_UPDATED_RPC_MESSAGE);
|
||||||
|
uplinkMsgBuilder = UplinkMsg.newBuilder();
|
||||||
|
uplinkMsgBuilder.addDashboardUpdateMsg(dashboardUpdateMsgBuilder.build());
|
||||||
|
|
||||||
|
edgeImitator.expectResponsesAmount(1);
|
||||||
|
edgeImitator.sendUplinkMsg(uplinkMsgBuilder.build());
|
||||||
|
|
||||||
|
Assert.assertTrue(edgeImitator.waitForResponses());
|
||||||
|
|
||||||
|
foundDashboard = doGet("/api/dashboard/" + dashboard.getUuidId(), Dashboard.class);
|
||||||
|
Assert.assertEquals(DASHBOARD_TITLE + " Updated", foundDashboard.getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -242,11 +269,12 @@ public class DashboardEdgeTest extends AbstractEdgeTest {
|
|||||||
return savedDashboard;
|
return savedDashboard;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Dashboard buildDashboardForUplinkMsg() {
|
private Dashboard buildDashboardForUplinkMsg(Customer savedCustomer) {
|
||||||
Dashboard dashboard = new Dashboard();
|
Dashboard dashboard = new Dashboard();
|
||||||
dashboard.setId(new DashboardId(UUID.randomUUID()));
|
dashboard.setId(new DashboardId(UUID.randomUUID()));
|
||||||
dashboard.setTenantId(tenantId);
|
dashboard.setTenantId(tenantId);
|
||||||
dashboard.setTitle("Edge Test Dashboard");
|
dashboard.setTitle(DASHBOARD_TITLE);
|
||||||
|
dashboard.setAssignedCustomers(Sets.newHashSet(new ShortCustomerInfo(savedCustomer.getId(), savedCustomer.getTitle(), savedCustomer.isPublic())));
|
||||||
return dashboard;
|
return dashboard;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -17,6 +17,7 @@ package org.thingsboard.server.edge;
|
|||||||
|
|
||||||
import com.datastax.oss.driver.api.core.uuid.Uuids;
|
import com.datastax.oss.driver.api.core.uuid.Uuids;
|
||||||
import com.google.protobuf.AbstractMessage;
|
import com.google.protobuf.AbstractMessage;
|
||||||
|
import com.google.protobuf.InvalidProtocolBufferException;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.thingsboard.common.util.JacksonUtil;
|
import org.thingsboard.common.util.JacksonUtil;
|
||||||
@ -98,30 +99,23 @@ public class ResourceEdgeTest extends AbstractEdgeTest {
|
|||||||
public void testSendResourceToCloud() throws Exception {
|
public void testSendResourceToCloud() throws Exception {
|
||||||
TbResource tbResource = createTbResource();
|
TbResource tbResource = createTbResource();
|
||||||
UUID uuid = Uuids.timeBased();
|
UUID uuid = Uuids.timeBased();
|
||||||
|
UplinkMsg uplinkMsg = getUplinkMsg(uuid, tbResource, UpdateMsgType.ENTITY_CREATED_RPC_MESSAGE);
|
||||||
|
|
||||||
UplinkMsg.Builder uplinkMsgBuilder = UplinkMsg.newBuilder();
|
checkResourceOnCloud(uplinkMsg, uuid, tbResource.getTitle());
|
||||||
ResourceUpdateMsg.Builder resourceUpdateMsgBuilder = ResourceUpdateMsg.newBuilder();
|
}
|
||||||
resourceUpdateMsgBuilder.setIdMSB(uuid.getMostSignificantBits());
|
|
||||||
resourceUpdateMsgBuilder.setIdLSB(uuid.getLeastSignificantBits());
|
|
||||||
resourceUpdateMsgBuilder.setEntity(JacksonUtil.toString(tbResource));
|
|
||||||
resourceUpdateMsgBuilder.setMsgType(UpdateMsgType.ENTITY_CREATED_RPC_MESSAGE);
|
|
||||||
testAutoGeneratedCodeByProtobuf(resourceUpdateMsgBuilder);
|
|
||||||
uplinkMsgBuilder.addResourceUpdateMsg(resourceUpdateMsgBuilder.build());
|
|
||||||
|
|
||||||
testAutoGeneratedCodeByProtobuf(uplinkMsgBuilder);
|
@Test
|
||||||
|
public void testUpdateResourceTitleOnCloud() throws Exception {
|
||||||
|
TbResource tbResource = createTbResource();
|
||||||
|
UUID uuid = Uuids.timeBased();
|
||||||
|
UplinkMsg uplinkMsg = getUplinkMsg(uuid, tbResource, UpdateMsgType.ENTITY_CREATED_RPC_MESSAGE);
|
||||||
|
|
||||||
edgeImitator.expectResponsesAmount(1);
|
checkResourceOnCloud(uplinkMsg, uuid, tbResource.getTitle());
|
||||||
edgeImitator.sendUplinkMsg(uplinkMsgBuilder.build());
|
|
||||||
|
|
||||||
Assert.assertTrue(edgeImitator.waitForResponses());
|
tbResource.setTitle("Updated Edge Test Resource");
|
||||||
|
UplinkMsg updatedUplinkMsg = getUplinkMsg(uuid, tbResource, UpdateMsgType.ENTITY_UPDATED_RPC_MESSAGE);
|
||||||
|
|
||||||
UplinkResponseMsg latestResponseMsg = edgeImitator.getLatestResponseMsg();
|
checkResourceOnCloud(updatedUplinkMsg, uuid, tbResource.getTitle());
|
||||||
Assert.assertTrue(latestResponseMsg.getSuccess());
|
|
||||||
|
|
||||||
TbResource tb = doGet("/api/resource/" + uuid, TbResource.class);
|
|
||||||
Assert.assertNotNull(tb);
|
|
||||||
Assert.assertEquals("Edge Test Resource", tb.getName());
|
|
||||||
Assert.assertEquals(TEST_DATA, tb.getEncodedData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -134,21 +128,12 @@ public class ResourceEdgeTest extends AbstractEdgeTest {
|
|||||||
|
|
||||||
UUID uuid = Uuids.timeBased();
|
UUID uuid = Uuids.timeBased();
|
||||||
|
|
||||||
UplinkMsg.Builder uplinkMsgBuilder = UplinkMsg.newBuilder();
|
UplinkMsg uplinkMsg = getUplinkMsg(uuid, resource, UpdateMsgType.ENTITY_CREATED_RPC_MESSAGE);
|
||||||
ResourceUpdateMsg.Builder resourceUpdateMsgBuilder = ResourceUpdateMsg.newBuilder();
|
|
||||||
resourceUpdateMsgBuilder.setIdMSB(uuid.getMostSignificantBits());
|
|
||||||
resourceUpdateMsgBuilder.setIdLSB(uuid.getLeastSignificantBits());
|
|
||||||
resourceUpdateMsgBuilder.setEntity(JacksonUtil.toString(resource));
|
|
||||||
resourceUpdateMsgBuilder.setMsgType(UpdateMsgType.ENTITY_CREATED_RPC_MESSAGE);
|
|
||||||
testAutoGeneratedCodeByProtobuf(resourceUpdateMsgBuilder);
|
|
||||||
uplinkMsgBuilder.addResourceUpdateMsg(resourceUpdateMsgBuilder.build());
|
|
||||||
|
|
||||||
testAutoGeneratedCodeByProtobuf(uplinkMsgBuilder);
|
|
||||||
|
|
||||||
edgeImitator.expectResponsesAmount(1);
|
edgeImitator.expectResponsesAmount(1);
|
||||||
edgeImitator.expectMessageAmount(1);
|
edgeImitator.expectMessageAmount(1);
|
||||||
|
|
||||||
edgeImitator.sendUplinkMsg(uplinkMsgBuilder.build());
|
edgeImitator.sendUplinkMsg(uplinkMsg);
|
||||||
|
|
||||||
Assert.assertTrue(edgeImitator.waitForResponses());
|
Assert.assertTrue(edgeImitator.waitForResponses());
|
||||||
Assert.assertTrue(edgeImitator.waitForMessages());
|
Assert.assertTrue(edgeImitator.waitForMessages());
|
||||||
@ -177,4 +162,35 @@ public class ResourceEdgeTest extends AbstractEdgeTest {
|
|||||||
tbResource.setEncodedData(TEST_DATA);
|
tbResource.setEncodedData(TEST_DATA);
|
||||||
return tbResource;
|
return tbResource;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private UplinkMsg getUplinkMsg(UUID uuid, TbResource tbResource, UpdateMsgType updateMsgType) throws InvalidProtocolBufferException {
|
||||||
|
UplinkMsg.Builder uplinkMsgBuilder = UplinkMsg.newBuilder();
|
||||||
|
ResourceUpdateMsg.Builder resourceUpdateMsgBuilder = ResourceUpdateMsg.newBuilder();
|
||||||
|
resourceUpdateMsgBuilder.setIdMSB(uuid.getMostSignificantBits());
|
||||||
|
resourceUpdateMsgBuilder.setIdLSB(uuid.getLeastSignificantBits());
|
||||||
|
resourceUpdateMsgBuilder.setEntity(JacksonUtil.toString(tbResource));
|
||||||
|
resourceUpdateMsgBuilder.setMsgType(updateMsgType);
|
||||||
|
testAutoGeneratedCodeByProtobuf(resourceUpdateMsgBuilder);
|
||||||
|
uplinkMsgBuilder.addResourceUpdateMsg(resourceUpdateMsgBuilder.build());
|
||||||
|
|
||||||
|
testAutoGeneratedCodeByProtobuf(uplinkMsgBuilder);
|
||||||
|
|
||||||
|
return uplinkMsgBuilder.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkResourceOnCloud(UplinkMsg uplinkMsg, UUID uuid, String resourceTitle) throws Exception {
|
||||||
|
edgeImitator.expectResponsesAmount(1);
|
||||||
|
edgeImitator.sendUplinkMsg(uplinkMsg);
|
||||||
|
|
||||||
|
Assert.assertTrue(edgeImitator.waitForResponses());
|
||||||
|
|
||||||
|
UplinkResponseMsg latestResponseMsg = edgeImitator.getLatestResponseMsg();
|
||||||
|
Assert.assertTrue(latestResponseMsg.getSuccess());
|
||||||
|
|
||||||
|
TbResource tb = doGet("/api/resource/" + uuid, TbResource.class);
|
||||||
|
Assert.assertNotNull(tb);
|
||||||
|
Assert.assertEquals(resourceTitle, tb.getName());
|
||||||
|
Assert.assertEquals(TEST_DATA, tb.getEncodedData());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -96,6 +96,7 @@ public class KafkaMonolithQueueFactory implements TbCoreQueueFactory, TbRuleEngi
|
|||||||
private final TbQueueAdmin edgeEventAdmin;
|
private final TbQueueAdmin edgeEventAdmin;
|
||||||
|
|
||||||
private final AtomicLong consumerCount = new AtomicLong();
|
private final AtomicLong consumerCount = new AtomicLong();
|
||||||
|
private final AtomicLong edgeConsumerCount = new AtomicLong();
|
||||||
|
|
||||||
public KafkaMonolithQueueFactory(TopicService topicService, TbKafkaSettings kafkaSettings,
|
public KafkaMonolithQueueFactory(TopicService topicService, TbKafkaSettings kafkaSettings,
|
||||||
TbServiceInfoProvider serviceInfoProvider,
|
TbServiceInfoProvider serviceInfoProvider,
|
||||||
@ -472,7 +473,7 @@ public class KafkaMonolithQueueFactory implements TbCoreQueueFactory, TbRuleEngi
|
|||||||
TbKafkaConsumerTemplate.TbKafkaConsumerTemplateBuilder<TbProtoQueueMsg<ToEdgeEventNotificationMsg>> consumerBuilder = TbKafkaConsumerTemplate.builder();
|
TbKafkaConsumerTemplate.TbKafkaConsumerTemplateBuilder<TbProtoQueueMsg<ToEdgeEventNotificationMsg>> consumerBuilder = TbKafkaConsumerTemplate.builder();
|
||||||
consumerBuilder.settings(kafkaSettings);
|
consumerBuilder.settings(kafkaSettings);
|
||||||
consumerBuilder.topic(topicService.buildTopicName("tb_edge_event.notifications." + tenantId + "." + edgeId));
|
consumerBuilder.topic(topicService.buildTopicName("tb_edge_event.notifications." + tenantId + "." + edgeId));
|
||||||
consumerBuilder.clientId("monolith-to-edge-event-consumer" + serviceInfoProvider.getServiceId());
|
consumerBuilder.clientId("monolith-to-edge-event-consumer-" + serviceInfoProvider.getServiceId() + "-" + edgeConsumerCount.incrementAndGet());
|
||||||
consumerBuilder.groupId(topicService.buildTopicName("monolith-edge-event-consumer"));
|
consumerBuilder.groupId(topicService.buildTopicName("monolith-edge-event-consumer"));
|
||||||
consumerBuilder.decoder(msg -> new TbProtoQueueMsg<>(msg.getKey(), ToEdgeEventNotificationMsg.parseFrom(msg.getData()), msg.getHeaders()));
|
consumerBuilder.decoder(msg -> new TbProtoQueueMsg<>(msg.getKey(), ToEdgeEventNotificationMsg.parseFrom(msg.getData()), msg.getHeaders()));
|
||||||
consumerBuilder.admin(edgeEventAdmin);
|
consumerBuilder.admin(edgeEventAdmin);
|
||||||
|
|||||||
@ -95,6 +95,7 @@ public class KafkaTbCoreQueueFactory implements TbCoreQueueFactory {
|
|||||||
private final TbQueueAdmin edgeEventAdmin;
|
private final TbQueueAdmin edgeEventAdmin;
|
||||||
|
|
||||||
private final AtomicLong consumerCount = new AtomicLong();
|
private final AtomicLong consumerCount = new AtomicLong();
|
||||||
|
private final AtomicLong edgeConsumerCount = new AtomicLong();
|
||||||
|
|
||||||
public KafkaTbCoreQueueFactory(TopicService topicService,
|
public KafkaTbCoreQueueFactory(TopicService topicService,
|
||||||
TbKafkaSettings kafkaSettings,
|
TbKafkaSettings kafkaSettings,
|
||||||
@ -421,7 +422,7 @@ public class KafkaTbCoreQueueFactory implements TbCoreQueueFactory {
|
|||||||
TbKafkaConsumerTemplate.TbKafkaConsumerTemplateBuilder<TbProtoQueueMsg<ToEdgeEventNotificationMsg>> consumerBuilder = TbKafkaConsumerTemplate.builder();
|
TbKafkaConsumerTemplate.TbKafkaConsumerTemplateBuilder<TbProtoQueueMsg<ToEdgeEventNotificationMsg>> consumerBuilder = TbKafkaConsumerTemplate.builder();
|
||||||
consumerBuilder.settings(kafkaSettings);
|
consumerBuilder.settings(kafkaSettings);
|
||||||
consumerBuilder.topic(topicService.buildTopicName("tb_edge_event.notifications." + tenantId + "." + edgeId));
|
consumerBuilder.topic(topicService.buildTopicName("tb_edge_event.notifications." + tenantId + "." + edgeId));
|
||||||
consumerBuilder.clientId("tb-core-edge-event-consumer" + serviceInfoProvider.getServiceId());
|
consumerBuilder.clientId("tb-core-edge-event-consumer-" + serviceInfoProvider.getServiceId() + "-" + edgeConsumerCount.incrementAndGet());
|
||||||
consumerBuilder.groupId(topicService.buildTopicName("tb-core-edge-event-consumer"));
|
consumerBuilder.groupId(topicService.buildTopicName("tb-core-edge-event-consumer"));
|
||||||
consumerBuilder.decoder(msg -> new TbProtoQueueMsg<>(msg.getKey(), ToEdgeEventNotificationMsg.parseFrom(msg.getData()), msg.getHeaders()));
|
consumerBuilder.decoder(msg -> new TbProtoQueueMsg<>(msg.getKey(), ToEdgeEventNotificationMsg.parseFrom(msg.getData()), msg.getHeaders()));
|
||||||
consumerBuilder.admin(edgeEventAdmin);
|
consumerBuilder.admin(edgeEventAdmin);
|
||||||
|
|||||||
@ -963,7 +963,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
|
|||||||
unSubResults.add((short) MqttReasonCodes.UnsubAck.NO_SUBSCRIPTION_EXISTED.byteValue());
|
unSubResults.add((short) MqttReasonCodes.UnsubAck.NO_SUBSCRIPTION_EXISTED.byteValue());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!activityReported) {
|
if (!activityReported && !deviceSessionCtx.isProvisionOnly()) {
|
||||||
transportService.recordActivity(deviceSessionCtx.getSessionInfo());
|
transportService.recordActivity(deviceSessionCtx.getSessionInfo());
|
||||||
}
|
}
|
||||||
ctx.writeAndFlush(createUnSubAckMessage(mqttMsg.variableHeader().messageId(), unSubResults));
|
ctx.writeAndFlush(createUnSubAckMessage(mqttMsg.variableHeader().messageId(), unSubResults));
|
||||||
|
|||||||
@ -736,7 +736,11 @@ public class DefaultTransportService extends TransportActivityManager implements
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void recordActivityInternal(TransportProtos.SessionInfoProto sessionInfo) {
|
private void recordActivityInternal(TransportProtos.SessionInfoProto sessionInfo) {
|
||||||
onActivity(toSessionId(sessionInfo), sessionInfo, getCurrentTimeMillis());
|
if (sessionInfo != null) {
|
||||||
|
onActivity(toSessionId(sessionInfo), sessionInfo, getCurrentTimeMillis());
|
||||||
|
} else {
|
||||||
|
log.warn("Session info is missing, unable to record activity");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@ -343,7 +343,7 @@ public class TbHttpClient {
|
|||||||
if (CredentialsType.BASIC == credentials.getType()) {
|
if (CredentialsType.BASIC == credentials.getType()) {
|
||||||
BasicCredentials basicCredentials = (BasicCredentials) credentials;
|
BasicCredentials basicCredentials = (BasicCredentials) credentials;
|
||||||
String authString = basicCredentials.getUsername() + ":" + basicCredentials.getPassword();
|
String authString = basicCredentials.getUsername() + ":" + basicCredentials.getPassword();
|
||||||
String encodedAuthString = new String(Base64.getDecoder().decode(authString.getBytes(StandardCharsets.UTF_8)));
|
String encodedAuthString = new String(Base64.getEncoder().encode(authString.getBytes(StandardCharsets.UTF_8)));
|
||||||
headers.add("Authorization", "Basic " + encodedAuthString);
|
headers.add("Authorization", "Basic " + encodedAuthString);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -63,7 +63,7 @@
|
|||||||
</tb-color-input>
|
</tb-color-input>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="tb-form-row space-between flex-1 flex-col">
|
<div class="tb-form-row space-between flex-1 !flex-col">
|
||||||
<div class="tb-flex row space-between align-center no-gap fill-width">
|
<div class="tb-flex row space-between align-center no-gap fill-width">
|
||||||
<div class="fixed-title-width" translate>widgets.liquid-level-card.shape</div>
|
<div class="fixed-title-width" translate>widgets.liquid-level-card.shape</div>
|
||||||
<tb-toggle-select formControlName="tankSelectionType">
|
<tb-toggle-select formControlName="tankSelectionType">
|
||||||
|
|||||||
@ -18,7 +18,7 @@
|
|||||||
<ng-container *ngIf="levelCardWidgetSettingsForm" [formGroup]="levelCardWidgetSettingsForm">
|
<ng-container *ngIf="levelCardWidgetSettingsForm" [formGroup]="levelCardWidgetSettingsForm">
|
||||||
<div class="tb-form-panel no-padding no-border">
|
<div class="tb-form-panel no-padding no-border">
|
||||||
<div class="tb-form-panel">
|
<div class="tb-form-panel">
|
||||||
<div class="tb-form-row space-between flex flex-1 flex-col">
|
<div class="tb-form-row space-between flex flex-1 !flex-col">
|
||||||
<div class="tb-flex row space-between align-center no-gap fill-width">
|
<div class="tb-flex row space-between align-center no-gap fill-width">
|
||||||
<div class="fixed-title-width" translate>widgets.liquid-level-card.shape</div>
|
<div class="fixed-title-width" translate>widgets.liquid-level-card.shape</div>
|
||||||
<tb-toggle-select formControlName="tankSelectionType">
|
<tb-toggle-select formControlName="tankSelectionType">
|
||||||
|
|||||||
@ -22,7 +22,7 @@
|
|||||||
[class.!hidden]="isEdit || isDetailsPage">
|
[class.!hidden]="isEdit || isDetailsPage">
|
||||||
{{'common.open-details-page' | translate }}
|
{{'common.open-details-page' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'delete')"
|
(click)="onEntityAction($event, 'delete')"
|
||||||
[class.!hidden]="hideDelete() || isEdit">
|
[class.!hidden]="hideDelete() || isEdit">
|
||||||
|
|||||||
@ -22,13 +22,13 @@
|
|||||||
[class.!hidden]="isEdit || isDetailsPage">
|
[class.!hidden]="isEdit || isDetailsPage">
|
||||||
{{'common.open-details-page' | translate }}
|
{{'common.open-details-page' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'downloadResource')"
|
(click)="onEntityAction($event, 'downloadResource')"
|
||||||
[class.!hidden]="isEdit">
|
[class.!hidden]="isEdit">
|
||||||
{{ 'javascript.download' | translate }}
|
{{ 'javascript.download' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'delete')"
|
(click)="onEntityAction($event, 'delete')"
|
||||||
[class.!hidden]="hideDelete() || isEdit">
|
[class.!hidden]="hideDelete() || isEdit">
|
||||||
|
|||||||
@ -22,13 +22,13 @@
|
|||||||
[class.!hidden]="isEdit || isDetailsPage">
|
[class.!hidden]="isEdit || isDetailsPage">
|
||||||
{{'common.open-details-page' | translate }}
|
{{'common.open-details-page' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'downloadResource')"
|
(click)="onEntityAction($event, 'downloadResource')"
|
||||||
[class.!hidden]="isEdit">
|
[class.!hidden]="isEdit">
|
||||||
{{ 'resource.download' | translate }}
|
{{ 'resource.download' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'delete')"
|
(click)="onEntityAction($event, 'delete')"
|
||||||
[class.!hidden]="hideDelete() || isEdit">
|
[class.!hidden]="hideDelete() || isEdit">
|
||||||
|
|||||||
@ -58,7 +58,7 @@
|
|||||||
[class.!hidden]="isEdit || deviceScope !== 'edge'">
|
[class.!hidden]="isEdit || deviceScope !== 'edge'">
|
||||||
{{ 'edge.unassign-from-edge' | translate }}
|
{{ 'edge.unassign-from-edge' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'delete')"
|
(click)="onEntityAction($event, 'delete')"
|
||||||
[class.!hidden]="hideDelete() || isEdit">
|
[class.!hidden]="hideDelete() || isEdit">
|
||||||
|
|||||||
@ -70,7 +70,7 @@
|
|||||||
[class.!hidden]="isEdit || edgeScope !== 'tenant'">
|
[class.!hidden]="isEdit || edgeScope !== 'tenant'">
|
||||||
{{'edge.manage-rulechains' | translate }}
|
{{'edge.manage-rulechains' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'delete')"
|
(click)="onEntityAction($event, 'delete')"
|
||||||
[class.!hidden]="hideDelete() || isEdit">
|
[class.!hidden]="hideDelete() || isEdit">
|
||||||
|
|||||||
@ -16,19 +16,19 @@
|
|||||||
|
|
||||||
-->
|
-->
|
||||||
<div class="tb-details-buttons xs:flex xs:flex-col">
|
<div class="tb-details-buttons xs:flex xs:flex-col">
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'open')"
|
(click)="onEntityAction($event, 'open')"
|
||||||
[class.!hidden]="isEdit || isDetailsPage">
|
[class.!hidden]="isEdit || isDetailsPage">
|
||||||
{{'common.open-details-page' | translate }}
|
{{'common.open-details-page' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async) || !(entity?.hasData && !entity?.url)"
|
[disabled]="(isLoading$ | async) || !(entity?.hasData && !entity?.url)"
|
||||||
(click)="onEntityAction($event, 'uploadPackage')"
|
(click)="onEntityAction($event, 'uploadPackage')"
|
||||||
[class.!hidden]="isEdit">
|
[class.!hidden]="isEdit">
|
||||||
{{ 'ota-update.download' | translate }}
|
{{ 'ota-update.download' | translate }}
|
||||||
</button>
|
</button>
|
||||||
<button mat-raised-button color="primary" class="xs:flex-1"
|
<button mat-raised-button color="primary"
|
||||||
[disabled]="(isLoading$ | async)"
|
[disabled]="(isLoading$ | async)"
|
||||||
(click)="onEntityAction($event, 'delete')"
|
(click)="onEntityAction($event, 'delete')"
|
||||||
[class.!hidden]="hideDelete() || isEdit">
|
[class.!hidden]="hideDelete() || isEdit">
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user