All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.espertech.esper.runtime.internal.schedulesvcimpl.SchedulingServiceImpl Maven / Gradle / Ivy

The newest version!
/*
 ***************************************************************************************
 *  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); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy