aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/expr/TemporalConfig.java60
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/expr/Value.java8
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseDelayedValue.java4
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseNumericValue.java14
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseValue.java20
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/BuiltinOperators.java2
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/DateTimeValue.java6
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultDateFunctions.java204
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctions.java51
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultNumberFunctions.java2
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/ExpressionTokenizer.java123
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/FunctionSupport.java1
-rw-r--r--src/main/java/com/healthmarketscience/jackcess/impl/expr/StringValue.java28
-rw-r--r--src/test/java/com/healthmarketscience/jackcess/PropertyExpressionTest.java2
-rw-r--r--src/test/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctionsTest.java30
15 files changed, 417 insertions, 138 deletions
diff --git a/src/main/java/com/healthmarketscience/jackcess/expr/TemporalConfig.java b/src/main/java/com/healthmarketscience/jackcess/expr/TemporalConfig.java
index d7061ca..b7de667 100644
--- a/src/main/java/com/healthmarketscience/jackcess/expr/TemporalConfig.java
+++ b/src/main/java/com/healthmarketscience/jackcess/expr/TemporalConfig.java
@@ -31,12 +31,14 @@ import java.util.Locale;
public class TemporalConfig
{
public static final String US_DATE_FORMAT = "M/d/yyyy";
+ public static final String US_DATE_IMPLICIT_YEAR_FORMAT = "M/d";
public static final String US_TIME_FORMAT_12 = "h:mm:ss a";
public static final String US_TIME_FORMAT_24 = "H:mm:ss";
/** default implementation which is configured for the US locale */
public static final TemporalConfig US_TEMPORAL_CONFIG = new TemporalConfig(
- US_DATE_FORMAT, US_TIME_FORMAT_12, US_TIME_FORMAT_24, '/', ':', Locale.US);
+ US_DATE_FORMAT, US_DATE_IMPLICIT_YEAR_FORMAT,
+ US_TIME_FORMAT_12, US_TIME_FORMAT_24, '/', ':', Locale.US);
public enum Type {
DATE, TIME, DATE_TIME, TIME_12, TIME_24, DATE_TIME_12, DATE_TIME_24;
@@ -74,9 +76,37 @@ public class TemporalConfig
throw new RuntimeException("invalid type " + this);
}
}
+
+ public boolean includesDate() {
+ switch(this) {
+ case DATE:
+ case DATE_TIME:
+ case DATE_TIME_12:
+ case DATE_TIME_24:
+ return true;
+ default:
+ return false;
+ }
+ }
+
+ public boolean includesTime() {
+ return (this != DATE);
+ }
+
+ public boolean isTimeOnly() {
+ switch(this) {
+ case TIME:
+ case TIME_12:
+ case TIME_24:
+ return true;
+ default:
+ return false;
+ }
+ }
}
private final String _dateFormat;
+ private final String _dateImplicitYearFormat;
private final String _timeFormat12;
private final String _timeFormat24;
private final char _dateSeparator;
@@ -92,6 +122,7 @@ public class TemporalConfig
* <time>".
*
* @param dateFormat the date (no time) format
+ * @param dateImplicitYearFormat the date (no time) with no year format
* @param timeFormat12 the 12 hour time format
* @param timeFormat24 the 24 hour time format
* @param dateSeparator the primary separator used to separate elements in
@@ -103,11 +134,12 @@ public class TemporalConfig
* string. This value should differ from the
* dateSeparator.
*/
- public TemporalConfig(String dateFormat, String timeFormat12,
- String timeFormat24, char dateSeparator,
- char timeSeparator, Locale locale)
+ public TemporalConfig(String dateFormat, String dateImplicitYearFormat,
+ String timeFormat12, String timeFormat24,
+ char dateSeparator, char timeSeparator, Locale locale)
{
_dateFormat = dateFormat;
+ _dateImplicitYearFormat = dateImplicitYearFormat;
_timeFormat12 = timeFormat12;
_timeFormat24 = timeFormat24;
_dateSeparator = dateSeparator;
@@ -178,7 +210,27 @@ public class TemporalConfig
}
}
+ public String getImplicitYearDateTimeFormat(Type type) {
+ switch(type) {
+ case DATE:
+ return _dateImplicitYearFormat;
+ case DATE_TIME:
+ return toDateTimeFormat(_dateImplicitYearFormat, getDefaultTimeFormat());
+ case DATE_TIME_12:
+ return toDateTimeFormat(_dateImplicitYearFormat, getTimeFormat12());
+ case DATE_TIME_24:
+ return toDateTimeFormat(_dateImplicitYearFormat, getTimeFormat24());
+ default:
+ throw new IllegalArgumentException(
+ "the given format does not include a date " + type);
+ }
+ }
+
public DateFormatSymbols getDateFormatSymbols() {
return _symbols;
}
+
+ private static String toDateTimeFormat(String dateFormat, String timeFormat) {
+ return dateFormat + " " + timeFormat;
+ }
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/expr/Value.java b/src/main/java/com/healthmarketscience/jackcess/expr/Value.java
index f9bbf96..e9508fa 100644
--- a/src/main/java/com/healthmarketscience/jackcess/expr/Value.java
+++ b/src/main/java/com/healthmarketscience/jackcess/expr/Value.java
@@ -98,6 +98,14 @@ public interface Value
public Date getAsDateTime(LocaleContext ctx);
/**
+ * Since date/time values have different types, it may be more convenient to
+ * get the date/time primitive value with the appropriate type information.
+ *
+ * @return this value converted to a date/time value
+ */
+ public Value getAsDateTimeValue(LocaleContext ctx);
+
+ /**
* @return this primitive value converted (rounded) to an int
*/
public Integer getAsLongInt(LocaleContext ctx);
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseDelayedValue.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseDelayedValue.java
index 6805d07..d527c69 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseDelayedValue.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseDelayedValue.java
@@ -64,6 +64,10 @@ public abstract class BaseDelayedValue implements Value
return getDelegate().getAsDateTime(ctx);
}
+ public Value getAsDateTimeValue(LocaleContext ctx) {
+ return getDelegate().getAsDateTimeValue(ctx);
+ }
+
public Integer getAsLongInt(LocaleContext ctx) {
return getDelegate().getAsLongInt(ctx);
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseNumericValue.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseNumericValue.java
index cdf686c..299cd2a 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseNumericValue.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseNumericValue.java
@@ -16,11 +16,9 @@ limitations under the License.
package com.healthmarketscience.jackcess.impl.expr;
-import java.text.SimpleDateFormat;
-import java.util.Date;
import com.healthmarketscience.jackcess.expr.LocaleContext;
-import com.healthmarketscience.jackcess.impl.ColumnImpl;
+import com.healthmarketscience.jackcess.expr.Value;
/**
*
@@ -44,9 +42,13 @@ public abstract class BaseNumericValue extends BaseValue
}
@Override
- public Date getAsDateTime(LocaleContext ctx) {
- double d = getNumber().doubleValue();
- return new Date(ColumnImpl.fromDateDouble(d, ctx.getCalendar()));
+ public Value getAsDateTimeValue(LocaleContext ctx) {
+ Value dateValue = DefaultDateFunctions.numberToDateValue(
+ ctx, getNumber().doubleValue());
+ if(dateValue == null) {
+ throw invalidConversion(Value.Type.DATE_TIME);
+ }
+ return dateValue;
}
protected abstract Number getNumber();
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseValue.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseValue.java
index d83af09..b444218 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseValue.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BaseValue.java
@@ -35,32 +35,36 @@ public abstract class BaseValue implements Value
}
public boolean getAsBoolean(LocaleContext ctx) {
- throw invalidConversion(Value.Type.LONG);
+ throw invalidConversion(Type.LONG);
}
public String getAsString(LocaleContext ctx) {
- throw invalidConversion(Value.Type.STRING);
+ throw invalidConversion(Type.STRING);
}
public Date getAsDateTime(LocaleContext ctx) {
- throw invalidConversion(Value.Type.DATE_TIME);
+ return (Date)getAsDateTimeValue(ctx).get();
+ }
+
+ public Value getAsDateTimeValue(LocaleContext ctx) {
+ throw invalidConversion(Type.DATE_TIME);
}
public Integer getAsLongInt(LocaleContext ctx) {
- throw invalidConversion(Value.Type.LONG);
+ throw invalidConversion(Type.LONG);
}
public Double getAsDouble(LocaleContext ctx) {
- throw invalidConversion(Value.Type.DOUBLE);
+ throw invalidConversion(Type.DOUBLE);
}
public BigDecimal getAsBigDecimal(LocaleContext ctx) {
- throw invalidConversion(Value.Type.BIG_DEC);
+ throw invalidConversion(Type.BIG_DEC);
}
- private EvalException invalidConversion(Value.Type newType) {
+ protected EvalException invalidConversion(Type newType) {
return new EvalException(
- getType() + " value cannot be converted to " + newType);
+ this + " cannot be converted to " + newType);
}
protected Integer roundToLongInt(LocaleContext ctx) {
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BuiltinOperators.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BuiltinOperators.java
index f53a6be..be893c3 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/BuiltinOperators.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/BuiltinOperators.java
@@ -655,7 +655,7 @@ public class BuiltinOperators
Value.Type.BIG_DEC : Value.Type.DOUBLE);
}
return prefType;
- } catch(NumberFormatException ignored) {
+ } catch(EvalException ignored) {
// not a number
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DateTimeValue.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DateTimeValue.java
index 0f6937f..e2de36d 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DateTimeValue.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DateTimeValue.java
@@ -21,6 +21,7 @@ import java.util.Date;
import com.healthmarketscience.jackcess.impl.ColumnImpl;
import com.healthmarketscience.jackcess.expr.LocaleContext;
+import com.healthmarketscience.jackcess.expr.Value;
/**
*
@@ -68,6 +69,11 @@ public class DateTimeValue extends BaseValue
}
@Override
+ public Value getAsDateTimeValue(LocaleContext ctx) {
+ return this;
+ }
+
+ @Override
public Integer getAsLongInt(LocaleContext ctx) {
return roundToLongInt(ctx);
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultDateFunctions.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultDateFunctions.java
index f5061d4..4039192 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultDateFunctions.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultDateFunctions.java
@@ -50,6 +50,18 @@ public class DefaultDateFunctions
private static final long SECONDS_PER_HOUR = 60L * 60L;
private static final long SECONDS_PER_MINUTE = 60L;
+ private static final String INTV_YEAR = "yyyy";
+ private static final String INTV_QUARTER = "q";
+ private static final String INTV_MONTH = "m";
+ private static final String INTV_DAY_OF_YEAR = "y";
+ private static final String INTV_DAY = "d";
+ private static final String INTV_WEEKDAY = "w";
+ private static final String INTV_WEEK = "ww";
+ private static final String INTV_HOUR = "h";
+ private static final String INTV_MINUTE = "n";
+ private static final String INTV_SECOND = "s";
+
+
private DefaultDateFunctions() {}
static void init() {
@@ -67,7 +79,7 @@ public class DefaultDateFunctions
public static final Function DATEVALUE = registerFunc(new Func1NullIsNull("DateValue") {
@Override
protected Value eval1(EvalContext ctx, Value param1) {
- Value dv = nonNullToDateValue(ctx, param1);
+ Value dv = param1.getAsDateTimeValue(ctx);
if(dv.getType() == Value.Type.DATE) {
return dv;
}
@@ -100,6 +112,51 @@ public class DefaultDateFunctions
}
});
+ public static final Function DATEPART = registerFunc(new FuncVar("DatePart", 2, 4) {
+ @Override
+ protected Value evalVar(EvalContext ctx, Value[] params) {
+ Value param2 = params[1];
+ if(param2.isNull()) {
+ return ValueSupport.NULL_VAL;
+ }
+
+ int firstDay = getFirstDayParam(ctx, params, 2);
+ int firstWeekType = getFirstWeekTypeParam(ctx, params, 3);
+
+ String intv = params[0].getAsString(ctx).trim();
+ int result = -1;
+ if(intv.equalsIgnoreCase(INTV_YEAR)) {
+ result = nonNullToCalendarField(ctx, param2, Calendar.YEAR);
+ } else if(intv.equalsIgnoreCase(INTV_QUARTER)) {
+ // month in is 0 based
+ int month = nonNullToCalendarField(ctx, param2, Calendar.MONTH);
+ result = (month / 3) + 1;
+ } else if(intv.equalsIgnoreCase(INTV_MONTH)) {
+ // convert from 0 based to 1 based value
+ result = nonNullToCalendarField(ctx, param2, Calendar.MONTH) + 1;
+ } else if(intv.equalsIgnoreCase(INTV_DAY_OF_YEAR)) {
+ result = nonNullToCalendarField(ctx, param2, Calendar.DAY_OF_YEAR);
+ } else if(intv.equalsIgnoreCase(INTV_DAY)) {
+ result = nonNullToCalendarField(ctx, param2, Calendar.DAY_OF_MONTH);
+ } else if(intv.equalsIgnoreCase(INTV_WEEKDAY)) {
+ int dayOfWeek = nonNullToCalendarField(ctx, param2, Calendar.DAY_OF_WEEK);
+ result = dayOfWeekToWeekDay(dayOfWeek, firstDay);
+ } else if(intv.equalsIgnoreCase(INTV_WEEK)) {
+ result = weekOfYear(ctx, param2, firstDay, firstWeekType);
+ } else if(intv.equalsIgnoreCase(INTV_HOUR)) {
+ result = nonNullToCalendarField(ctx, param2, Calendar.HOUR_OF_DAY);
+ } else if(intv.equalsIgnoreCase(INTV_MINUTE)) {
+ result = nonNullToCalendarField(ctx, param2, Calendar.MINUTE);
+ } else if(intv.equalsIgnoreCase(INTV_SECOND)) {
+ result = nonNullToCalendarField(ctx, param2, Calendar.SECOND);
+ } else {
+ throw new EvalException("Invalid interval " + intv);
+ }
+
+ return ValueSupport.toValue(result);
+ }
+ });
+
public static final Function NOW = registerFunc(new Func0("Now") {
@Override
protected Value eval0(EvalContext ctx) {
@@ -118,7 +175,7 @@ public class DefaultDateFunctions
public static final Function TIMEVALUE = registerFunc(new Func1NullIsNull("TimeValue") {
@Override
protected Value eval1(EvalContext ctx, Value param1) {
- Value dv = nonNullToDateValue(ctx, param1);
+ Value dv = param1.getAsDateTimeValue(ctx);
if(dv.getType() == Value.Type.TIME) {
return dv;
}
@@ -202,8 +259,8 @@ public class DefaultDateFunctions
@Override
protected Value evalVar(EvalContext ctx, Value[] params) {
Value param1 = params[0];
- if(param1 == null) {
- return null;
+ if(param1.isNull()) {
+ return ValueSupport.NULL_VAL;
}
// convert from 1 based to 0 based value
int month = param1.getAsLongInt(ctx) - 1;
@@ -230,8 +287,8 @@ public class DefaultDateFunctions
@Override
protected Value evalVar(EvalContext ctx, Value[] params) {
Value param1 = params[0];
- if(param1 == null) {
- return null;
+ if(param1.isNull()) {
+ return ValueSupport.NULL_VAL;
}
int dayOfWeek = nonNullToCalendarField(ctx, param1, Calendar.DAY_OF_WEEK);
@@ -245,8 +302,8 @@ public class DefaultDateFunctions
@Override
protected Value evalVar(EvalContext ctx, Value[] params) {
Value param1 = params[0];
- if(param1 == null) {
- return null;
+ if(param1.isNull()) {
+ return ValueSupport.NULL_VAL;
}
int weekday = param1.getAsLongInt(ctx);
@@ -270,59 +327,46 @@ public class DefaultDateFunctions
}
private static Calendar nonNullToCalendar(EvalContext ctx, Value param) {
- Value origParam = param;
- param = nonNullToDateValue(ctx, param);
- if(param == null) {
- // not a date/time
- throw new EvalException("Invalid date/time expression '" +
- origParam + "'");
- }
-
Calendar cal = ctx.getCalendar();
cal.setTime(param.getAsDateTime(ctx));
return cal;
}
- static Value nonNullToDateValue(EvalContext ctx, Value param) {
- Value.Type type = param.getType();
- if(type.isTemporal()) {
- return param;
- }
-
- if(type == Value.Type.STRING) {
+ static Value stringToDateValue(LocaleContext ctx, String valStr) {
+ // see if we can coerce to date/time
+ TemporalConfig.Type valTempType = ExpressionTokenizer.determineDateType(
+ valStr, ctx);
- // see if we can coerce to date/time or double
- String valStr = param.getAsString(ctx);
- TemporalConfig.Type valTempType = ExpressionTokenizer.determineDateType(
- valStr, ctx);
+ if(valTempType != null) {
- if(valTempType != null) {
+ DateFormat parseDf = ExpressionTokenizer.createParseDateTimeFormat(
+ valTempType, ctx);
- try {
- DateFormat parseDf = ExpressionTokenizer.createParseDateTimeFormat(
+ try {
+ Date dateVal = ExpressionTokenizer.parseComplete(parseDf, valStr);
+ return ValueSupport.toValue(valTempType.getValueType(), dateVal);
+ } catch(java.text.ParseException pe) {
+
+ if(valTempType.includesDate()) {
+ // the date may not include a year value, in which case it means
+ // to use the "current" year. see if this is an implicit year date
+ parseDf = ExpressionTokenizer.createParseImplicitYearDateTimeFormat(
valTempType, ctx);
- Date dateVal = ExpressionTokenizer.parseComplete(parseDf, valStr);
- return ValueSupport.toValue(valTempType.getValueType(), dateVal);
- } catch(java.text.ParseException pe) {
- // not a valid date string, not a date/time
- return null;
+ try {
+ Date dateVal = ExpressionTokenizer.parseComplete(parseDf, valStr);
+ return ValueSupport.toValue(valTempType.getValueType(), dateVal);
+ } catch(java.text.ParseException pe2) {
+ // guess not, continue on to failure
+ }
}
}
-
- // see if string can be coerced to number
- try {
- return numberToDateValue(ctx, param.getAsDouble(ctx));
- } catch(NumberFormatException ignored) {
- // not a number, not a date/time
- return null;
- }
}
- // must be a number
- return numberToDateValue(ctx, param.getAsDouble(ctx));
+ // not a valid date string, not a date/time
+ return null;
}
- private static Value numberToDateValue(EvalContext ctx, double dd) {
+ static Value numberToDateValue(LocaleContext ctx, double dd) {
if((dd < MIN_DATE) || (dd > MAX_DATE)) {
// outside valid date range
return null;
@@ -345,7 +389,7 @@ public class DefaultDateFunctions
private static double timeOnly(double dd) {
// the fractional part of the date/time double is the time value. discard
- // the integral portion and convert to seconds
+ // the integral portion
return new BigDecimal(dd).remainder(BigDecimal.ONE).doubleValue();
}
@@ -367,15 +411,67 @@ public class DefaultDateFunctions
private static int getFirstDayParam(
LocaleContext ctx, Value[] params, int idx) {
- // vbSunday (default)
- int firstDay = 1;
+ // vbSunday (default) 1
+ // vbUseSystem 0
+ return getOptionalIntParam(ctx, params, idx, 1, 0);
+ }
+
+ private static int getFirstWeekTypeParam(
+ LocaleContext ctx, Value[] params, int idx) {
+ // vbFirstJan1 (default) 1
+ // vbUseSystem 0
+ return getOptionalIntParam(ctx, params, idx, 1, 0);
+ }
+
+ private static int getOptionalIntParam(
+ LocaleContext ctx, Value[] params, int idx, int defValue, int useDefValue) {
+ int val = defValue;
if(params.length > idx) {
- firstDay = params[idx].getAsLongInt(ctx);
- if(firstDay == 0) {
- // 0 == vbUseSystem, so we will use the default "sunday"
- firstDay = 1;
+ val = params[idx].getAsLongInt(ctx);
+ if(val == useDefValue) {
+ val = defValue;
+ }
+ }
+ return val;
+ }
+
+ private static int weekOfYear(EvalContext ctx, Value param, int firstDay,
+ int firstWeekType) {
+ Calendar cal = nonNullToCalendar(ctx, param);
+
+ // need to mess with some calendar settings, but they need to be restored
+ // when done because the Calendar instance may be shared
+ int origFirstDay = cal.getFirstDayOfWeek();
+ int origMinDays = cal.getMinimalDaysInFirstWeek();
+ try {
+
+ int minDays = 1;
+ switch(firstWeekType) {
+ case 1:
+ // vbUseSystem 0
+ // vbFirstJan1 1 (default)
+ break;
+ case 2:
+ // vbFirstFourDays 2
+ minDays = 4;
+ break;
+ case 3:
+ // vbFirstFullWeek 3
+ minDays = 7;
+ break;
+ default:
+ throw new EvalException("Invalid first week of year type " +
+ firstWeekType);
}
+
+ cal.setFirstDayOfWeek(firstDay);
+ cal.setMinimalDaysInFirstWeek(minDays);
+
+ return cal.get(Calendar.WEEK_OF_YEAR);
+
+ } finally {
+ cal.setFirstDayOfWeek(origFirstDay);
+ cal.setMinimalDaysInFirstWeek(origMinDays);
}
- return firstDay;
}
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctions.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctions.java
index f794a62..331a47e 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctions.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctions.java
@@ -163,7 +163,7 @@ public class DefaultFunctions
public static final Function CDATE = registerFunc(new Func1("CDate") {
@Override
protected Value eval1(EvalContext ctx, Value param1) {
- return DefaultDateFunctions.nonNullToDateValue(ctx, param1);
+ return param1.getAsDateTimeValue(ctx);
}
});
static {
@@ -240,9 +240,21 @@ public class DefaultFunctions
public static final Function ISDATE = registerFunc(new Func1("IsDate") {
@Override
protected Value eval1(EvalContext ctx, Value param1) {
- return ValueSupport.toValue(
- !param1.isNull() &&
- (DefaultDateFunctions.nonNullToDateValue(ctx, param1) != null));
+ if(param1.getType().isTemporal()) {
+ return ValueSupport.TRUE_VAL;
+ }
+
+ // for the purposes of this method, a string literal should only
+ // return true if it is explicitly a date/time, not if it is just a
+ // number (even though casting a number string to a date/time works in
+ // general)
+ if((param1.getType() == Value.Type.STRING) &&
+ !stringIsNumeric(ctx, param1) &&
+ stringIsTemporal(ctx, param1)) {
+ return ValueSupport.TRUE_VAL;
+ }
+
+ return ValueSupport.FALSE_VAL;
}
});
@@ -253,13 +265,11 @@ public class DefaultFunctions
return ValueSupport.TRUE_VAL;
}
- if(param1.getType() == Value.Type.STRING) {
- try {
- param1.getAsBigDecimal(ctx);
- return ValueSupport.TRUE_VAL;
- } catch(NumberFormatException ignored) {
- // fall through to FALSE_VAL
- }
+ // note, only a string can be considered numberic for this function,
+ // even though a date/time can be cast to a number in general
+ if((param1.getType() == Value.Type.STRING) &&
+ stringIsNumeric(ctx, param1)) {
+ return ValueSupport.TRUE_VAL;
}
return ValueSupport.FALSE_VAL;
@@ -338,7 +348,26 @@ public class DefaultFunctions
}
});
+ private static boolean stringIsNumeric(EvalContext ctx, Value param) {
+ try {
+ param.getAsBigDecimal(ctx);
+ return true;
+ } catch(EvalException ignored) {
+ // fall through to false
+ }
+ return false;
+ }
+ private static boolean stringIsTemporal(EvalContext ctx, Value param) {
+ try {
+ // see if we can coerce to date/time
+ param.getAsDateTimeValue(ctx);
+ return true;
+ } catch(EvalException ignored) {
+ // not a date/time
+ }
+ return false;
+ }
// https://www.techonthenet.com/access/functions/
// https://support.office.com/en-us/article/Access-Functions-by-category-b8b136c3-2716-4d39-94a2-658ce330ed83
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultNumberFunctions.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultNumberFunctions.java
index 4473af4..2e8a04e 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultNumberFunctions.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/DefaultNumberFunctions.java
@@ -130,7 +130,7 @@ public class DefaultNumberFunctions
protected Value evalVar(EvalContext ctx, Value[] params) {
Value param1 = params[0];
if(param1.isNull()) {
- return null;
+ return ValueSupport.NULL_VAL;
}
if(param1.getType().isIntegral()) {
return param1;
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/ExpressionTokenizer.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/ExpressionTokenizer.java
index 3dfa361..c94a609 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/ExpressionTokenizer.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/ExpressionTokenizer.java
@@ -54,8 +54,10 @@ class ExpressionTokenizer
private static final char EQUALS_CHAR = '=';
// access times are based on this date (not the UTC base)
- private static final String BASE_DATE = "12/30/1899";
- private static final String BASE_DATE_FMT = "M/d/yyyy";
+ private static final String BASE_DATE_PREFIX = "1899/12/30 ";
+ private static final String BASE_DATE_FMT_PREFIX = "yyyy/M/d ";
+
+ private static final String IMPLICIT_YEAR_FMT_PREFIX = "yyyy ";
private static final byte IS_OP_FLAG = 0x01;
private static final byte IS_COMP_FLAG = 0x02;
@@ -350,47 +352,18 @@ class ExpressionTokenizer
static DateFormat createParseDateTimeFormat(TemporalConfig.Type type,
LocaleContext ctx)
{
- switch(type) {
- case TIME:
- return createParseTimeFormat(TemporalConfig.Type.DATE_TIME, ctx);
- case TIME_12:
- return createParseTimeFormat(TemporalConfig.Type.DATE_TIME_12, ctx);
- case TIME_24:
- return createParseTimeFormat(TemporalConfig.Type.DATE_TIME_24, ctx);
- default:
- // use normal formatter
+ if(type.isTimeOnly()) {
+ return new ParseTimeFormat(type, ctx);
}
TemporalConfig cfg = ctx.getTemporalConfig();
return ctx.createDateFormat(cfg.getDateTimeFormat(type));
}
- private static DateFormat createParseTimeFormat(TemporalConfig.Type parseType,
- LocaleContext ctx)
+ static DateFormat createParseImplicitYearDateTimeFormat(
+ TemporalConfig.Type type, LocaleContext ctx)
{
- TemporalConfig cfg = ctx.getTemporalConfig();
- // we need to use a special DateFormat impl which manipulates the parsed
- // time-only value so it becomes the right Date value
- String baseDate = getBaseDatePrefix(ctx);
- DateFormat parseDf = ctx.createDateFormat(
- cfg.getDateTimeFormat(parseType));
- return new ParseTimeFormat(parseDf, baseDate);
- }
-
- private static String getBaseDatePrefix(LocaleContext ctx) {
- String dateFmt = ctx.getTemporalConfig().getDateFormat();
- String baseDate = BASE_DATE;
- if(!BASE_DATE_FMT.equals(dateFmt)) {
- try {
- // need to reformat the base date to the relevant date format
- DateFormat parseDf = ctx.createDateFormat(BASE_DATE_FMT);
- DateFormat df = ctx.createDateFormat(dateFmt);
- baseDate = df.format(parseComplete(parseDf, baseDate));
- } catch(Exception e) {
- throw new ParseException("Could not parse base date", e);
- }
- }
- return baseDate + " ";
+ return new ParseImplicitYearFormat(type, ctx);
}
private static Token maybeParseNumberLiteral(char firstChar, ExprBuf buf) {
@@ -630,32 +603,37 @@ class ExpressionTokenizer
}
/**
- * Special date/time format which will parse time-only strings "correctly"
- * according to how access handles time-only values.
+ * Base DateFormat implementation for parsing date/time formats where
+ * additional information is added on to the format in order for it to be
+ * parsed correctly.
*/
- private static final class ParseTimeFormat extends DateFormat
+ private static abstract class ParsePrefixFormat extends DateFormat
{
private static final long serialVersionUID = 0L;
private final DateFormat _parseDelegate;
- private final String _baseDate;
- private ParseTimeFormat(DateFormat parseDelegate, String baseDate)
- {
- _parseDelegate = parseDelegate;
- _baseDate = baseDate;
+ private ParsePrefixFormat(String formatPrefix, String formatStr,
+ LocaleContext ctx) {
+ _parseDelegate = ctx.createDateFormat(formatPrefix + formatStr);
}
@Override
- public StringBuffer format(Date date, StringBuffer toAppendTo, FieldPosition fieldPosition) {
+ public StringBuffer format(Date date, StringBuffer toAppendTo,
+ FieldPosition fieldPosition) {
throw new UnsupportedOperationException();
}
@Override
public Date parse(String source, ParsePosition pos) {
- // we parse as a full date/time in order to get the correct "base date"
- // used by access
- return _parseDelegate.parse(_baseDate + source, pos);
+ String prefix = getPrefix();
+
+ Date result = _parseDelegate.parse(prefix + source, pos);
+
+ // adjust index for original string
+ pos.setIndex(pos.getIndex() - prefix.length());
+
+ return result;
}
@Override
@@ -667,6 +645,55 @@ class ExpressionTokenizer
public TimeZone getTimeZone() {
return _parseDelegate.getTimeZone();
}
+
+ protected abstract String getPrefix();
+ }
+
+ /**
+ * Special date/time format which will parse time-only strings "correctly"
+ * according to how access handles time-only values.
+ */
+ private static final class ParseTimeFormat extends ParsePrefixFormat
+ {
+ private static final long serialVersionUID = 0L;
+
+ private ParseTimeFormat(TemporalConfig.Type timeType, LocaleContext ctx) {
+ super(BASE_DATE_FMT_PREFIX,
+ ctx.getTemporalConfig().getDateTimeFormat(timeType), ctx);
+ }
+
+ @Override
+ protected String getPrefix() {
+ // we parse as a full date/time in order to get the correct "base date"
+ // used by access
+ return BASE_DATE_PREFIX;
+ }
+ }
+
+ /**
+ * Special date/time format which will parse dates with implicit (current)
+ * years.
+ */
+ private static final class ParseImplicitYearFormat extends ParsePrefixFormat
+ {
+ private static final long serialVersionUID = 0L;
+
+ private ParseImplicitYearFormat(TemporalConfig.Type type,
+ LocaleContext ctx) {
+ super(IMPLICIT_YEAR_FMT_PREFIX,
+ ctx.getTemporalConfig().getImplicitYearDateTimeFormat(type),
+ ctx);
+ }
+
+ @Override
+ protected String getPrefix() {
+ // need to get the current year
+ Calendar cal = getCalendar();
+ cal.setTimeInMillis(System.currentTimeMillis());
+ int year = cal.get(Calendar.YEAR);
+ // return a value matching IMPLICIT_YEAR_FMT_PREFIX
+ return year + " ";
+ }
}
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/FunctionSupport.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/FunctionSupport.java
index 91d98a1..79e4609 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/FunctionSupport.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/FunctionSupport.java
@@ -254,5 +254,4 @@ public class FunctionSupport
}
return defValue;
}
-
}
diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/expr/StringValue.java b/src/main/java/com/healthmarketscience/jackcess/impl/expr/StringValue.java
index 9988477..12a5efb 100644
--- a/src/main/java/com/healthmarketscience/jackcess/impl/expr/StringValue.java
+++ b/src/main/java/com/healthmarketscience/jackcess/impl/expr/StringValue.java
@@ -21,7 +21,9 @@ import java.math.BigInteger;
import java.text.DecimalFormatSymbols;
import java.util.regex.Pattern;
+import com.healthmarketscience.jackcess.expr.EvalException;
import com.healthmarketscience.jackcess.expr.LocaleContext;
+import com.healthmarketscience.jackcess.expr.Value;
import org.apache.commons.lang.StringUtils;
/**
@@ -81,6 +83,30 @@ public class StringValue extends BaseValue
return getNumber(ctx);
}
+ @Override
+ public Value getAsDateTimeValue(LocaleContext ctx) {
+ Value dateValue = DefaultDateFunctions.stringToDateValue(ctx, _val);
+
+ if(dateValue == null) {
+ // see if string can be coerced to number and then to value date (note,
+ // numberToDateValue may return null for out of range numbers)
+ try {
+ dateValue = DefaultDateFunctions.numberToDateValue(
+ ctx, getNumber(ctx).doubleValue());
+ } catch(EvalException ignored) {
+ // not a number, not a date/time
+ }
+
+ if(dateValue == null) {
+ throw invalidConversion(Type.DATE_TIME);
+ }
+ }
+
+ // TODO, for now, we can't cache the date value becuase it could be an
+ // "implicit" date which would need to be re-calculated on each call
+ return dateValue;
+ }
+
protected BigDecimal getNumber(LocaleContext ctx) {
if(_num instanceof BigDecimal) {
return (BigDecimal)_num;
@@ -114,7 +140,7 @@ public class StringValue extends BaseValue
}
_num = NOT_A_NUMBER;
}
- throw new NumberFormatException("Invalid number '" + _val + "'");
+ throw invalidConversion(Type.DOUBLE);
}
private BigDecimal parseIntegerString(String tmpVal, int radix) {
diff --git a/src/test/java/com/healthmarketscience/jackcess/PropertyExpressionTest.java b/src/test/java/com/healthmarketscience/jackcess/PropertyExpressionTest.java
index d9f4932..ffb2dd3 100644
--- a/src/test/java/com/healthmarketscience/jackcess/PropertyExpressionTest.java
+++ b/src/test/java/com/healthmarketscience/jackcess/PropertyExpressionTest.java
@@ -294,7 +294,7 @@ public class PropertyExpressionTest extends TestCase
public static void testCustomEvalConfig() throws Exception
{
- TemporalConfig tempConf = new TemporalConfig("yyyy/M/d",
+ TemporalConfig tempConf = new TemporalConfig("yyyy/M/d", "M/d",
"hh.mm.ss a",
"HH.mm.ss", '/', '.',
Locale.US);
diff --git a/src/test/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctionsTest.java b/src/test/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctionsTest.java
index 575da31..c3f5f68 100644
--- a/src/test/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctionsTest.java
+++ b/src/test/java/com/healthmarketscience/jackcess/impl/expr/DefaultFunctionsTest.java
@@ -17,6 +17,8 @@ limitations under the License.
package com.healthmarketscience.jackcess.impl.expr;
import java.math.BigDecimal;
+import java.util.Calendar;
+import java.util.Date;
import com.healthmarketscience.jackcess.expr.EvalException;
import junit.framework.TestCase;
@@ -34,8 +36,6 @@ public class DefaultFunctionsTest extends TestCase
super(name);
}
- // FIXME, test more number/string functions
-
public void testFuncs() throws Exception
{
assertEquals("foo", eval("=IIf(10 > 1, \"foo\", \"bar\")"));
@@ -78,15 +78,21 @@ public class DefaultFunctionsTest extends TestCase
eval("=CSng(\"57.12345\")"));
assertEquals("9786", eval("=CStr(9786)"));
assertEquals("-42", eval("=CStr(-42)"));
+ assertEquals(new Date(1041483600000L), eval("=CDate('01/02/2003')"));
+ assertEquals(new Date(1041508800000L), eval("=CDate('01/02/2003 7:00:00 AM')"));
+ assertEquals(new Date(-1948781520000L), eval("=CDate(3013.45)"));
+
assertEquals(-1, eval("=IsNull(Null)"));
assertEquals(0, eval("=IsNull(13)"));
assertEquals(-1, eval("=IsDate(#01/02/2003#)"));
assertEquals(0, eval("=IsDate('foo')"));
+ assertEquals(0, eval("=IsDate('200')"));
assertEquals(0, eval("=IsNumeric(Null)"));
assertEquals(0, eval("=IsNumeric('foo')"));
assertEquals(0, eval("=IsNumeric(#01/02/2003#)"));
+ assertEquals(0, eval("=IsNumeric('01/02/2003')"));
assertEquals(-1, eval("=IsNumeric(37)"));
assertEquals(-1, eval("=IsNumeric(' 37 ')"));
assertEquals(-1, eval("=IsNumeric(' -37.5e2 ')"));
@@ -245,6 +251,8 @@ public class DefaultFunctionsTest extends TestCase
assertEquals(2003, eval("=Year('01/02/2003 7:00:00 AM')"));
assertEquals(1899, eval("=Year(#7:00:00 AM#)"));
+ assertEquals(Calendar.getInstance().get(Calendar.YEAR),
+ eval("=Year('01/02 7:00:00 AM')"));
assertEquals("January", eval("=MonthName(1)"));
assertEquals("Feb", eval("=MonthName(2,True)"));
@@ -280,6 +288,24 @@ public class DefaultFunctionsTest extends TestCase
assertEquals("2/12/2010", eval("=CStr(DateSerial(10,2,12))"));
assertEquals("7/12/2013", eval("=CStr(DateSerial(2014,-5,12))"));
assertEquals("8/7/2013", eval("=CStr(DateSerial(2014,-5,38))"));
+
+ assertEquals(1, eval("=DatePart('ww',#01/03/2018#)"));
+ assertEquals(2, eval("=DatePart('ww',#01/03/2018#,4)"));
+ assertEquals(1, eval("=DatePart('ww',#01/03/2018#,5)"));
+ assertEquals(1, eval("=DatePart('ww',#01/03/2018#,4,3)"));
+ assertEquals(52, eval("=DatePart('ww',#01/03/2018#,5,3)"));
+ assertEquals(1, eval("=DatePart('ww',#01/03/2018#,4,2)"));
+ assertEquals(53, eval("=DatePart('ww',#01/03/2018#,5,2)"));
+ assertEquals(2003, eval("=DatePart('yyyy',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(4, eval("=DatePart('q',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(11, eval("=DatePart('m',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(326, eval("=DatePart('y',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(22, eval("=DatePart('d',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(7, eval("=DatePart('w',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(3, eval("=DatePart('w',#11/22/2003 5:45:13 AM#, 5)"));
+ assertEquals(5, eval("=DatePart('h',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(45, eval("=DatePart('n',#11/22/2003 5:45:13 AM#)"));
+ assertEquals(13, eval("=DatePart('s',#11/22/2003 5:45:13 AM#)"));
}
public void testFinancialFuncs() throws Exception