default state service: refactored events order
This commit is contained in:
parent
a9c655b179
commit
133eddb855
@ -134,7 +134,7 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
private int initFetchPackSize;
|
||||
|
||||
private ListeningScheduledExecutorService scheduledExecutor;
|
||||
private ExecutorService dbCallbackExecutorService;
|
||||
private ExecutorService deviceStateExecutor;
|
||||
private final ConcurrentMap<TopicPartitionInfo, Set<DeviceId>> partitionedDevices = new ConcurrentHashMap<>();
|
||||
private final ConcurrentMap<DeviceId, DeviceStateData> deviceStates = new ConcurrentHashMap<>();
|
||||
private final ConcurrentMap<DeviceId, Long> deviceLastSavedActivity = new ConcurrentHashMap<>();
|
||||
@ -159,7 +159,7 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
|
||||
@PostConstruct
|
||||
public void init() {
|
||||
dbCallbackExecutorService = Executors.newFixedThreadPool(
|
||||
deviceStateExecutor = Executors.newFixedThreadPool(
|
||||
Runtime.getRuntime().availableProcessors(), ThingsBoardThreadFactory.forName("device-state"));
|
||||
// Should be always single threaded due to absence of locks.
|
||||
scheduledExecutor = MoreExecutors.listeningDecorator(Executors.newSingleThreadScheduledExecutor(ThingsBoardThreadFactory.forName("device-state-scheduled")));
|
||||
@ -168,8 +168,8 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
|
||||
@PreDestroy
|
||||
public void stop() {
|
||||
if (dbCallbackExecutorService != null) {
|
||||
dbCallbackExecutorService.shutdownNow();
|
||||
if (deviceStateExecutor != null) {
|
||||
deviceStateExecutor.shutdownNow();
|
||||
}
|
||||
if (scheduledExecutor != null) {
|
||||
scheduledExecutor.shutdownNow();
|
||||
@ -193,13 +193,13 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
|
||||
@Override
|
||||
public void onDeviceConnect(DeviceId deviceId) {
|
||||
log.trace("on Device Connect [{}]", deviceId.getId());
|
||||
DeviceStateData stateData = getOrFetchDeviceStateData(deviceId);
|
||||
if (stateData != null) {
|
||||
long ts = System.currentTimeMillis();
|
||||
stateData.getState().setLastConnectTime(ts);
|
||||
pushRuleEngineMessage(stateData, CONNECT_EVENT);
|
||||
save(deviceId, LAST_CONNECT_TIME, ts);
|
||||
}
|
||||
pushRuleEngineMessage(stateData, CONNECT_EVENT);
|
||||
checkAndUpdateState(deviceId, stateData);
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -232,32 +232,21 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
@Override
|
||||
public void onDeviceDisconnect(DeviceId deviceId) {
|
||||
DeviceStateData stateData = getOrFetchDeviceStateData(deviceId);
|
||||
if (stateData != null) {
|
||||
long ts = System.currentTimeMillis();
|
||||
stateData.getState().setLastDisconnectTime(ts);
|
||||
pushRuleEngineMessage(stateData, DISCONNECT_EVENT);
|
||||
save(deviceId, LAST_DISCONNECT_TIME, ts);
|
||||
}
|
||||
pushRuleEngineMessage(stateData, DISCONNECT_EVENT);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDeviceInactivityTimeoutUpdate(DeviceId deviceId, long inactivityTimeout) {
|
||||
if (inactivityTimeout == 0L) {
|
||||
if (inactivityTimeout <= 0L) {
|
||||
return;
|
||||
}
|
||||
log.trace("on Device Activity Timeout Update device id {} inactivityTimeout {}", deviceId, inactivityTimeout);
|
||||
DeviceStateData stateData = deviceStates.get(deviceId);
|
||||
log.trace("on Device Activity Timeout Update - fetched state {} for device {}", stateData, deviceId);
|
||||
if (stateData != null) {
|
||||
long ts = System.currentTimeMillis();
|
||||
DeviceState state = stateData.getState();
|
||||
state.setInactivityTimeout(inactivityTimeout);
|
||||
boolean oldActive = state.isActive();
|
||||
state.setActive(isActive(ts, state));
|
||||
if (!oldActive && state.isActive() || oldActive && !state.isActive()) {
|
||||
save(deviceId, ACTIVITY_STATE, state.isActive());
|
||||
}
|
||||
}
|
||||
DeviceStateData stateData = getOrFetchDeviceStateData(deviceId);
|
||||
stateData.getState().setInactivityTimeout(inactivityTimeout);
|
||||
checkAndUpdateState(deviceId, stateData);
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -291,7 +280,7 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
log.warn("Failed to register device to the state service", t);
|
||||
callback.onFailure(t);
|
||||
}
|
||||
}, dbCallbackExecutorService);
|
||||
}, deviceStateExecutor);
|
||||
} else if (proto.getUpdated()) {
|
||||
DeviceStateData stateData = getOrFetchDeviceStateData(device.getId());
|
||||
if (stateData != null) {
|
||||
@ -396,11 +385,16 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
List<Tenant> tenants = tenantService.findTenants(new PageLink(Integer.MAX_VALUE)).getData();
|
||||
for (Tenant tenant : tenants) {
|
||||
log.debug("Finding devices for tenant [{}]", tenant.getName());
|
||||
PageLink pageLink = new PageLink(initFetchPackSize);
|
||||
while (pageLink != null) {
|
||||
final PageLink pageLink = new PageLink(initFetchPackSize);
|
||||
scheduledExecutor.submit(() -> submitNextPage(addedPartitions, tenant, pageLink, scheduledExecutor));
|
||||
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
private void submitNextPage(final Set<TopicPartitionInfo> addedPartitions, final Tenant tenant, final PageLink pageLink, final ExecutorService executor) {
|
||||
List<ListenableFuture<Void>> fetchFutures = new ArrayList<>();
|
||||
PageData<Device> page = deviceService.findDevicesByTenantId(tenant.getId(), pageLink);
|
||||
pageLink = page.hasNext() ? pageLink.nextPageLink() : null;
|
||||
for (Device device : page.getData()) {
|
||||
TopicPartitionInfo tpi = partitionService.resolve(ServiceType.TB_CORE, tenant.getId(), device.getId());
|
||||
if (addedPartitions.contains(tpi)) {
|
||||
@ -411,34 +405,47 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
public Void apply(@Nullable DeviceStateData state) {
|
||||
if (state != null) {
|
||||
addDeviceUsingState(tpi, state);
|
||||
|
||||
if (state.getState().isActive()) {
|
||||
updateInactivityStateIfExpired(System.currentTimeMillis(), device.getId(), state);
|
||||
} else {
|
||||
//trying to fix activity state
|
||||
if (isActive(System.currentTimeMillis(), state.getState())) {
|
||||
updateActivityState(device.getId(), state, state.getState().getLastActivityTime());
|
||||
}
|
||||
}
|
||||
checkAndUpdateState(device.getId(), state);
|
||||
} else {
|
||||
log.warn("{}][{}] Fetched null state from DB", device.getName(), device.getId());
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}, dbCallbackExecutorService);
|
||||
}, deviceStateExecutor);
|
||||
fetchFutures.add(future);
|
||||
} else {
|
||||
log.debug("[{}][{}] Device doesn't belong to current partition. tpi [{}]", device.getName(), device.getId(), tpi);
|
||||
}
|
||||
}
|
||||
try {
|
||||
Futures.successfulAsList(fetchFutures).get();
|
||||
} catch (InterruptedException | ExecutionException e) {
|
||||
log.warn("Failed to init device state service from DB", e);
|
||||
|
||||
Futures.addCallback(Futures.successfulAsList(fetchFutures), new FutureCallback<List<Void>>() {
|
||||
@Override
|
||||
public void onSuccess(List<Void> result) {
|
||||
log.trace("[{}] Success init device state from DB for batch size {}", tenant.getId(), result.size());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onFailure(Throwable t) {
|
||||
log.warn("[" + tenant.getId() + "] Failed to init device state service from DB", t);
|
||||
log.warn("[{}] Failed to init device state service from DB", tenant.getId(), t);
|
||||
}
|
||||
}, deviceStateExecutor);
|
||||
|
||||
final PageLink nextPageLink = page.hasNext() ? pageLink.nextPageLink() : null;
|
||||
if (nextPageLink != null) {
|
||||
executor.submit(() -> submitNextPage(addedPartitions, tenant, nextPageLink, executor));
|
||||
}
|
||||
}
|
||||
|
||||
void checkAndUpdateState(@Nonnull DeviceId deviceId, @Nonnull DeviceStateData state) {
|
||||
if (state.getState().isActive()) {
|
||||
updateInactivityStateIfExpired(System.currentTimeMillis(), deviceId, state);
|
||||
} else {
|
||||
//trying to fix activity state
|
||||
if (isActive(System.currentTimeMillis(), state.getState())) {
|
||||
updateActivityState(deviceId, state, state.getState().getLastActivityTime());
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
private void addDeviceUsingState(TopicPartitionInfo tpi, DeviceStateData state) {
|
||||
@ -467,9 +474,9 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
if (!isActive(ts, state) && (state.getLastInactivityAlarmTime() == 0L || state.getLastInactivityAlarmTime() < state.getLastActivityTime()) && stateData.getDeviceCreationTime() + state.getInactivityTimeout() < ts) {
|
||||
state.setActive(false);
|
||||
state.setLastInactivityAlarmTime(ts);
|
||||
pushRuleEngineMessage(stateData, INACTIVITY_EVENT);
|
||||
save(deviceId, INACTIVITY_ALARM_TIME, ts);
|
||||
save(deviceId, ACTIVITY_STATE, false);
|
||||
pushRuleEngineMessage(stateData, INACTIVITY_EVENT);
|
||||
}
|
||||
} else {
|
||||
log.debug("[{}] Device that belongs to other server is detected and removed.", deviceId);
|
||||
@ -484,7 +491,7 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
|
||||
@Nonnull
|
||||
private DeviceStateData getOrFetchDeviceStateData(DeviceId deviceId) {
|
||||
DeviceStateData deviceStateData = deviceStates.get(deviceId);
|
||||
DeviceStateData deviceStateData = getOrFetchDeviceStateData(deviceId);
|
||||
if (deviceStateData != null) {
|
||||
return deviceStateData;
|
||||
}
|
||||
@ -530,10 +537,10 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
ListenableFuture<DeviceStateData> future;
|
||||
if (persistToTelemetry) {
|
||||
ListenableFuture<List<TsKvEntry>> tsData = tsService.findLatest(TenantId.SYS_TENANT_ID, device.getId(), PERSISTENT_ATTRIBUTES);
|
||||
future = Futures.transform(tsData, extractDeviceStateData(device), dbCallbackExecutorService);
|
||||
future = Futures.transform(tsData, extractDeviceStateData(device), deviceStateExecutor);
|
||||
} else {
|
||||
ListenableFuture<List<AttributeKvEntry>> attrData = attributesService.find(TenantId.SYS_TENANT_ID, device.getId(), DataConstants.SERVER_SCOPE, PERSISTENT_ATTRIBUTES);
|
||||
future = Futures.transform(attrData, extractDeviceStateData(device), dbCallbackExecutorService);
|
||||
future = Futures.transform(attrData, extractDeviceStateData(device), deviceStateExecutor);
|
||||
}
|
||||
return transformInactivityTimeout(future);
|
||||
}
|
||||
@ -563,9 +570,9 @@ public class DefaultDeviceStateService extends TbApplicationEventListener<Partit
|
||||
resultFuture.setException(t);
|
||||
}
|
||||
},
|
||||
dbCallbackExecutorService);
|
||||
deviceStateExecutor);
|
||||
return resultFuture;
|
||||
}, dbCallbackExecutorService);
|
||||
}, deviceStateExecutor);
|
||||
}
|
||||
|
||||
private <T extends KvEntry> Function<List<T>, DeviceStateData> extractDeviceStateData(Device device) {
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user