/*
* Copyright 2010 Red Hat, Inc. and/or its affiliates.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.drools.core.time.impl;
import org.drools.core.base.mvel.MVELObjectExpression;
import org.drools.core.common.InternalWorkingMemory;
import org.drools.core.rule.ConditionalElement;
import org.drools.core.rule.Declaration;
import org.drools.core.spi.Tuple;
import org.drools.core.time.Trigger;
import org.kie.api.runtime.Calendars;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.Date;
import java.util.Map;
import static org.drools.core.time.TimeUtils.evalDateExpression;
import static org.drools.core.time.TimeUtils.evalTimeExpression;
public class ExpressionIntervalTimer extends BaseTimer
implements
Timer,
Externalizable {
private MVELObjectExpression startTime;
private MVELObjectExpression endTime;
private int repeatLimit;
private MVELObjectExpression delay;
private MVELObjectExpression period;
public ExpressionIntervalTimer() {
}
public ExpressionIntervalTimer(MVELObjectExpression startTime,
MVELObjectExpression endTime,
int repeatLimit,
MVELObjectExpression delay,
MVELObjectExpression period) {
this.startTime = startTime;
this.endTime = endTime;
this.repeatLimit = repeatLimit;
this.delay = delay;
this.period = period;
}
public void writeExternal(ObjectOutput out) throws IOException {
out.writeObject( startTime );
out.writeObject( endTime );
out.writeInt( repeatLimit );
out.writeObject( delay );
out.writeObject( period );
}
public void readExternal(ObjectInput in) throws IOException,
ClassNotFoundException {
this.startTime = (MVELObjectExpression) in.readObject();
this.endTime = (MVELObjectExpression) in.readObject();
this.repeatLimit = in.readInt();
this.delay = (MVELObjectExpression) in.readObject();
this.period = (MVELObjectExpression) in.readObject();
}
public Declaration[] getStartDeclarations() {
return this.startTime != null ? this.startTime.getMVELCompilationUnit().getPreviousDeclarations() : null;
}
public Declaration[] getEndDeclarations() {
return this.endTime != null ? this.endTime.getMVELCompilationUnit().getPreviousDeclarations() : null;
}
public Declaration[] getDelayDeclarations() {
return this.delay.getMVELCompilationUnit().getPreviousDeclarations();
}
public Declaration[] getPeriodDeclarations() {
return this.period.getMVELCompilationUnit().getPreviousDeclarations();
}
public Declaration[][] getTimerDeclarations(Map<String, Declaration> outerDeclrs) {
return new Declaration[][] { sortDeclarations(outerDeclrs, getDelayDeclarations()),
sortDeclarations(outerDeclrs, getPeriodDeclarations()),
sortDeclarations(outerDeclrs, getStartDeclarations()),
sortDeclarations(outerDeclrs, getEndDeclarations()) };
}
public Trigger createTrigger(long timestamp,
Tuple leftTuple,
DefaultJobHandle jh,
String[] calendarNames,
Calendars calendars,
Declaration[][] declrs,
InternalWorkingMemory wm) {
long timeSinceLastFire = 0;
Declaration[] delayDeclarations = declrs[0];
Declaration[] periodDeclarations = declrs[1];
Declaration[] startDeclarations = declrs[2];
Declaration[] endDeclarations = declrs[3];
Date lastFireTime = null;
Date createdTime = null;
long newDelay = 0;
if ( jh != null ) {
IntervalTrigger preTrig = (IntervalTrigger) jh.getTimerJobInstance().getTrigger();
lastFireTime = preTrig.getLastFireTime();
createdTime = preTrig.getCreatedTime();
if (lastFireTime != null) {
// it is already fired calculate the new delay using the period instead of the delay
newDelay = evalTimeExpression(this.period, leftTuple, delayDeclarations, wm) - timestamp + lastFireTime.getTime();
} else {
newDelay = evalTimeExpression(this.delay, leftTuple, delayDeclarations, wm) - timestamp + createdTime.getTime();
}
} else {
newDelay = evalTimeExpression(this.delay, leftTuple, delayDeclarations, wm);
}
if (newDelay < 0) {
newDelay = 0;
}
return new IntervalTrigger(timestamp,
evalDateExpression( this.startTime, leftTuple, startDeclarations, wm ),
evalDateExpression( this.endTime, leftTuple, startDeclarations, wm ),
this.repeatLimit,
newDelay,
period != null ? evalTimeExpression(this.period, leftTuple, periodDeclarations, wm) : 0,
calendarNames,
calendars,
createdTime,
lastFireTime);
}
public Trigger createTrigger(long timestamp,
String[] calendarNames,
Calendars calendars) {
return new IntervalTrigger( timestamp,
null, // this.startTime,
null, // this.endTime,
this.repeatLimit,
0,
0,
calendarNames,
calendars );
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + delay.hashCode();
result = prime * result + ((endTime == null) ? 0 : endTime.hashCode());
result = prime * result + period.hashCode();
result = prime * result + repeatLimit;
result = prime * result + ((startTime == null) ? 0 : startTime.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if ( this == obj ) return true;
if ( obj == null ) return false;
if ( getClass() != obj.getClass() ) return false;
ExpressionIntervalTimer other = (ExpressionIntervalTimer) obj;
if ( delay != other.delay ) return false;
if ( repeatLimit != other.repeatLimit ) return false;
if ( endTime == null ) {
if ( other.endTime != null ) return false;
} else if ( !endTime.equals( other.endTime ) ) return false;
if ( period != other.period ) return false;
if ( startTime == null ) {
if ( other.startTime != null ) return false;
} else if ( !startTime.equals( other.startTime ) ) return false;
return true;
}
@Override
public ConditionalElement clone() {
return new ExpressionIntervalTimer(startTime,
endTime,
repeatLimit,
delay,
period);
}
}