com.github.kagkarlsson.shaded.cronutils.descriptor.TimeDescriptionStrategy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of db-scheduler Show documentation
Show all versions of db-scheduler Show documentation
Simple persistent scheduler for scheduled tasks, recurring or ad-hoc.
/*
* Copyright 2014 jmrozanec
* 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 com.github.kagkarlsson.shaded.cronutils.descriptor;
import com.github.kagkarlsson.shaded.cronutils.Function;
import com.github.kagkarlsson.shaded.cronutils.model.field.expression.*;
import com.github.kagkarlsson.shaded.cronutils.model.field.value.IntegerFieldValue;
import com.github.kagkarlsson.shaded.cronutils.utils.Preconditions;
import com.github.kagkarlsson.shaded.cronutils.utils.StringUtils;
import java.util.HashSet;
import java.util.ResourceBundle;
import java.util.Set;
import static com.github.kagkarlsson.shaded.cronutils.model.field.expression.FieldExpression.always;
/**
* Strategy to provide a human readable description to hh:mm:ss variations.
*/
class TimeDescriptionStrategy extends DescriptionStrategy {
private final FieldExpression hours;
private final FieldExpression minutes;
private final FieldExpression seconds;
private final Set> descriptions;
private static final int DEFAULTSECONDS = 0;
private static final String EVERY = "every";
private static final String SECOND = "second";
private static final String MINUTE = "minute";
private static final String HOUR = "hour";
private static final String EVERY_MINUTE_FORMAT = "%s %s ";
/**
* Constructor.
*
* @param bundle - locale considered when creating the description
* @param hours - CronFieldExpression for hours. If no instance is provided,
* an Always instance is created.
* @param minutes - CronFieldExpression for minutes. If no instance is provided,
* an Always instance is created.
* @param seconds - CronFieldExpression for seconds. If no instance is provided,
* an On instance is created.
*/
TimeDescriptionStrategy(final ResourceBundle bundle, final FieldExpression hours, final FieldExpression minutes,
final FieldExpression seconds) {
super(bundle);
this.hours = ensureInstance(hours, always());
this.minutes = ensureInstance(minutes, always());
this.seconds = ensureInstance(seconds, new On(new IntegerFieldValue(DEFAULTSECONDS)));
descriptions = new HashSet<>();
registerFunctions();
}
/**
* Give an expression instance, will return it if is not null. Otherwise will
* return the defaultExpression;
*
* @param expression - CronFieldExpression instance; may be null
* @param defaultExpression - CronFieldExpression, never null;
* @return the given expression or the given defaultExpression in case the given
* expression is {@code null}
*/
private FieldExpression ensureInstance(final FieldExpression expression, final FieldExpression defaultExpression) {
Preconditions.checkNotNull(defaultExpression, "Default expression must not be null");
if (expression != null) {
return expression;
} else {
return defaultExpression;
}
}
@Override
public String describe() {
final TimeFields fields = new TimeFields(hours, minutes, seconds);
for (final Function function : descriptions) {
if (!"".equals(function.apply(fields))) {
return function.apply(fields);
}
}
String secondsDesc = "";
String minutesDesc = "";
String hoursDesc = "";
if (!(hours instanceof Always)) {
hoursDesc = addTimeExpressions(describe(hours), bundle.getString(HOUR), bundle.getString("hours"));
}
if (!(minutes instanceof On && isDefault((On) minutes)) && !((minutes instanceof Always) && (hours instanceof Always))) {
minutesDesc = addTimeExpressions(describe(minutes), bundle.getString(MINUTE), bundle.getString("minutes"));
}
if (!(seconds instanceof On && isDefault((On) seconds))) {
secondsDesc = addTimeExpressions(describe(seconds), bundle.getString(SECOND), bundle.getString("seconds"));
}
return String.format("%s %s %s", secondsDesc, minutesDesc, hoursDesc);
}
protected String describe(final Always always, final boolean and) {
return describe(new Every(new IntegerFieldValue(1)), and);
}
private String addTimeExpressions(final String description, final String singular, final String plural) {
return description.replaceAll("%s", singular).replaceAll("replace_plural", plural);
}
/**
* Registers functions that map TimeFields to a human readable description.
*/
private void registerFunctions() {
// case: every second
// case: every minute at x second
descriptions.add(timeFields -> {
if (timeFields.hours instanceof Always && timeFields.minutes instanceof Always) {
if (timeFields.seconds instanceof Always) {
return String.format(EVERY_MINUTE_FORMAT, bundle.getString(EVERY), bundle.getString(SECOND));
}
if (timeFields.seconds instanceof On) {
if (TimeDescriptionStrategy.this.isDefault((On) timeFields.seconds)) {
return String.format(EVERY_MINUTE_FORMAT, bundle.getString(EVERY), bundle.getString(MINUTE));
} else {
return String.format("%s %s %s %s %02d", bundle.getString(EVERY), bundle.getString(MINUTE),
bundle.getString("at"), bundle.getString(SECOND),
((On) timeFields.seconds).getTime().getValue());
}
}
}
return StringUtils.EMPTY;
});
// case: At minute x
descriptions.add(timeFields -> {
if (timeFields.hours instanceof Always && timeFields.minutes instanceof On
&& timeFields.seconds instanceof On) {
if (TimeDescriptionStrategy.this.isDefault((On) timeFields.seconds)) {
if (TimeDescriptionStrategy.this.isDefault((On) timeFields.minutes)) {
return String.format(EVERY_MINUTE_FORMAT, bundle.getString(EVERY), bundle.getString(HOUR));
}
return String.format("%s %s %s %s %s", bundle.getString(EVERY), bundle.getString(HOUR),
bundle.getString("at"), bundle.getString(MINUTE),
((On) timeFields.minutes).getTime().getValue());
} else {
return String.format("%s %s %s %s %s %s %s %s", bundle.getString(EVERY), bundle.getString(HOUR),
bundle.getString("at"), bundle.getString(MINUTE),
((On) timeFields.minutes).getTime().getValue(), bundle.getString("and"),
bundle.getString(SECOND), ((On) timeFields.seconds).getTime().getValue());
}
}
return StringUtils.EMPTY;
});
// case: 11:45
descriptions.add(timeFields -> {
if (timeFields.hours instanceof On && timeFields.minutes instanceof On
&& timeFields.seconds instanceof Always) {
return String.format("%s %s %s %02d:%02d", bundle.getString(EVERY), bundle.getString(SECOND),
bundle.getString("at"), ((On) hours).getTime().getValue(), ((On) minutes).getTime().getValue());
}
return StringUtils.EMPTY;
});
// case: 11:30:45
// case: 11:30:00 -> 11:30
descriptions.add(timeFields -> {
if (timeFields.hours instanceof On && timeFields.minutes instanceof On
&& timeFields.seconds instanceof On) {
if (TimeDescriptionStrategy.this.isDefault((On) timeFields.seconds)) {
return String.format("%s %02d:%02d", bundle.getString("at"), ((On) hours).getTime().getValue(),
((On) minutes).getTime().getValue());
} else {
return String.format("%s %02d:%02d:%02d", bundle.getString("at"), ((On) hours).getTime().getValue(),
((On) minutes).getTime().getValue(), ((On) seconds).getTime().getValue());
}
}
return StringUtils.EMPTY;
});
// 11 -> 11:00
descriptions.add(timeFields -> {
if (timeFields.hours instanceof On && timeFields.minutes instanceof Always
&& timeFields.seconds instanceof Always) {
return String.format("%s %02d:00", bundle.getString("at"), ((On) hours).getTime().getValue());
}
return StringUtils.EMPTY;
});
// case: every minute between 11:00 and 11:10
// case: every second between 11:00 and 11:10
descriptions.add(timeFields -> {
if (timeFields.hours instanceof On && timeFields.minutes instanceof Between) {
if (timeFields.seconds instanceof On) {
return String.format("%s %s %s %02d:%02d %s %02d:%02d", bundle.getString(EVERY),
bundle.getString(MINUTE), bundle.getString("between"),
((On) timeFields.hours).getTime().getValue(),
((Between) timeFields.minutes).getFrom().getValue(), bundle.getString("and"),
((On) timeFields.hours).getTime().getValue(),
((Between) timeFields.minutes).getTo().getValue());
}
if (timeFields.seconds instanceof Always) {
return String.format("%s %s %s %02d:%02d %s %02d:%02d", bundle.getString(EVERY),
bundle.getString(SECOND), bundle.getString("between"),
((On) timeFields.hours).getTime().getValue(),
((Between) timeFields.minutes).getFrom().getValue(), bundle.getString("and"),
((On) timeFields.hours).getTime().getValue(),
((Between) timeFields.minutes).getTo().getValue());
}
}
return StringUtils.EMPTY;
});
// case: every x minutes
descriptions.add(timeFields -> {
if (timeFields.hours instanceof Always && timeFields.minutes instanceof Every
&& timeFields.seconds instanceof On) {
final Every minute = (Every) timeFields.minutes;
String desc;
if (minute.getPeriod().getValue() == 1
&& TimeDescriptionStrategy.this.isDefault((On) timeFields.seconds)) {
desc = String.format(EVERY_MINUTE_FORMAT, bundle.getString(EVERY), bundle.getString(MINUTE));
} else {
desc = String.format("%s %s %s ", bundle.getString(EVERY), minute.getPeriod().getValue(),
bundle.getString("minutes"));
}
if (minute.getExpression() instanceof Between) {
return StringUtils.EMPTY;
}
return desc;
}
return StringUtils.EMPTY;
});
// case: every x hours
descriptions.add(timeFields -> {
if (timeFields.hours instanceof Every && timeFields.minutes instanceof On
&& timeFields.seconds instanceof On) {
// every hour
if (((On) timeFields.minutes).getTime().getValue() == 0
&& ((On) timeFields.seconds).getTime().getValue() == 0) {
final Integer period = ((Every) timeFields.hours).getPeriod().getValue();
if (period == null || period == 1) {
return String.format(EVERY_MINUTE_FORMAT, bundle.getString(EVERY), bundle.getString(HOUR));
}
}
final String result = String.format("%s %s %s %s %s %s ", bundle.getString(EVERY),
((Every) hours).getPeriod().getValue(), bundle.getString("hours"), bundle.getString("at"),
bundle.getString(MINUTE), ((On) minutes).getTime().getValue());
if (TimeDescriptionStrategy.this.isDefault((On) timeFields.seconds)) {
return result;
} else {
return String.format("%s %s %s", bundle.getString("and"), bundle.getString(SECOND),
((On) seconds).getTime().getValue());
}
}
return StringUtils.EMPTY;
});
// case: every second at minute 00
descriptions.add(timeFields -> {
if (timeFields.hours instanceof Always && timeFields.seconds instanceof Always) {
if (timeFields.minutes instanceof On) {
// Every
return String.format("%s %s %s %s %02d %s", bundle.getString(EVERY), bundle.getString(SECOND),
bundle.getString("at"), bundle.getString(MINUTE),
((On) timeFields.minutes).getTime().getValue(), bundle.getString("of_every_hour"));
}
}
return StringUtils.EMPTY;
});
// case: every second at minute 00 and every x minutes
descriptions.add(timeFields -> {
if (timeFields.hours instanceof Always && timeFields.seconds instanceof Always) {
if (timeFields.minutes instanceof And) {
// Every
String minutesDesc = addTimeExpressions(describe(timeFields.minutes),
bundle.getString("minute"), bundle.getString("minutes"));
return String.format("%s %s %s", bundle.getString(EVERY), bundle.getString(SECOND), minutesDesc);
}
}
return StringUtils.EMPTY;
});
// case: every second at x, y and z hours
descriptions.add(timeFields -> {
if (timeFields.hours instanceof And && timeFields.minutes instanceof Always && timeFields.seconds instanceof Always) {
// Every
String hoursDesc = addTimeExpressions(describe(timeFields.hours),
bundle.getString(HOUR), bundle.getString("hours"));
return String.format("%s %s %s", bundle.getString(EVERY), bundle.getString(SECOND), hoursDesc);
}
return StringUtils.EMPTY;
});
}
/**
* Contains CronFieldExpression instances for hours, minutes and seconds.
*/
class TimeFields {
private final FieldExpression seconds;
private final FieldExpression minutes;
private final FieldExpression hours;
public TimeFields(final FieldExpression hours, final FieldExpression minutes, final FieldExpression seconds) {
this.hours = hours;
this.minutes = minutes;
this.seconds = seconds;
}
}
/**
* Checks if On instance has a default value.
*
* @param on - On instance
* @return boolean - true if time value matches a default; false otherwise.
*/
private boolean isDefault(final On on) {
return on.getTime().getValue() == DEFAULTSECONDS;
}
}