com.espertech.esper.runtime.internal.namedwindow.NamedWindowDispatchServiceImpl 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.namedwindow;
import com.espertech.esper.common.client.EPException;
import com.espertech.esper.common.client.EventBean;
import com.espertech.esper.common.client.hook.exception.ExceptionHandlerExceptionType;
import com.espertech.esper.common.internal.context.util.EPStatementAgentInstanceHandle;
import com.espertech.esper.common.internal.epl.namedwindow.consume.*;
import com.espertech.esper.common.internal.epl.table.core.TableManagementService;
import com.espertech.esper.common.internal.epl.variable.core.VariableManagementService;
import com.espertech.esper.common.internal.metrics.stmtmetrics.MetricReportingService;
import com.espertech.esper.common.internal.schedule.SchedulingService;
import com.espertech.esper.common.internal.settings.ExceptionHandlingService;
import com.espertech.esper.common.internal.util.ManagedReadWriteLock;
import com.espertech.esper.common.internal.util.MetricUtil;
import com.espertech.esper.runtime.internal.metrics.instrumentation.InstrumentationHelper;
import java.util.*;
/**
* This service hold for each named window a dedicated processor and a lock to the named window.
* This lock is shrared between the named window and on-delete statements.
*/
public class NamedWindowDispatchServiceImpl implements NamedWindowDispatchService {
private final SchedulingService schedulingService;
private final VariableManagementService variableService;
private final TableManagementService tableManagementService;
private final ExceptionHandlingService exceptionHandlingService;
private final boolean isPrioritized;
private final ManagedReadWriteLock eventProcessingRWLock;
private final MetricReportingService metricReportingService;
private ThreadLocal threadLocal = new ThreadLocal() {
protected synchronized DispatchesTL initialValue() {
return new DispatchesTL();
}
};
public NamedWindowDispatchServiceImpl(SchedulingService schedulingService,
VariableManagementService variableService,
TableManagementService tableManagementService,
boolean isPrioritized,
ManagedReadWriteLock eventProcessingRWLock,
ExceptionHandlingService exceptionHandlingService,
MetricReportingService metricReportingService) {
this.schedulingService = schedulingService;
this.variableService = variableService;
this.tableManagementService = tableManagementService;
this.isPrioritized = isPrioritized;
this.eventProcessingRWLock = eventProcessingRWLock;
this.exceptionHandlingService = exceptionHandlingService;
this.metricReportingService = metricReportingService;
}
public void destroy() {
threadLocal.remove();
}
public void addDispatch(NamedWindowConsumerLatchFactory latchFactory, NamedWindowDeltaData delta, Map> consumers) {
NamedWindowConsumerLatch latch = latchFactory.newLatch(delta, consumers);
threadLocal.get().getDispatches().add(latch);
}
public boolean dispatch() {
DispatchesTL dispatchesTL = threadLocal.get();
if (dispatchesTL.getDispatches().isEmpty()) {
return false;
}
while (!dispatchesTL.getDispatches().isEmpty()) {
// Acquire main processing lock which locks out statement management
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().qNamedWindowDispatch(exceptionHandlingService.getRuntimeURI());
}
eventProcessingRWLock.acquireReadLock();
try {
// since dispatches can cause dispatches, copy the contents
dispatchesTL.getCurrent().addAll(dispatchesTL.getDispatches());
dispatchesTL.getDispatches().clear();
processDispatches(dispatchesTL.getCurrent(), dispatchesTL.getWork(), dispatchesTL.getDispatchesPerStmt());
} catch (RuntimeException ex) {
throw new EPException(ex);
} finally {
dispatchesTL.getCurrent().clear();
eventProcessingRWLock.releaseReadLock();
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aNamedWindowDispatch();
}
}
}
return true;
}
private void processDispatches(ArrayDeque dispatches, ArrayDeque work, Map dispatchesPerStmt) {
if (dispatches.size() == 1) {
NamedWindowConsumerLatch latch = dispatches.getFirst();
try {
latch.await();
EventBean[] newData = latch.getDeltaData().getNewData();
EventBean[] oldData = latch.getDeltaData().getOldData();
if (metricReportingService.isMetricsReportingEnabled()) {
for (Map.Entry> entry : latch.getDispatchTo().entrySet()) {
EPStatementAgentInstanceHandle handle = entry.getKey();
if (handle.getStatementHandle().getMetricsHandle().isEnabled()) {
long cpuTimeBefore = MetricUtil.getCPUCurrentThread();
long wallTimeBefore = MetricUtil.getWall();
processHandle(handle, entry.getValue(), newData, oldData);
long wallTimeAfter = MetricUtil.getWall();
long cpuTimeAfter = MetricUtil.getCPUCurrentThread();
long deltaCPU = cpuTimeAfter - cpuTimeBefore;
long deltaWall = wallTimeAfter - wallTimeBefore;
metricReportingService.accountTime(handle.getStatementHandle().getMetricsHandle(), deltaCPU, deltaWall, 1);
} else {
processHandle(handle, entry.getValue(), newData, oldData);
}
if (isPrioritized && handle.isPreemptive()) {
break;
}
}
} else {
for (Map.Entry> entry : latch.getDispatchTo().entrySet()) {
EPStatementAgentInstanceHandle handle = entry.getKey();
processHandle(handle, entry.getValue(), newData, oldData);
if (isPrioritized && handle.isPreemptive()) {
break;
}
}
}
} finally {
latch.done();
}
return;
}
// Multiple different-result dispatches to same or different statements are needed in two situations:
// a) an event comes in, triggers two insert-into statements inserting into the same named window and the window produces 2 results
// b) a time batch is grouped in the named window, and a timer fires for both groups at the same time producing more then one result
// c) two on-merge/update/delete statements fire for the same arriving event each updating the named window
// Most likely all dispatches go to different statements since most statements are not joins of
// named windows that produce results at the same time. Therefore sort by statement handle.
// We need to process in N-element chains to preserve dispatches that are next to each other for the same thread.
while (!dispatches.isEmpty()) {
// the first latch always gets awaited
NamedWindowConsumerLatch first = dispatches.removeFirst();
first.await();
work.add(first);
// determine which further latches are in this chain and add these, skipping await for any latches in the chain
Iterator it = dispatches.iterator();
while (it.hasNext()) {
NamedWindowConsumerLatch next = it.next();
NamedWindowConsumerLatch earlier = next.getEarlier();
if (earlier == null || work.contains(earlier)) {
work.add(next);
it.remove();
} else {
break;
}
}
processDispatches(work, dispatchesPerStmt);
}
}
private void processDispatches(ArrayDeque dispatches, Map dispatchesPerStmt) {
try {
for (NamedWindowConsumerLatch latch : dispatches) {
for (Map.Entry> entry : latch.getDispatchTo().entrySet()) {
EPStatementAgentInstanceHandle handle = entry.getKey();
Object perStmtObj = dispatchesPerStmt.get(handle);
if (perStmtObj == null) {
dispatchesPerStmt.put(handle, latch);
} else if (perStmtObj instanceof List) {
List list = (List) perStmtObj;
list.add(latch);
} else {
// convert from object to list
NamedWindowConsumerLatch unitObj = (NamedWindowConsumerLatch) perStmtObj;
List list = new ArrayList();
list.add(unitObj);
list.add(latch);
dispatchesPerStmt.put(handle, list);
}
}
}
// Dispatch - with or without metrics reporting
if (metricReportingService.isMetricsReportingEnabled()) {
for (Map.Entry entry : dispatchesPerStmt.entrySet()) {
EPStatementAgentInstanceHandle handle = entry.getKey();
Object perStmtObj = entry.getValue();
// dispatch of a single result to the statement
if (perStmtObj instanceof NamedWindowConsumerLatch) {
NamedWindowConsumerLatch unit = (NamedWindowConsumerLatch) perStmtObj;
EventBean[] newData = unit.getDeltaData().getNewData();
EventBean[] oldData = unit.getDeltaData().getOldData();
if (handle.getStatementHandle().getMetricsHandle().isEnabled()) {
long cpuTimeBefore = MetricUtil.getCPUCurrentThread();
long wallTimeBefore = MetricUtil.getWall();
processHandle(handle, unit.getDispatchTo().get(handle), newData, oldData);
long wallTimeAfter = MetricUtil.getWall();
long cpuTimeAfter = MetricUtil.getCPUCurrentThread();
long deltaCPU = cpuTimeAfter - cpuTimeBefore;
long deltaWall = wallTimeAfter - wallTimeBefore;
metricReportingService.accountTime(handle.getStatementHandle().getMetricsHandle(), deltaCPU, deltaWall, 1);
} else {
Map> entries = unit.getDispatchTo();
List items = entries.get(handle);
if (items != null) {
processHandle(handle, items, newData, oldData);
}
}
if (isPrioritized && handle.isPreemptive()) {
break;
}
continue;
}
// dispatch of multiple results to a the same statement, need to aggregate per consumer view
LinkedHashMap deltaPerConsumer = getDeltaPerConsumer(perStmtObj, handle);
if (handle.getStatementHandle().getMetricsHandle().isEnabled()) {
long cpuTimeBefore = MetricUtil.getCPUCurrentThread();
long wallTimeBefore = MetricUtil.getWall();
processHandleMultiple(handle, deltaPerConsumer);
long wallTimeAfter = MetricUtil.getWall();
long cpuTimeAfter = MetricUtil.getCPUCurrentThread();
long deltaCPU = cpuTimeAfter - cpuTimeBefore;
long deltaWall = wallTimeAfter - wallTimeBefore;
metricReportingService.accountTime(handle.getStatementHandle().getMetricsHandle(), deltaCPU, deltaWall, 1);
} else {
processHandleMultiple(handle, deltaPerConsumer);
}
if (isPrioritized && handle.isPreemptive()) {
break;
}
}
} else {
for (Map.Entry entry : dispatchesPerStmt.entrySet()) {
EPStatementAgentInstanceHandle handle = entry.getKey();
Object perStmtObj = entry.getValue();
// dispatch of a single result to the statement
if (perStmtObj instanceof NamedWindowConsumerLatch) {
NamedWindowConsumerLatch unit = (NamedWindowConsumerLatch) perStmtObj;
EventBean[] newData = unit.getDeltaData().getNewData();
EventBean[] oldData = unit.getDeltaData().getOldData();
processHandle(handle, unit.getDispatchTo().get(handle), newData, oldData);
if (isPrioritized && handle.isPreemptive()) {
break;
}
continue;
}
// dispatch of multiple results to a the same statement, need to aggregate per consumer view
LinkedHashMap deltaPerConsumer = getDeltaPerConsumer(perStmtObj, handle);
processHandleMultiple(handle, deltaPerConsumer);
if (isPrioritized && handle.isPreemptive()) {
break;
}
}
}
} finally {
for (NamedWindowConsumerLatch latch : dispatches) {
latch.done();
}
dispatchesPerStmt.clear();
dispatches.clear();
}
}
private void processHandleMultiple(EPStatementAgentInstanceHandle handle, Map deltaPerConsumer) {
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().qNamedWindowCPMulti(exceptionHandlingService.getRuntimeURI(), deltaPerConsumer, handle, schedulingService.getTime());
}
handle.getStatementAgentInstanceLock().acquireWriteLock();
try {
if (handle.isHasVariables()) {
variableService.setLocalVersion();
}
for (Map.Entry entryDelta : deltaPerConsumer.entrySet()) {
EventBean[] newData = entryDelta.getValue().getNewData();
EventBean[] oldData = entryDelta.getValue().getOldData();
entryDelta.getKey().update(newData, oldData);
}
// internal join processing, if applicable
handle.internalDispatch();
} catch (RuntimeException ex) {
exceptionHandlingService.handleException(ex, handle, ExceptionHandlerExceptionType.PROCESS, null);
} finally {
if (handle.isHasTableAccess()) {
tableManagementService.getTableExprEvaluatorContext().releaseAcquiredLocks();
}
handle.getStatementAgentInstanceLock().releaseWriteLock();
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aNamedWindowCPMulti();
}
}
}
private void processHandle(EPStatementAgentInstanceHandle handle, List value, EventBean[] newData, EventBean[] oldData) {
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().qNamedWindowCPSingle(exceptionHandlingService.getRuntimeURI(), value.size(), newData, oldData, handle, schedulingService.getTime());
}
handle.getStatementAgentInstanceLock().acquireWriteLock();
try {
if (handle.isHasVariables()) {
variableService.setLocalVersion();
}
for (NamedWindowConsumerView consumerView : value) {
consumerView.update(newData, oldData);
}
// internal join processing, if applicable
handle.internalDispatch();
} catch (RuntimeException ex) {
exceptionHandlingService.handleException(ex, handle, ExceptionHandlerExceptionType.PROCESS, null);
} finally {
if (handle.isHasTableAccess()) {
tableManagementService.getTableExprEvaluatorContext().releaseAcquiredLocks();
}
handle.getStatementAgentInstanceLock().releaseWriteLock();
if (InstrumentationHelper.ENABLED) {
InstrumentationHelper.get().aNamedWindowCPSingle();
}
}
}
public LinkedHashMap getDeltaPerConsumer(Object perStmtObj, EPStatementAgentInstanceHandle handle) {
List list = (List) perStmtObj;
LinkedHashMap deltaPerConsumer = new LinkedHashMap();
for (NamedWindowConsumerLatch unit : list) {
// for each unit
for (NamedWindowConsumerView consumerView : unit.getDispatchTo().get(handle)) {
// each consumer
NamedWindowDeltaData deltaForConsumer = deltaPerConsumer.get(consumerView);
if (deltaForConsumer == null) {
deltaPerConsumer.put(consumerView, unit.getDeltaData());
} else {
NamedWindowDeltaData aggregated = new NamedWindowDeltaData(deltaForConsumer, unit.getDeltaData());
deltaPerConsumer.put(consumerView, aggregated);
}
}
}
return deltaPerConsumer;
}
private static class DispatchesTL {
private final ArrayDeque dispatches = new ArrayDeque<>();
private final ArrayDeque current = new ArrayDeque<>();
private final ArrayDeque work = new ArrayDeque<>();
private final Map dispatchesPerStmt = new HashMap<>();
public ArrayDeque getDispatches() {
return dispatches;
}
public ArrayDeque getCurrent() {
return current;
}
public ArrayDeque getWork() {
return work;
}
public Map getDispatchesPerStmt() {
return dispatchesPerStmt;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy