com.espertech.esper.runtime.internal.schedulesvcimpl.SchedulingServiceImpl Maven / Gradle / Ivy
/*
***************************************************************************************
* Copyright (C) 2006 EsperTech, Inc. All rights reserved. *
* http://www.espertech.com/esper *
* http://www.espertech.com *
* ---------------------------------------------------------------------------------- *
* The software in this package is published under the terms of the GPL license *
* a copy of which has been included with this distribution in the license.txt file. *
***************************************************************************************
*/
package com.espertech.esper.runtime.internal.schedulesvcimpl;
import com.espertech.esper.common.client.util.DateTime;
import com.espertech.esper.common.internal.schedule.ScheduleHandle;
import com.espertech.esper.common.internal.schedule.ScheduleServiceException;
import com.espertech.esper.common.internal.schedule.TimeSourceService;
import com.espertech.esper.runtime.internal.metrics.instrumentation.InstrumentationHelper;
import com.espertech.esper.runtime.internal.metrics.jmx.JmxGetter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
import java.util.*;
/**
* Implements the schedule service by simply keeping a sorted set of long millisecond
* values and a set of handles for each.
*
* Synchronized since statement creation and event evaluation by multiple (event send) threads
* can lead to callbacks added/removed asynchronously.
*/
public final class SchedulingServiceImpl implements SchedulingServiceSPI {
private final int stageId;
private final DateTimeFormatter defaultDateTimeFormatter;
// Map of time and handle
private final SortedMap> timeHandleMap;
// Map of handle and handle list for faster removal
private final Map> handleSetMap;
// Current time - used for evaluation as well as for adding new handles
private volatile long currentTime;
/**
* Constructor.
*
* @param timeSourceService time source provider
* @param stageId stage id or -1 when not applicable
* @param defaultFormatterTimeZone time zone for audit formatter
*/
public SchedulingServiceImpl(int stageId, TimeSourceService timeSourceService, ZoneId defaultFormatterTimeZone) {
this.stageId = stageId;
this.defaultDateTimeFormatter = DateTimeFormatter.ofPattern(DateTime.DEFAULT_XMLLIKE_DATE_FORMAT).withZone(defaultFormatterTimeZone);
this.timeHandleMap = new TreeMap>();
this.handleSetMap = new HashMap>();
// initialize time to just before now as there is a check for duplicate external time events
this.currentTime = timeSourceService.getTimeMillis() - 1;
}
public void destroy() {
log.debug("Destroying scheduling service");
handleSetMap.clear();
timeHandleMap.clear();
}
public long getTime() {
// note that this.currentTime is volatile
return this.currentTime;
}
public synchronized final void setTime(long currentTime) {
this.currentTime = currentTime;
}
public synchronized final void add(long afterTime, ScheduleHandle handle, long slot)
throws ScheduleServiceException {
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().qScheduleAdd(currentTime, afterTime, handle, slot);
}
if (handleSetMap.containsKey(handle)) {
remove(handle, slot);
}
long triggerOnTime = currentTime + afterTime;
addTrigger(slot, handle, triggerOnTime);
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aScheduleAdd();
}
}
public synchronized final void remove(ScheduleHandle handle, long slot) {
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().qScheduleRemove(handle, slot);
}
SortedMap handleSet = handleSetMap.get(handle);
if (handleSet == null) {
// If it already has been removed then that's fine;
// Such could be the case when 2 timers fireStatementStopped at the same time, and one stops the other
return;
}
handleSet.remove(slot);
handleSetMap.remove(handle);
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aScheduleRemove();
}
}
public synchronized final void evaluate(Collection handles) {
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().qScheduleEval(currentTime);
}
// Get the values on or before the current time - to get those that are exactly on the
// current time we just add one to the current time for getting the head map
SortedMap> headMap = timeHandleMap.headMap(currentTime + 1);
if (headMap.isEmpty()) {
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aScheduleEval(handles);
}
return;
}
// First determine all triggers to shoot
List removeKeys = new ArrayList();
for (Map.Entry> entry : headMap.entrySet()) {
Long key = entry.getKey();
SortedMap value = entry.getValue();
removeKeys.add(key);
for (ScheduleHandle handle : value.values()) {
handles.add(handle);
}
}
// Next remove all handles
for (Map.Entry> entry : headMap.entrySet()) {
for (ScheduleHandle handle : entry.getValue().values()) {
handleSetMap.remove(handle);
}
}
// Remove all triggered msec values
for (Long key : removeKeys) {
timeHandleMap.remove(key);
}
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aScheduleEval(handles);
}
}
public void transfer(Set statementIds, SchedulingServiceSPI schedulingService) {
long currentTime = getTime();
long targetTime = schedulingService.getTime();
for (Map.Entry> schedule : timeHandleMap.entrySet()) {
for (Map.Entry entry : schedule.getValue().entrySet()) {
if (statementIds.contains(entry.getValue().getStatementId())) {
long relative = ScheduleTransferHelper.computeTransferTime(currentTime, targetTime, schedule.getKey());
remove(entry.getValue(), entry.getKey());
schedulingService.add(relative, entry.getValue(), entry.getKey());
}
}
}
}
public void init() {
// no action required
}
private void addTrigger(long slot, ScheduleHandle handle, long triggerTime) {
SortedMap handleSet = timeHandleMap.get(triggerTime);
if (handleSet == null) {
handleSet = new TreeMap();
timeHandleMap.put(triggerTime, handleSet);
}
handleSet.put(slot, handle);
handleSetMap.put(handle, handleSet);
}
@JmxGetter(name = "TimeHandleCount", description = "Number of outstanding time evaluations")
public int getTimeHandleCount() {
return timeHandleMap.size();
}
@JmxGetter(name = "FurthestTimeHandle", description = "Furthest outstanding time evaluation")
public String getFurthestTimeHandleDate() {
Long handle = getFurthestTimeHandle();
if (handle != null) {
return DateTime.print(handle);
}
return null;
}
@JmxGetter(name = "NearestTimeHandle", description = "Nearest outstanding time evaluation")
public String getNearestTimeHandleDate() {
Long handle = getNearestTimeHandle();
if (handle != null) {
return DateTime.print(handle);
}
return null;
}
public Long getFurthestTimeHandle() {
if (!timeHandleMap.isEmpty()) {
return timeHandleMap.lastKey();
}
return null;
}
public int getScheduleHandleCount() {
return handleSetMap.size();
}
public boolean isScheduled(ScheduleHandle handle) {
return handleSetMap.containsKey(handle);
}
@Override
public synchronized Long getNearestTimeHandle() {
if (timeHandleMap.isEmpty()) {
return null;
}
for (Map.Entry> entry : timeHandleMap.entrySet()) {
if (entry.getValue().isEmpty()) {
continue;
}
return entry.getKey();
}
return null;
}
public void visitSchedules(ScheduleVisitor visitor) {
ScheduleVisit visit = new ScheduleVisit();
for (Map.Entry> entry : timeHandleMap.entrySet()) {
visit.setTimestamp(entry.getKey());
for (Map.Entry inner : entry.getValue().entrySet()) {
visit.setStatementId(inner.getValue().getStatementId());
visit.setAgentInstanceId(inner.getValue().getAgentInstanceId());
visitor.visit(visit);
}
}
}
public DateTimeFormatter getDefaultFormatter() {
return defaultDateTimeFormatter;
}
private static final Logger log = LoggerFactory.getLogger(SchedulingServiceImpl.class);
}