org.opensearch.jobscheduler.spi.schedule.CronSchedule Maven / Gradle / Ivy
/*
* Copyright OpenSearch Contributors
* SPDX-License-Identifier: Apache-2.0
*
* The OpenSearch Contributors require contributions made to
* this file be licensed under the Apache-2.0 license or a
* compatible open source license.
*/
package org.opensearch.jobscheduler.spi.schedule;
import com.cronutils.model.CronType;
import com.cronutils.model.definition.CronDefinitionBuilder;
import com.cronutils.model.time.ExecutionTime;
import com.cronutils.parser.CronParser;
import com.cronutils.utils.VisibleForTesting;
import org.opensearch.common.Strings;
import org.opensearch.common.collect.Tuple;
import org.opensearch.common.io.stream.StreamInput;
import org.opensearch.common.io.stream.StreamOutput;
import org.opensearch.core.xcontent.XContentBuilder;
import org.opensearch.common.xcontent.XContentType;
import java.io.IOException;
import java.time.Clock;
import java.time.Duration;
import java.time.Instant;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.temporal.ChronoUnit;
import java.util.Objects;
import java.util.Optional;
/**
* UnixCron {@link Schedule} implementation. Refer to https://en.wikipedia.org/wiki/Cron for cron syntax.
*/
public class CronSchedule implements Schedule {
static final String CRON_FIELD = "cron";
static final String EXPRESSION_FIELD = "expression";
static final String TIMEZONE_FIELD = "timezone";
private static CronParser cronParser = new CronParser(CronDefinitionBuilder.instanceDefinitionFor(CronType.UNIX));
private ZoneId timezone;
private String expression;
private ExecutionTime executionTime;
private Clock clock;
private Long scheduleDelay;
public CronSchedule(String expression, ZoneId timezone) {
this.expression = expression;
this.timezone = timezone;
this.executionTime = ExecutionTime.forCron(cronParser.parse(this.expression));
clock = Clock.system(timezone);
}
public CronSchedule(String expression, ZoneId timezone, long scheduleDelay) {
this(expression, timezone);
this.scheduleDelay = scheduleDelay;
}
public CronSchedule(StreamInput input) throws IOException {
timezone = input.readZoneId();
expression = input.readString();
scheduleDelay = input.readOptionalLong();
executionTime = ExecutionTime.forCron(cronParser.parse(expression));
clock = Clock.system(timezone);
}
@VisibleForTesting
void setClock(Clock clock) {
this.clock = clock;
}
@VisibleForTesting
void setExecutionTime(ExecutionTime executionTime) {
this.executionTime = executionTime;
}
public ZoneId getTimeZone() {
return this.timezone;
}
public String getCronExpression() {
return this.expression;
}
public Long getDelay() {
return this.scheduleDelay;
}
@Override
public Instant getNextExecutionTime(Instant time) {
Instant baseTime = time == null ? this.clock.instant() : time;
long delay = scheduleDelay == null ? 0 : scheduleDelay;
// The executionTime object doesn't know about the delay, so first subtract the delay from the baseTime in case
// this moves to the previous interval, then add the delay to the returned execution time to get the correct time.
// For example, say it is 10:07 AM with an hourly schedule and a delay of 15 minutes. The next execution time
// should be 10:15 AM, but executionTime.nextExecution( 10:07 AM ) would return the next execution as 11 AM.
// By subtracting the delay first, the ExecutionTime object is given the input time as 9:52 AM, it returns
// 10:00 AM, and after adding the delay, we get the correct next execution time of 10:15 AM.
ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(baseTime.minusMillis(delay), this.timezone);
ZonedDateTime nextExecutionTime = this.executionTime.nextExecution(zonedDateTime).orElse(null);
return nextExecutionTime == null ? null : nextExecutionTime.toInstant().plusMillis(delay);
}
@Override
public Duration nextTimeToExecute() {
long delay = scheduleDelay == null ? 0 : scheduleDelay;
Instant now = this.clock.instant().minusMillis(delay);
ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(now, this.timezone);
Optional timeToNextExecution = this.executionTime.timeToNextExecution(zonedDateTime);
return timeToNextExecution.orElse(null);
}
@Override
public Tuple getPeriodStartingAt(Instant startTime) {
long delay = scheduleDelay == null ? 0 : scheduleDelay;
Instant realStartTime;
if (startTime != null) {
realStartTime = startTime;
} else {
Instant now = this.clock.instant();
Optional lastExecutionTime = this.executionTime.lastExecution(
ZonedDateTime.ofInstant(now.minusMillis(delay), this.timezone)
);
if (!lastExecutionTime.isPresent()) {
return new Tuple<>(now, now);
}
realStartTime = lastExecutionTime.get().toInstant().plusMillis(delay);
}
ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(realStartTime.minusMillis(delay), this.timezone);
ZonedDateTime newEndTime = executionTime.nextExecution(zonedDateTime).orElse(null);
return new Tuple<>(realStartTime, newEndTime == null ? null : newEndTime.toInstant().plusMillis(delay));
}
@Override
public Boolean runningOnTime(Instant lastExecutionTime) {
long delay = scheduleDelay == null ? 0 : scheduleDelay;
if (lastExecutionTime == null) {
return true;
}
Instant now = this.clock.instant().minusMillis(delay);
ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(now, timezone);
Optional expectedExecutionTime = this.executionTime.lastExecution(zonedDateTime);
if (!expectedExecutionTime.isPresent()) {
return false;
}
ZonedDateTime actualExecutionTime = ZonedDateTime.ofInstant(lastExecutionTime, timezone);
return ChronoUnit.SECONDS.between(expectedExecutionTime.get().plus(delay, ChronoUnit.MILLIS), actualExecutionTime) == 0L;
}
@Override
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
return this.scheduleDelay == null ? toXContentNoDelay(builder) : toXContentWithDelay(builder);
}
private XContentBuilder toXContentNoDelay(XContentBuilder builder) throws IOException {
builder.startObject()
.startObject(CRON_FIELD)
.field(EXPRESSION_FIELD, this.expression)
.field(TIMEZONE_FIELD, this.timezone.getId())
.endObject()
.endObject();
return builder;
}
private XContentBuilder toXContentWithDelay(XContentBuilder builder) throws IOException {
builder.startObject()
.startObject(CRON_FIELD)
.field(EXPRESSION_FIELD, this.expression)
.field(TIMEZONE_FIELD, this.timezone.getId())
.field(DELAY_FIELD, this.scheduleDelay)
.endObject()
.endObject();
return builder;
}
@Override
public String toString() {
return Strings.toString(XContentType.JSON, this, false, true);
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
CronSchedule cronSchedule = (CronSchedule) o;
return timezone.equals(cronSchedule.timezone)
&& expression.equals(cronSchedule.expression)
&& Objects.equals(scheduleDelay, cronSchedule.scheduleDelay);
}
@Override
public int hashCode() {
return scheduleDelay == null ? Objects.hash(timezone, expression) : Objects.hash(timezone, expression, scheduleDelay);
}
@Override
public void writeTo(StreamOutput out) throws IOException {
out.writeZoneId(timezone);
out.writeString(expression);
out.writeOptionalLong(scheduleDelay);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy