com.espertech.esper.pattern.EvalFollowedByWithMaxStateNode Maven / Gradle / Ivy
/**************************************************************************************
* Copyright (C) 2008 EsperTech, Inc. All rights reserved. *
* http://esper.codehaus.org *
* 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.pattern;
import com.espertech.esper.client.hook.ConditionPatternSubexpressionMax;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import java.util.HashMap;
/**
* This class represents the state of a followed-by operator in the evaluation state tree.
*/
public final class EvalFollowedByWithMaxStateNode extends EvalStateNode implements Evaluator
{
private final EvalFollowedByNode evalFollowedByNode;
private final HashMap nodes;
private final int[] countActivePerChild;
/**
* Constructor.
* @param parentNode is the parent evaluator to call to indicate truth value
* @param beginState contains the events that make up prior matches
* @param evalFollowedByNode is the factory node associated to the state
*/
public EvalFollowedByWithMaxStateNode(Evaluator parentNode,
EvalFollowedByNode evalFollowedByNode,
MatchedEventMap beginState)
{
super(parentNode, null);
this.evalFollowedByNode = evalFollowedByNode;
this.nodes = new HashMap();
this.countActivePerChild = new int[evalFollowedByNode.getChildNodes().size() - 1];
EvalNode child = evalFollowedByNode.getChildNodes().get(0);
EvalStateNode childState = child.newState(this, beginState, evalFollowedByNode.getContext(), null);
nodes.put(childState, 0);
}
@Override
public EvalNode getFactoryNode() {
return evalFollowedByNode;
}
public final void start()
{
if (nodes.isEmpty())
{
throw new IllegalStateException("Followed by state node is inactive");
}
for (EvalStateNode child : nodes.keySet())
{
child.start();
}
}
public final void evaluateTrue(MatchedEventMap matchEvent, EvalStateNode fromNode, boolean isQuitted)
{
Integer index = nodes.get(fromNode);
if (isQuitted)
{
nodes.remove(fromNode);
if (index != null && index > 0) {
countActivePerChild[index - 1]--;
}
}
// the node may already have quit as a result of an outer state quitting this state,
// however the callback may still be received; It is fine to ignore this callback.
if (index == null)
{
return;
}
// If the match came from the very last filter, need to escalate
int numChildNodes = getFactoryNode().getChildNodes().size();
if (index == (numChildNodes - 1))
{
boolean isFollowedByQuitted = false;
if (nodes.isEmpty())
{
isFollowedByQuitted = true;
}
this.getParentEvaluator().evaluateTrue(matchEvent, this, isFollowedByQuitted);
}
// Else start a new sub-expression for the next-in-line filter
else
{
int max = evalFollowedByNode.getMax(index);
if ((max != -1) && (max >=0)) {
if (countActivePerChild[index] >= max) {
evalFollowedByNode.getContext().getExceptionHandlingService().handleCondition(new ConditionPatternSubexpressionMax(max), evalFollowedByNode.getContext().getEpStatementHandle());
return;
}
}
countActivePerChild[index]++;
EvalNode child = getFactoryNode().getChildNodes().get(index + 1);
EvalStateNode childState = child.newState(this, matchEvent, evalFollowedByNode.getContext(), null);
nodes.put(childState, index + 1);
childState.start();
}
}
public final void evaluateFalse(EvalStateNode fromNode)
{
fromNode.quit();
Integer index = nodes.remove(fromNode);
if (index != null && index > 0) {
countActivePerChild[index - 1]--;
}
if (nodes.isEmpty())
{
this.getParentEvaluator().evaluateFalse(this);
quit();
}
}
public boolean isNotOperator() {
return false;
}
public boolean isFilterStateNode() {
return false;
}
public boolean isFilterChildNonQuitting() {
return false;
}
public final void quit()
{
for (EvalStateNode child : nodes.keySet())
{
child.quit();
}
}
public final Object accept(EvalStateNodeVisitor visitor, Object data)
{
return visitor.visit(this, data);
}
public final Object childrenAccept(EvalStateNodeVisitor visitor, Object data)
{
for (EvalStateNode node : nodes.keySet())
{
node.accept(visitor, data);
}
return data;
}
public final String toString()
{
return "EvalFollowedByStateNode nodes=" + nodes.size();
}
private static final Log log = LogFactory.getLog(EvalFollowedByWithMaxStateNode.class);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy