From 872fa38feba7a44832b77146ec7a1209f0dfb917 Mon Sep 17 00:00:00 2001 From: graynk Date: Fri, 7 Dec 2018 12:19:46 +0500 Subject: [PATCH 01/12] basic functionality for java.time. some tests fail, uncompliant dbs not yet supported --- .../com/j256/ormlite/db/BaseDatabaseType.java | 55 ++++++++ .../java/com/j256/ormlite/field/DataType.java | 69 +++++----- .../java/com/j256/ormlite/field/SqlType.java | 6 + .../j256/ormlite/field/types/InstantType.java | 4 + .../field/types/LocalDateTimeType.java | 4 + .../ormlite/field/types/LocalDateType.java | 123 ++++++++++++++++++ .../ormlite/field/types/LocalTimeType.java | 4 + .../field/types/OffsetDateTimeType.java | 4 + .../ormlite/field/types/OffsetTimeType.java | 4 + .../ormlite/field/types/InstantTypeTest.java | 7 + .../field/types/LocalDateTimeTypeTest.java | 7 + .../field/types/LocalDateTypeTest.java | 7 + .../field/types/LocalTimeTypeTest.java | 7 + .../field/types/OffsetDateTimeTypeTest.java | 7 + .../field/types/OffsetTimeTypeTest.java | 7 + .../j256/ormlite/h2/H2CompiledStatement.java | 10 ++ .../j256/ormlite/h2/H2DatabaseResults.java | 26 ++++ .../com/j256/ormlite/h2/H2DatabaseType.java | 5 + 18 files changed, 318 insertions(+), 38 deletions(-) create mode 100644 src/main/java/com/j256/ormlite/field/types/InstantType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/LocalDateType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/LocalTimeType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java create mode 100644 src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java diff --git a/src/main/java/com/j256/ormlite/db/BaseDatabaseType.java b/src/main/java/com/j256/ormlite/db/BaseDatabaseType.java index e394bf447..613516755 100644 --- a/src/main/java/com/j256/ormlite/db/BaseDatabaseType.java +++ b/src/main/java/com/j256/ormlite/db/BaseDatabaseType.java @@ -86,6 +86,26 @@ public void appendColumnArg(String tableName, StringBuilder sb, FieldType fieldT appendDateType(sb, fieldType, fieldWidth); break; + case LOCAL_DATE: + appendLocalDateType(sb, fieldType, fieldWidth); + break; + + case LOCAL_TIME: + appendLocalTimeType(sb, fieldType, fieldWidth); + break; + + case LOCAL_DATE_TIME: + appendLocalDateTimeType(sb, fieldType, fieldWidth); + break; + + case OFFSET_TIME: + appendOffsetTimeType(sb, fieldType, fieldWidth); + break; + + case OFFSET_DATE_TIME: + appendOffsetDateTimeType(sb, fieldType, fieldWidth); + break; + case CHAR: appendCharType(sb, fieldType, fieldWidth); break; @@ -208,6 +228,41 @@ protected void appendDateType(StringBuilder sb, FieldType fieldType, int fieldWi sb.append("TIMESTAMP"); } + /** + * Output the SQL type for a Java LocalDate. + */ + protected void appendLocalDateType(StringBuilder sb, FieldType fieldType, int fieldWidth) { + sb.append("DATE"); + } + + /** + * Output the SQL type for a Java LocalTime. + */ + protected void appendLocalTimeType(StringBuilder sb, FieldType fieldType, int fieldWidth) { + sb.append("TIME"); + } + + /** + * Output the SQL type for a Java LocalDateTime. + */ + protected void appendLocalDateTimeType(StringBuilder sb, FieldType fieldType, int fieldWidth) { + sb.append("TIMESTAMP"); + } + + /** + * Output the SQL type for a Java OffsetTime. + */ + protected void appendOffsetTimeType(StringBuilder sb, FieldType fieldType, int fieldWidth) { + sb.append("TIME WITH TIME ZONE"); + } + + /** + * Output the SQL type for a Java OffsetDateTime or Instant. + */ + protected void appendOffsetDateTimeType(StringBuilder sb, FieldType fieldType, int fieldWidth) { + sb.append("TIMESTAMP WITH TIME ZONE"); + } + /** * Output the SQL type for a Java boolean. */ diff --git a/src/main/java/com/j256/ormlite/field/DataType.java b/src/main/java/com/j256/ormlite/field/DataType.java index 48fb7900d..472478ab9 100644 --- a/src/main/java/com/j256/ormlite/field/DataType.java +++ b/src/main/java/com/j256/ormlite/field/DataType.java @@ -3,44 +3,7 @@ import java.math.BigDecimal; import java.math.BigInteger; -import com.j256.ormlite.field.types.BigDecimalNumericType; -import com.j256.ormlite.field.types.BigDecimalStringType; -import com.j256.ormlite.field.types.BigIntegerType; -import com.j256.ormlite.field.types.BooleanCharType; -import com.j256.ormlite.field.types.BooleanIntegerType; -import com.j256.ormlite.field.types.BooleanObjectType; -import com.j256.ormlite.field.types.BooleanType; -import com.j256.ormlite.field.types.ByteArrayType; -import com.j256.ormlite.field.types.ByteObjectType; -import com.j256.ormlite.field.types.ByteType; -import com.j256.ormlite.field.types.CharType; -import com.j256.ormlite.field.types.CharacterObjectType; -import com.j256.ormlite.field.types.DateLongType; -import com.j256.ormlite.field.types.DateIntegerType; -import com.j256.ormlite.field.types.DateStringType; -import com.j256.ormlite.field.types.DateTimeType; -import com.j256.ormlite.field.types.DateType; -import com.j256.ormlite.field.types.DoubleObjectType; -import com.j256.ormlite.field.types.DoubleType; -import com.j256.ormlite.field.types.EnumIntegerType; -import com.j256.ormlite.field.types.EnumStringType; -import com.j256.ormlite.field.types.EnumToStringType; -import com.j256.ormlite.field.types.FloatObjectType; -import com.j256.ormlite.field.types.FloatType; -import com.j256.ormlite.field.types.IntType; -import com.j256.ormlite.field.types.IntegerObjectType; -import com.j256.ormlite.field.types.LongObjectType; -import com.j256.ormlite.field.types.LongStringType; -import com.j256.ormlite.field.types.LongType; -import com.j256.ormlite.field.types.NativeUuidType; -import com.j256.ormlite.field.types.SerializableType; -import com.j256.ormlite.field.types.ShortObjectType; -import com.j256.ormlite.field.types.ShortType; -import com.j256.ormlite.field.types.SqlDateType; -import com.j256.ormlite.field.types.StringBytesType; -import com.j256.ormlite.field.types.StringType; -import com.j256.ormlite.field.types.TimeStampType; -import com.j256.ormlite.field.types.UuidType; +import com.j256.ormlite.field.types.*; /** * Data type enumeration that corresponds to a {@link DataPersister}. @@ -249,6 +212,36 @@ public enum DataType { * Marker for fields that are unknown. */ UNKNOWN(null), + /** + * Persists the {@link java.time.LocalDate} Java class. + * + */ + LOCAL_DATE(LocalDateType.getSingleton()), + /** + * Persists the {@link java.time.LocalTime} Java class. + * + */ + LOCAL_TIME(LocalTimeType.getSingleton()), + /** + * Persists the {@link java.time.LocalDateTime} Java class. + * + */ + LOCAL_DATE_TIME(LocalDateTimeType.getSingleton()), + /** + * Persists the {@link java.time.OffsetTime} Java class. + * + */ + OFFSET_TIME(OffsetTimeType.getSingleton()), + /** + * Persists the {@link java.time.OffsetDateTime} Java class. + * + */ + OFFSET_DATE_TIME(OffsetDateTimeType.getSingleton()), + /** + * Persists the {@link java.time.Instant} Java class. + * + */ + INSTANT(InstantType.getSingleton()), // end ; diff --git a/src/main/java/com/j256/ormlite/field/SqlType.java b/src/main/java/com/j256/ormlite/field/SqlType.java index 0cddcfcd9..bb9bff161 100644 --- a/src/main/java/com/j256/ormlite/field/SqlType.java +++ b/src/main/java/com/j256/ormlite/field/SqlType.java @@ -32,6 +32,12 @@ public enum SqlType { // for other types handled by custom persisters OTHER, UNKNOWN, + // java.time + LOCAL_DATE, + LOCAL_TIME, + LOCAL_DATE_TIME, + OFFSET_TIME, + OFFSET_DATE_TIME, // end ; } diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java new file mode 100644 index 000000000..e552c1601 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -0,0 +1,4 @@ +package com.j256.ormlite.field.types; + +public class InstantType { +} diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java new file mode 100644 index 000000000..b4084a93c --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -0,0 +1,4 @@ +package com.j256.ormlite.field.types; + +public class LocalDateTimeType { +} diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java new file mode 100644 index 000000000..9d10c39fe --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java @@ -0,0 +1,123 @@ +package com.j256.ormlite.field.types; + +import java.sql.SQLException; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDateTime class in the database as Date object. + * + * @author graynk + */ +public class LocalDateTimeType extends BaseDataType { + + private static final LocalDateTimeType singleton = new LocalDateTimeType(); + public static LocalDateTimeType getSingleton() { return singleton; } + private LocalDateTimeType() { super(SqlType.LONG); } + + protected LocalDateTimeType(SqlType sqlType, Class[] classes) { + super(sqlType, classes); + } + + @Override + public Object javaToSqlArg(FieldType fieldType, Object javaObject) throws SQLException { + return extractMillis(javaObject); + } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return Long.parseLong(defaultStr); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default DateTime value: " + + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getLong(columnPos); + } + + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) throws SQLException { + return createInstance((Long) sqlArg); + } + + @Override + public boolean isEscapedValue() { + return false; + } + + @Override + public boolean isAppropriateId() { + return false; + } + + @Override + public Class getPrimaryClass() { + try { + return getDateTimeClass(); + } catch (ClassNotFoundException e) { + // ignore the exception + return null; + } + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) throws SQLException { + long newVal = System.currentTimeMillis(); + if (currentValue == null) { + return createInstance(newVal); + } + Long currentVal = extractMillis(currentValue); + if (newVal == currentVal) { + return createInstance(newVal + 1L); + } else { + return createInstance(newVal); + } + } + + private Object createInstance(Long sqlArg) throws SQLException { + try { + if (millisConstructor == null) { + Class clazz = getDateTimeClass(); + millisConstructor = clazz.getConstructor(long.class); + } + return millisConstructor.newInstance(sqlArg); + } catch (Exception e) { + throw SqlExceptionUtil.create("Could not use reflection to construct a Joda DateTime", e); + } + } + + private Long extractMillis(Object javaObject) throws SQLException { + try { + if (getMillisMethod == null) { + Class clazz = getDateTimeClass(); + getMillisMethod = clazz.getMethod("getMillis"); + } + if (javaObject == null) { + return null; + } else { + return (Long) getMillisMethod.invoke(javaObject); + } + } catch (Exception e) { + throw SqlExceptionUtil.create("Could not use reflection to get millis from Joda DateTime: " + javaObject, e); + } + } + + private Class getDateTimeClass() throws ClassNotFoundException { + if (dateTimeClass == null) { + dateTimeClass = Class.forName("org.joda.time.DateTime"); + } + return dateTimeClass; + } +} diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java new file mode 100644 index 000000000..ff6761f77 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -0,0 +1,4 @@ +package com.j256.ormlite.field.types; + +public class LocalTimeType { +} diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java new file mode 100644 index 000000000..70c9f100e --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -0,0 +1,4 @@ +package com.j256.ormlite.field.types; + +public class OffsetDateTimeType { +} diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java new file mode 100644 index 000000000..7f078bc03 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -0,0 +1,4 @@ +package com.j256.ormlite.field.types; + +public class OffsetTimeType { +} diff --git a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java new file mode 100644 index 000000000..25acc4523 --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java @@ -0,0 +1,7 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.*; + +public class InstantTypeTest { + +} \ No newline at end of file diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java new file mode 100644 index 000000000..75bfdfebd --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java @@ -0,0 +1,7 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.*; + +public class LocalDateTimeTypeTest { + +} \ No newline at end of file diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java new file mode 100644 index 000000000..409c55c54 --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java @@ -0,0 +1,7 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.*; + +public class LocalDateTypeTest { + +} \ No newline at end of file diff --git a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java new file mode 100644 index 000000000..3c34f22c2 --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java @@ -0,0 +1,7 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.*; + +public class LocalTimeTypeTest { + +} \ No newline at end of file diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java new file mode 100644 index 000000000..ef7d5147d --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java @@ -0,0 +1,7 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.*; + +public class OffsetDateTimeTypeTest { + +} \ No newline at end of file diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java new file mode 100644 index 000000000..1ea3e8d46 --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java @@ -0,0 +1,7 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.*; + +public class OffsetTimeTypeTest { + +} \ No newline at end of file diff --git a/src/test/java/com/j256/ormlite/h2/H2CompiledStatement.java b/src/test/java/com/j256/ormlite/h2/H2CompiledStatement.java index b4483dc34..d42852d5c 100644 --- a/src/test/java/com/j256/ormlite/h2/H2CompiledStatement.java +++ b/src/test/java/com/j256/ormlite/h2/H2CompiledStatement.java @@ -98,6 +98,16 @@ public static int sqlTypeToJdbcInt(SqlType sqlType) { return Types.LONGVARCHAR; case DATE: return Types.TIMESTAMP; + case LOCAL_DATE: + return Types.DATE; + case LOCAL_TIME: + return Types.TIME; + case LOCAL_DATE_TIME: + return Types.TIMESTAMP; + case OFFSET_TIME: + return Types.TIME_WITH_TIMEZONE; + case OFFSET_DATE_TIME: + return Types.TIMESTAMP_WITH_TIMEZONE; case BOOLEAN: return Types.BOOLEAN; case CHAR: diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java index 9c2d30b6c..08005ceb9 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java @@ -8,6 +8,7 @@ import java.sql.ResultSetMetaData; import java.sql.SQLException; import java.sql.Timestamp; +import java.time.*; import com.j256.ormlite.dao.ObjectCache; import com.j256.ormlite.misc.IOUtils; @@ -149,6 +150,31 @@ public Timestamp getTimestamp(int columnIndex) throws SQLException { return resultSet.getTimestamp(columnIndex + 1); } + @Override + public LocalDate getLocalDate(int columnIndex) throws SQLException { + return resultSet.getObject(columnIndex + 1, LocalDate.class); + } + + @Override + public LocalTime getLocalTime(int columnIndex) throws SQLException { + return resultSet.getObject(columnIndex + 1, LocalTime.class); + } + + @Override + public LocalDateTime getLocalDateTime(int columnIndex) throws SQLException { + return resultSet.getObject(columnIndex + 1, LocalDateTime.class); + } + + @Override + public OffsetTime getOffsetTime(int columnIndex) throws SQLException { + return resultSet.getObject(columnIndex + 1, OffsetTime.class); + } + + @Override + public OffsetDateTime getOffsetDateTime(int columnIndex) throws SQLException { + return resultSet.getObject(columnIndex + 1, OffsetDateTime.class); + } + @Override public InputStream getBlobStream(int columnIndex) throws SQLException { Blob blob = resultSet.getBlob(columnIndex + 1); diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java index 993adf3af..2d383f7b0 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java @@ -54,6 +54,11 @@ public void appendOffsetValue(StringBuilder sb, long offset) { throw new IllegalStateException("Offset is part of the LIMIT in database type " + getClass()); } + @Override + public void appendOffsetTimeType(StringBuilder sb, FieldType fieldType, int fieldWidth) { + sb.append("TIMESTAMP WITH TIME ZONE"); + } + @Override protected void configureGeneratedId(String tableName, StringBuilder sb, FieldType fieldType, List statementsBefore, List statementsAfter, List additionalArgs, From f97cdbca415f8769dcb157d5e5e6e555a99c7a72 Mon Sep 17 00:00:00 2001 From: graynk Date: Fri, 7 Dec 2018 12:38:17 +0500 Subject: [PATCH 02/12] forgot this --- .../j256/ormlite/support/DatabaseResults.java | 26 +++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/src/main/java/com/j256/ormlite/support/DatabaseResults.java b/src/main/java/com/j256/ormlite/support/DatabaseResults.java index 4f13cbe05..2da52d58c 100644 --- a/src/main/java/com/j256/ormlite/support/DatabaseResults.java +++ b/src/main/java/com/j256/ormlite/support/DatabaseResults.java @@ -5,6 +5,7 @@ import java.math.BigDecimal; import java.sql.SQLException; import java.sql.Timestamp; +import java.time.*; import com.j256.ormlite.dao.ObjectCache; @@ -141,6 +142,31 @@ public interface DatabaseResults extends Closeable { */ public Timestamp getTimestamp(int columnIndex) throws SQLException; + /** + * Returns the LocalDate value from the results at the column index + */ + public LocalDate getLocalDate(int columnIndex) throws SQLException; + + /** + * Returns the LocalTime value from the results at the column index. + */ + public LocalTime getLocalTime(int columnIndex) throws SQLException; + + /** + * Returns the LocalDateTime value from the results at the column index. + */ + public LocalDateTime getLocalDateTime(int columnIndex) throws SQLException; + + /** + * Returns the OffsetTime value from the results at the column index. + */ + public OffsetTime getOffsetTime(int columnIndex) throws SQLException; + + /** + * Returns the OffsetDateTime value from the results at the column index. + */ + public OffsetDateTime getOffsetDateTime(int columnIndex) throws SQLException; + /** * Returns an input stream for a blob value from the results at the column index. */ From 06dce09191b62fb9428d1c0e6e3dcb2e7c8c6811 Mon Sep 17 00:00:00 2001 From: graynk Date: Tue, 11 Dec 2018 10:35:25 +0500 Subject: [PATCH 03/12] Pushed a lot of unwanted code in the last commit. Added new base sql types for local_x types --- .../j256/ormlite/field/types/InstantType.java | 78 ++++++++++++- .../ormlite/field/types/LocalDateSqlType.java | 80 +++++++++++++ .../field/types/LocalDateTimeSqlType.java | 80 +++++++++++++ .../field/types/LocalDateTimeType.java | 65 ++++++++++- .../ormlite/field/types/LocalDateType.java | 110 +++++------------- .../ormlite/field/types/LocalTimeSqlType.java | 80 +++++++++++++ .../ormlite/field/types/LocalTimeType.java | 64 +++++++++- .../field/types/OffsetDateTimeType.java | 64 +++++++++- .../ormlite/field/types/OffsetTimeType.java | 64 +++++++++- .../j256/ormlite/support/DatabaseResults.java | 12 ++ .../com/j256/ormlite/dao/BaseDaoImplTest.java | 10 ++ .../ormlite/field/types/InstantTypeTest.java | 76 +++++++++++- .../field/types/LocalDateTimeTypeTest.java | 76 +++++++++++- .../field/types/LocalDateTypeTest.java | 75 +++++++++++- .../field/types/LocalTimeTypeTest.java | 75 +++++++++++- .../field/types/OffsetDateTimeTypeTest.java | 78 ++++++++++++- .../field/types/OffsetTimeTypeTest.java | 75 +++++++++++- .../j256/ormlite/h2/H2DatabaseResults.java | 16 ++- 18 files changed, 1067 insertions(+), 111 deletions(-) create mode 100644 src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java create mode 100644 src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java index e552c1601..3852a2c16 100644 --- a/src/main/java/com/j256/ormlite/field/types/InstantType.java +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -1,4 +1,80 @@ package com.j256.ormlite.field.types; -public class InstantType { +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.time.Instant; +import java.time.OffsetDateTime; +import java.time.ZoneOffset; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class InstantType extends BaseDataType { + + private static final InstantType singleton = new InstantType(); + public static InstantType getSingleton() { + try { + Class.forName("java.time.Instant", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; } + private InstantType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { Instant.class }); } + protected InstantType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.Sx")).toInstant(); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getOffsetDateTime(columnPos); + } + + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { + OffsetDateTime value = (OffsetDateTime) sqlArg; + return value.toInstant(); + } + + @Override + public Object javaToSqlArg(FieldType fieldType, Object javaObject) { + Instant instant = (Instant) javaObject; + return OffsetDateTime.ofInstant(instant, ZoneOffset.UTC); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + Instant datetime = (Instant) currentValue; + return datetime.plusNanos(1); + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == Instant.class); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java new file mode 100644 index 000000000..d844d3d61 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -0,0 +1,80 @@ +package com.j256.ormlite.field.types; + +import java.lang.reflect.Field; +import java.sql.Date; +import java.sql.SQLException; +import java.time.LocalDate; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class LocalDateSqlType extends BaseDataType { + + private static final LocalDateSqlType singleton = new LocalDateSqlType(); + public static LocalDateSqlType getSingleton() { + try { + Class.forName("java.time.LocalDate", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; + } + private LocalDateSqlType() { super(SqlType.LOCAL_DATE, new Class[] { LocalDate.class }); } + protected LocalDateSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return LocalDate.parse(defaultStr, DateTimeFormatter.ISO_LOCAL_DATE); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDate value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getDate(columnPos).toLocalDate(); + } + + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { + Date value = (Date) sqlArg; + return value.toLocalDate(); + } + + @Override + public Object javaToSqlArg(FieldType fieldType, Object javaObject) { + LocalDate date = (LocalDate) javaObject; + return Date.valueOf(date); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + LocalDate date = (LocalDate) currentValue; + return date.plusDays(1); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == LocalDate.class); + } +} diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java new file mode 100644 index 000000000..c2a28c950 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -0,0 +1,80 @@ +package com.j256.ormlite.field.types; + +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.sql.Timestamp; +import java.time.LocalDateTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDateTime class in the database as Timestamp object. + * + * @author graynk + */ +public class LocalDateTimeSqlType extends BaseDataType { + + private static final LocalDateTimeSqlType singleton = new LocalDateTimeSqlType(); + public static LocalDateTimeSqlType getSingleton() { + try { + Class.forName("java.time.LocalDateTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; + } + private LocalDateTimeSqlType() { super(SqlType.DATE, new Class[] { LocalDateTime.class }); } + protected LocalDateTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getTimestamp(columnPos).toLocalDateTime(); + } + + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { + Timestamp value = (Timestamp) sqlArg; + return value.toLocalDateTime(); + } + + @Override + public Object javaToSqlArg(FieldType fieldType, Object javaObject) { + LocalDateTime datetime = (LocalDateTime) javaObject; + return Timestamp.valueOf(datetime); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + LocalDateTime date = (LocalDateTime) currentValue; + return date.plusNanos(1); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == LocalDateTime.class); + } +} diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index b4084a93c..46749a925 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -1,4 +1,67 @@ package com.j256.ormlite.field.types; -public class LocalDateTimeType { +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.time.LocalDateTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class LocalDateTimeType extends BaseDataType { + + private static final LocalDateTimeType singleton = new LocalDateTimeType(); + public static LocalDateTimeType getSingleton() { + try { + Class.forName("java.time.LocalDateTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; + } + private LocalDateTimeType() { super(SqlType.LOCAL_DATE_TIME, new Class[] { LocalDateTime.class }); } + protected LocalDateTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getLocalDateTime(columnPos); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + LocalDateTime datetime = (LocalDateTime) currentValue; + return datetime.plusNanos(1); + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == LocalDateTime.class); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java index 9d10c39fe..b6e9e3b9b 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java @@ -1,6 +1,9 @@ package com.j256.ormlite.field.types; +import java.lang.reflect.Field; import java.sql.SQLException; +import java.time.LocalDate; +import java.time.format.DateTimeFormatter; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; @@ -8,116 +11,57 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDateTime class in the database as Date object. + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. * * @author graynk */ -public class LocalDateTimeType extends BaseDataType { +public class LocalDateType extends BaseDataType { - private static final LocalDateTimeType singleton = new LocalDateTimeType(); - public static LocalDateTimeType getSingleton() { return singleton; } - private LocalDateTimeType() { super(SqlType.LONG); } - - protected LocalDateTimeType(SqlType sqlType, Class[] classes) { - super(sqlType, classes); - } - - @Override - public Object javaToSqlArg(FieldType fieldType, Object javaObject) throws SQLException { - return extractMillis(javaObject); + private static final LocalDateType singleton = new LocalDateType(); + public static LocalDateType getSingleton() { + try { + Class.forName("java.time.LocalDate", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; } + private LocalDateType() { super(SqlType.LOCAL_DATE, new Class[] { LocalDate.class }); } + protected LocalDateType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return Long.parseLong(defaultStr); + return LocalDate.parse(defaultStr, DateTimeFormatter.ISO_LOCAL_DATE); } catch (NumberFormatException e) { - throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default DateTime value: " - + defaultStr, e); + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDate value: " + defaultStr, e); } } @Override public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { - return results.getLong(columnPos); - } - - @Override - public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) throws SQLException { - return createInstance((Long) sqlArg); - } - - @Override - public boolean isEscapedValue() { - return false; + return results.getLocalDate(columnPos); } @Override - public boolean isAppropriateId() { - return false; + public boolean isValidForVersion() { + return true; } @Override - public Class getPrimaryClass() { - try { - return getDateTimeClass(); - } catch (ClassNotFoundException e) { - // ignore the exception - return null; - } + public Object moveToNextValue(Object currentValue) { + LocalDate date = (LocalDate) currentValue; + return date.plusDays(1); } @Override - public boolean isValidForVersion() { + public boolean isArgumentHolderRequired() { return true; } @Override - public Object moveToNextValue(Object currentValue) throws SQLException { - long newVal = System.currentTimeMillis(); - if (currentValue == null) { - return createInstance(newVal); - } - Long currentVal = extractMillis(currentValue); - if (newVal == currentVal) { - return createInstance(newVal + 1L); - } else { - return createInstance(newVal); - } - } - - private Object createInstance(Long sqlArg) throws SQLException { - try { - if (millisConstructor == null) { - Class clazz = getDateTimeClass(); - millisConstructor = clazz.getConstructor(long.class); - } - return millisConstructor.newInstance(sqlArg); - } catch (Exception e) { - throw SqlExceptionUtil.create("Could not use reflection to construct a Joda DateTime", e); - } - } - - private Long extractMillis(Object javaObject) throws SQLException { - try { - if (getMillisMethod == null) { - Class clazz = getDateTimeClass(); - getMillisMethod = clazz.getMethod("getMillis"); - } - if (javaObject == null) { - return null; - } else { - return (Long) getMillisMethod.invoke(javaObject); - } - } catch (Exception e) { - throw SqlExceptionUtil.create("Could not use reflection to get millis from Joda DateTime: " + javaObject, e); - } - } - - private Class getDateTimeClass() throws ClassNotFoundException { - if (dateTimeClass == null) { - dateTimeClass = Class.forName("org.joda.time.DateTime"); - } - return dateTimeClass; + public boolean isValidForField(Field field) { + return (field.getType() == LocalDate.class); } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java new file mode 100644 index 000000000..2ce937459 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -0,0 +1,80 @@ +package com.j256.ormlite.field.types; + +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.sql.Time; +import java.time.LocalTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalTime class in the database as Time object. + * + * @author graynk + */ +public class LocalTimeSqlType extends BaseDataType { + + private static final LocalTimeSqlType singleton = new LocalTimeSqlType(); + public static LocalTimeSqlType getSingleton() { + try { + Class.forName("java.time.LocalTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; + } + private LocalTimeSqlType() { super(SqlType.LOCAL_TIME, new Class[] { LocalTime.class }); } + protected LocalTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss.SSSSSS")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getTime(columnPos).toLocalTime(); + } + + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { + Time value = (Time) sqlArg; + return value.toLocalTime(); + } + + @Override + public Object javaToSqlArg(FieldType fieldType, Object javaObject) { + LocalTime time = (LocalTime) javaObject; + return Time.valueOf(time); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + LocalTime time = (LocalTime) currentValue; + return time.plusNanos(1); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == LocalTime.class); + } +} diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index ff6761f77..1a83f5f04 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -1,4 +1,66 @@ package com.j256.ormlite.field.types; -public class LocalTimeType { +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.time.LocalTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class LocalTimeType extends BaseDataType { + + private static final LocalTimeType singleton = new LocalTimeType(); + public static LocalTimeType getSingleton() { + try { + Class.forName("java.time.LocalTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; } + private LocalTimeType() { super(SqlType.LOCAL_TIME, new Class[] { LocalTime.class }); } + protected LocalTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getLocalTime(columnPos); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + LocalTime time = (LocalTime) currentValue; + return time.plusNanos(1); + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == LocalTime.class); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java index 70c9f100e..876c2be79 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -1,4 +1,66 @@ package com.j256.ormlite.field.types; -public class OffsetDateTimeType { +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.time.OffsetDateTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class OffsetDateTimeType extends BaseDataType { + + private static final OffsetDateTimeType singleton = new OffsetDateTimeType(); + public static OffsetDateTimeType getSingleton() { + try { + Class.forName("java.time.OffsetDateTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; } + private OffsetDateTimeType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { OffsetDateTime.class }); } + protected OffsetDateTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSSx")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getOffsetDateTime(columnPos); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + OffsetDateTime datetime = (OffsetDateTime) currentValue; + return datetime.plusNanos(1); + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == OffsetDateTime.class); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index 7f078bc03..6d7a23c6e 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -1,4 +1,66 @@ package com.j256.ormlite.field.types; -public class OffsetTimeType { +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.time.OffsetTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class OffsetTimeType extends BaseDataType { + + private static final OffsetTimeType singleton = new OffsetTimeType(); + public static OffsetTimeType getSingleton() { + try { + Class.forName("java.time.OffsetTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; } + private OffsetTimeType() { super(SqlType.OFFSET_TIME, new Class[] { OffsetTime.class }); } + protected OffsetTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return OffsetTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ssx")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getOffsetTime(columnPos); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + OffsetTime time = (OffsetTime) currentValue; + return time.plusNanos(1); + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == OffsetTime.class); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } } diff --git a/src/main/java/com/j256/ormlite/support/DatabaseResults.java b/src/main/java/com/j256/ormlite/support/DatabaseResults.java index 2da52d58c..d6a5607bb 100644 --- a/src/main/java/com/j256/ormlite/support/DatabaseResults.java +++ b/src/main/java/com/j256/ormlite/support/DatabaseResults.java @@ -3,7 +3,9 @@ import java.io.Closeable; import java.io.InputStream; import java.math.BigDecimal; +import java.sql.Date; import java.sql.SQLException; +import java.sql.Time; import java.sql.Timestamp; import java.time.*; @@ -142,6 +144,16 @@ public interface DatabaseResults extends Closeable { */ public Timestamp getTimestamp(int columnIndex) throws SQLException; + /** + * Returns the SQL date value from the results at the column index. + */ + public Date getDate(int columnIndex) throws SQLException; + + /** + * Returns the SQL time value from the results at the column index. + */ + public Time getTime(int columnIndex) throws SQLException; + /** * Returns the LocalDate value from the results at the column index */ diff --git a/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java b/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java index 86ae3d6a0..dbf8d2a4b 100644 --- a/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java +++ b/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java @@ -1641,6 +1641,16 @@ public void init(Connection conn, String schemaName, String triggerName, String public void fire(Connection conn, Object[] oldRow, Object[] newRow) { callC++; } + + @Override + public void close() { + + } + + @Override + public void remove() { + + } } @Test diff --git a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java index 25acc4523..5a03d14b6 100644 --- a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java @@ -1,7 +1,77 @@ package com.j256.ormlite.field.types; -import static org.junit.Assert.*; +import static org.junit.Assert.assertEquals; -public class InstantTypeTest { +import java.sql.SQLException; +import java.time.Instant; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; -} \ No newline at end of file +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class InstantTypeTest extends BaseTypeTest { + + private static final String INSTANT_COLUMN = "instant"; + + @Test + public void testInstant() throws Exception { + Class clazz = InstantTable.class; + Dao dao = createDao(clazz, true); + Instant val = Instant.now().truncatedTo(ChronoUnit.MILLIS); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS"); + String valStr = val.toString(); + InstantTable foo = new InstantTable(); + foo.instant = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val, val, valStr, DataType.INSTANT, INSTANT_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = InstantTable.class; + Dao dao = createDao(clazz, true); + InstantTable foo = new InstantTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.INSTANT, + INSTANT_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + InstantTable.class.getDeclaredField(INSTANT_COLUMN), InstantTable.class); + DataType.INSTANT.getDataPersister().parseDefaultString(fieldType, "not valid instant string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notInstant"), + InstantTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.INSTANT) + String notInstant; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class InstantTable { + @DatabaseField(columnName = INSTANT_COLUMN) + Instant instant; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java index 75bfdfebd..24d971d27 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java @@ -1,7 +1,77 @@ package com.j256.ormlite.field.types; -import static org.junit.Assert.*; +import static org.junit.Assert.assertEquals; -public class LocalDateTimeTypeTest { +import java.sql.SQLException; +import java.time.LocalDateTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; -} \ No newline at end of file +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class LocalDateTimeTypeTest extends BaseTypeTest { + + private static final String DATE_TIME_COLUMN = "dateTime"; + + @Test + public void testDateTime() throws Exception { + Class clazz = DateTimeTable.class; + Dao dao = createDao(clazz, true); + LocalDateTime val = LocalDateTime.now(); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS"); + String valStr = formatter.format(val); + DateTimeTable foo = new DateTimeTable(); + foo.dateTime = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val, val, valStr, DataType.LOCAL_DATE_TIME, DATE_TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = DateTimeTable.class; + Dao dao = createDao(clazz, true); + DateTimeTable foo = new DateTimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.LOCAL_DATE_TIME, + DATE_TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + DateTimeTable.class.getDeclaredField(DATE_TIME_COLUMN), DateTimeTable.class); + DataType.LOCAL_DATE_TIME.getDataPersister().parseDefaultString(fieldType, "not valid datetime string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notDateTime"), + DateTimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.LOCAL_DATE_TIME) + String notDateTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class DateTimeTable { + @DatabaseField(columnName = DATE_TIME_COLUMN) + LocalDateTime dateTime; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java index 409c55c54..3b49604da 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java @@ -1,7 +1,76 @@ package com.j256.ormlite.field.types; -import static org.junit.Assert.*; +import static org.junit.Assert.assertEquals; -public class LocalDateTypeTest { +import java.sql.SQLException; +import java.time.LocalDate; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; -} \ No newline at end of file +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class LocalDateTypeTest extends BaseTypeTest { + + private static final String DATE_COLUMN = "date"; + + @Test + public void testDate() throws Exception { + Class clazz = DateTable.class; + Dao dao = createDao(clazz, true); + LocalDate val = LocalDate.now(); + LocalDate val2 = LocalDate.now(); + DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE; + String valStr = formatter.format(val); + DateTable foo = new DateTable(); + foo.date = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.LOCAL_DATE, DATE_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = DateTable.class; + Dao dao = createDao(clazz, true); + DateTable foo = new DateTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.LOCAL_DATE, DATE_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + DateTable.class.getDeclaredField(DATE_COLUMN), DateTable.class); + DataType.LOCAL_DATE.getDataPersister().parseDefaultString(fieldType, "not valid date string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notDate"), + DateTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.LOCAL_DATE) + String notDate; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class DateTable { + @DatabaseField(columnName = DATE_COLUMN) + LocalDate date; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java index 3c34f22c2..25f7354c8 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java @@ -1,7 +1,76 @@ package com.j256.ormlite.field.types; -import static org.junit.Assert.*; +import static org.junit.Assert.assertEquals; -public class LocalTimeTypeTest { +import java.sql.SQLException; +import java.time.LocalTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; -} \ No newline at end of file +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class LocalTimeTypeTest extends BaseTypeTest { + + private static final String TIME_COLUMN = "time"; + + @Test + public void testTime() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + LocalTime val = LocalTime.now().truncatedTo(ChronoUnit.SECONDS); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss"); + String valStr = formatter.format(val); + TimeTable foo = new TimeTable(); + foo.time = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val, val, valStr, DataType.LOCAL_TIME, TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + TimeTable foo = new TimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.LOCAL_TIME, TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + TimeTable.class.getDeclaredField(TIME_COLUMN), TimeTable.class); + DataType.LOCAL_TIME.getDataPersister().parseDefaultString(fieldType, "not valid time string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notTime"), + TimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.LOCAL_TIME) + String notTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class TimeTable { + @DatabaseField(columnName = TIME_COLUMN) + LocalTime time; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java index ef7d5147d..90ef320aa 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java @@ -1,7 +1,79 @@ package com.j256.ormlite.field.types; -import static org.junit.Assert.*; +import static org.junit.Assert.assertEquals; -public class OffsetDateTimeTypeTest { +import java.sql.SQLException; +import java.time.OffsetDateTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; -} \ No newline at end of file +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class OffsetDateTimeTypeTest extends BaseTypeTest { + + private static final String DATE_TIME_COLUMN = "dateTime"; + + @Test + public void testDateTime() throws Exception { + Class clazz = DateTimeTable.class; + Dao dao = createDao(clazz, true); + OffsetDateTime val = OffsetDateTime.now().truncatedTo(ChronoUnit.MILLIS); + + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSSx"); + String valStr = formatter.format(val); + + DateTimeTable foo = new DateTimeTable(); + foo.dateTime = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val, val, valStr, DataType.OFFSET_DATE_TIME, DATE_TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = DateTimeTable.class; + Dao dao = createDao(clazz, true); + DateTimeTable foo = new DateTimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.OFFSET_DATE_TIME, + DATE_TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + DateTimeTable.class.getDeclaredField(DATE_TIME_COLUMN), DateTimeTable.class); + DataType.OFFSET_DATE_TIME.getDataPersister().parseDefaultString(fieldType, "not valid datetime string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notDateTime"), + DateTimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.OFFSET_DATE_TIME) + String notDateTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class DateTimeTable { + @DatabaseField(columnName = DATE_TIME_COLUMN) + OffsetDateTime dateTime; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java index 1ea3e8d46..d33f565e3 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java @@ -1,7 +1,76 @@ package com.j256.ormlite.field.types; -import static org.junit.Assert.*; +import static org.junit.Assert.assertEquals; -public class OffsetTimeTypeTest { +import java.sql.SQLException; +import java.time.OffsetTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; -} \ No newline at end of file +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class OffsetTimeTypeTest extends BaseTypeTest { + + private static final String TIME_COLUMN = "time"; + + @Test + public void testTime() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + OffsetTime val = OffsetTime.now().truncatedTo(ChronoUnit.SECONDS); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ssx"); + String valStr = formatter.format(val); + TimeTable foo = new TimeTable(); + foo.time = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val, val, valStr, DataType.OFFSET_TIME, TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + TimeTable foo = new TimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.OFFSET_TIME, TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + TimeTable.class.getDeclaredField(TIME_COLUMN), TimeTable.class); + DataType.OFFSET_TIME.getDataPersister().parseDefaultString(fieldType, "not valid time string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notTime"), + TimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.OFFSET_TIME) + String notTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class TimeTable { + @DatabaseField(columnName = TIME_COLUMN) + OffsetTime time; + } +} diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java index 08005ceb9..a44747687 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java @@ -3,11 +3,7 @@ import java.io.IOException; import java.io.InputStream; import java.math.BigDecimal; -import java.sql.Blob; -import java.sql.ResultSet; -import java.sql.ResultSetMetaData; -import java.sql.SQLException; -import java.sql.Timestamp; +import java.sql.*; import java.time.*; import com.j256.ormlite.dao.ObjectCache; @@ -150,6 +146,16 @@ public Timestamp getTimestamp(int columnIndex) throws SQLException { return resultSet.getTimestamp(columnIndex + 1); } + @Override + public Time getTime(int columnIndex) throws SQLException { + return resultSet.getTime(columnIndex + 1); + } + + @Override + public Date getDate(int columnIndex) throws SQLException { + return resultSet.getDate(columnIndex + 1); + } + @Override public LocalDate getLocalDate(int columnIndex) throws SQLException { return resultSet.getObject(columnIndex + 1, LocalDate.class); From 92e2b219964eda055d93c20855c4a7535599f076 Mon Sep 17 00:00:00 2001 From: graynk Date: Tue, 11 Dec 2018 13:15:08 +0500 Subject: [PATCH 04/12] Added SQL conversions for acceptable java.time types. New tests pass on latest H2. No need to @override methods in BaseDaoImplTest - compat. for newer H2 versions --- .../java/com/j256/ormlite/field/DataType.java | 24 ++++++ .../j256/ormlite/field/types/InstantType.java | 2 +- .../ormlite/field/types/LocalDateSqlType.java | 16 ++-- .../field/types/LocalDateTimeSqlType.java | 4 +- .../field/types/LocalDateTimeType.java | 2 +- .../ormlite/field/types/LocalTimeSqlType.java | 4 +- .../ormlite/field/types/LocalTimeType.java | 2 +- .../field/types/OffsetDateTimeType.java | 2 +- .../field/types/OffsetTimeSqlType.java | 80 +++++++++++++++++++ .../ormlite/field/types/OffsetTimeType.java | 2 +- .../com/j256/ormlite/dao/BaseDaoImplTest.java | 2 - .../ormlite/field/types/InstantTypeTest.java | 13 +-- .../field/types/LocalDateSqlTypeTest.java | 76 ++++++++++++++++++ .../field/types/LocalDateTimeSqlTypeTest.java | 77 ++++++++++++++++++ .../field/types/LocalDateTimeTypeTest.java | 4 +- .../field/types/LocalDateTypeTest.java | 1 - .../field/types/LocalTimeSqlTypeTest.java | 77 ++++++++++++++++++ .../field/types/LocalTimeTypeTest.java | 1 - .../field/types/OffsetDateTimeTypeTest.java | 2 +- .../field/types/OffsetTimeSqlTypeTest.java | 77 ++++++++++++++++++ .../field/types/OffsetTimeTypeTest.java | 5 +- 21 files changed, 439 insertions(+), 34 deletions(-) create mode 100644 src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java create mode 100644 src/test/java/com/j256/ormlite/field/types/LocalDateSqlTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/LocalDateTimeSqlTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/LocalTimeSqlTypeTest.java create mode 100644 src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java diff --git a/src/main/java/com/j256/ormlite/field/DataType.java b/src/main/java/com/j256/ormlite/field/DataType.java index 472478ab9..596f0afc7 100644 --- a/src/main/java/com/j256/ormlite/field/DataType.java +++ b/src/main/java/com/j256/ormlite/field/DataType.java @@ -212,6 +212,30 @@ public enum DataType { * Marker for fields that are unknown. */ UNKNOWN(null), + /** + * Persists the {@link java.time.LocalDate} Java class. By default this will use + * {@link #LOCAL_DATE} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + LOCAL_DATE_SQL(LocalDateSqlType.getSingleton()), + /** + * Persists the {@link java.time.LocalTime} Java class. By default this will use + * {@link #LOCAL_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + LOCAL_TIME_SQL(LocalTimeSqlType.getSingleton()), + /** + * Persists the {@link java.time.LocalDateTime} Java class. By default this will use + * {@link #LOCAL_DATE_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + LOCAL_DATE_TIME_SQL(LocalDateTimeSqlType.getSingleton()), + /** + * Persists the {@link java.time.OffsetTime} Java class. By default this will use + * {@link #OFFSET_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + OFFSET_TIME_SQL(OffsetTimeSqlType.getSingleton()), /** * Persists the {@link java.time.LocalDate} Java class. * diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java index 3852a2c16..12c707fee 100644 --- a/src/main/java/com/j256/ormlite/field/types/InstantType.java +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -33,7 +33,7 @@ public static InstantType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.Sx")).toInstant(); + return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java index d844d3d61..7f555c2c4 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -33,7 +33,7 @@ public static LocalDateSqlType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return LocalDate.parse(defaultStr, DateTimeFormatter.ISO_LOCAL_DATE); + return Date.valueOf(LocalDate.parse(defaultStr, DateTimeFormatter.ISO_LOCAL_DATE)); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDate value: " + defaultStr, e); @@ -42,13 +42,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws @Override public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { - return results.getDate(columnPos).toLocalDate(); - } - - @Override - public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { - Date value = (Date) sqlArg; - return value.toLocalDate(); + return results.getDate(columnPos); } @Override @@ -57,6 +51,12 @@ public Object javaToSqlArg(FieldType fieldType, Object javaObject) { return Date.valueOf(date); } + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { + Date date = (Date) sqlArg; + return date.toLocalDate(); + } + @Override public boolean isValidForVersion() { return true; diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java index c2a28c950..1c1f129a3 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -33,7 +33,7 @@ public static LocalDateTimeSqlType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS")); + return Timestamp.valueOf(LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]"))); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); @@ -42,7 +42,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws @Override public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { - return results.getTimestamp(columnPos).toLocalDateTime(); + return results.getTimestamp(columnPos); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index 46749a925..55e04d581 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -32,7 +32,7 @@ public static LocalDateTimeType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS")); + return LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java index 2ce937459..bcf6c952f 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -33,7 +33,7 @@ public static LocalTimeSqlType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss.SSSSSS")); + return Time.valueOf(LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]"))); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalTime value: " + defaultStr, e); @@ -42,7 +42,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws @Override public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { - return results.getTime(columnPos).toLocalTime(); + return results.getTime(columnPos); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index 1a83f5f04..eacc3e487 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -31,7 +31,7 @@ public static LocalTimeType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss")); + return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java index 876c2be79..3d858647d 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -31,7 +31,7 @@ public static OffsetDateTimeType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSSx")); + return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java new file mode 100644 index 000000000..7068cae23 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java @@ -0,0 +1,80 @@ +package com.j256.ormlite.field.types; + +import java.lang.reflect.Field; +import java.sql.SQLException; +import java.time.LocalDate; +import java.time.OffsetDateTime; +import java.time.OffsetTime; +import java.time.format.DateTimeFormatter; + +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.misc.SqlExceptionUtil; +import com.j256.ormlite.support.DatabaseResults; + +/** + * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * + * @author graynk + */ +public class OffsetTimeSqlType extends BaseDataType { + + private static final OffsetTimeSqlType singleton = new OffsetTimeSqlType(); + public static OffsetTimeSqlType getSingleton() { + try { + Class.forName("java.time.OffsetTime", false, null); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } + return singleton; } + private OffsetTimeSqlType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { OffsetTime.class }); } + protected OffsetTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + + @Override + public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { + try { + return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); + } catch (NumberFormatException e) { + throw SqlExceptionUtil.create("Problems with field " + fieldType + + " parsing default LocalDateTime value: " + defaultStr, e); + } + } + + @Override + public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { + return results.getOffsetDateTime(columnPos); + } + + @Override + public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { + OffsetDateTime value = (OffsetDateTime) sqlArg; + return value.toOffsetTime(); + } + + @Override + public Object javaToSqlArg(FieldType fieldType, Object javaObject) { + OffsetTime time = (OffsetTime) javaObject; + return time.atDate(LocalDate.ofEpochDay(0)); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public Object moveToNextValue(Object currentValue) { + OffsetTime time = (OffsetTime) currentValue; + return time.plusNanos(1); + } + + @Override + public boolean isValidForField(Field field) { + return (field.getType() == OffsetTime.class); + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } +} diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index 6d7a23c6e..3f8299f51 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -31,7 +31,7 @@ public static OffsetTimeType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ssx")); + return OffsetTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); diff --git a/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java b/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java index dbf8d2a4b..1026a52b4 100644 --- a/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java +++ b/src/test/java/com/j256/ormlite/dao/BaseDaoImplTest.java @@ -1642,12 +1642,10 @@ public void fire(Connection conn, Object[] oldRow, Object[] newRow) { callC++; } - @Override public void close() { } - @Override public void remove() { } diff --git a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java index 5a03d14b6..92740bd56 100644 --- a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java @@ -2,11 +2,11 @@ import static org.junit.Assert.assertEquals; -import java.sql.SQLException; import java.time.Instant; +import java.time.OffsetDateTime; +import java.time.ZoneOffset; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; -import java.time.temporal.ChronoUnit; import org.junit.Test; @@ -24,14 +24,15 @@ public class InstantTypeTest extends BaseTypeTest { public void testInstant() throws Exception { Class clazz = InstantTable.class; Dao dao = createDao(clazz, true); - Instant val = Instant.now().truncatedTo(ChronoUnit.MILLIS); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS"); - String valStr = val.toString(); + Instant val = Instant.now(); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x"); + OffsetDateTime val2 = OffsetDateTime.ofInstant(val, ZoneOffset.UTC); + String valStr = formatter.format(val2); InstantTable foo = new InstantTable(); foo.instant = val; assertEquals(1, dao.create(foo)); - testType(dao, foo, clazz, val, val, val, valStr, DataType.INSTANT, INSTANT_COLUMN, false, + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.INSTANT, INSTANT_COLUMN, false, true, true, false, true, false, true, false); } diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateSqlTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateSqlTypeTest.java new file mode 100644 index 000000000..4c9c9a1ea --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateSqlTypeTest.java @@ -0,0 +1,76 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.assertEquals; + +import java.sql.Date; +import java.time.LocalDate; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; + +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class LocalDateSqlTypeTest extends BaseTypeTest { + + private static final String DATE_COLUMN = "date"; + + @Test + public void testDate() throws Exception { + Class clazz = DateTable.class; + Dao dao = createDao(clazz, true); + LocalDate val = LocalDate.now(); + Date val2 = Date.valueOf(val); + DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE; + String valStr = formatter.format(val); + DateTable foo = new DateTable(); + foo.date = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.LOCAL_DATE_SQL, DATE_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = DateTable.class; + Dao dao = createDao(clazz, true); + DateTable foo = new DateTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.LOCAL_DATE_SQL, DATE_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + DateTable.class.getDeclaredField(DATE_COLUMN), DateTable.class); + DataType.LOCAL_DATE.getDataPersister().parseDefaultString(fieldType, "not valid date string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notDate"), + DateTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.LOCAL_DATE_SQL) + String notDate; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class DateTable { + @DatabaseField(columnName = DATE_COLUMN, dataType = DataType.LOCAL_DATE_SQL) + LocalDate date; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeSqlTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeSqlTypeTest.java new file mode 100644 index 000000000..fa5e9b680 --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeSqlTypeTest.java @@ -0,0 +1,77 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.assertEquals; + +import java.sql.Timestamp; +import java.time.LocalDateTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; + +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class LocalDateTimeSqlTypeTest extends BaseTypeTest { + + private static final String DATE_TIME_COLUMN = "dateTime"; + + @Test + public void testDateTime() throws Exception { + Class clazz = DateTimeTable.class; + Dao dao = createDao(clazz, true); + LocalDateTime val = LocalDateTime.now(); + Timestamp val2 = Timestamp.valueOf(val); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS"); + String valStr = formatter.format(val); + DateTimeTable foo = new DateTimeTable(); + foo.dateTime = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.LOCAL_DATE_TIME_SQL, DATE_TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = DateTimeTable.class; + Dao dao = createDao(clazz, true); + DateTimeTable foo = new DateTimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.LOCAL_DATE_TIME_SQL, + DATE_TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + DateTimeTable.class.getDeclaredField(DATE_TIME_COLUMN), DateTimeTable.class); + DataType.LOCAL_DATE_TIME.getDataPersister().parseDefaultString(fieldType, "not valid datetime string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notDateTime"), + DateTimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.LOCAL_DATE_TIME_SQL) + String notDateTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class DateTimeTable { + @DatabaseField(columnName = DATE_TIME_COLUMN, dataType = DataType.LOCAL_DATE_TIME_SQL) + LocalDateTime dateTime; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java index 24d971d27..75badfe04 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java @@ -2,11 +2,9 @@ import static org.junit.Assert.assertEquals; -import java.sql.SQLException; import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; -import java.time.temporal.ChronoUnit; import org.junit.Test; @@ -25,7 +23,7 @@ public void testDateTime() throws Exception { Class clazz = DateTimeTable.class; Dao dao = createDao(clazz, true); LocalDateTime val = LocalDateTime.now(); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSS"); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS"); String valStr = formatter.format(val); DateTimeTable foo = new DateTimeTable(); foo.dateTime = val; diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java index 3b49604da..cca27175e 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTypeTest.java @@ -2,7 +2,6 @@ import static org.junit.Assert.assertEquals; -import java.sql.SQLException; import java.time.LocalDate; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; diff --git a/src/test/java/com/j256/ormlite/field/types/LocalTimeSqlTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalTimeSqlTypeTest.java new file mode 100644 index 000000000..0805eea47 --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/LocalTimeSqlTypeTest.java @@ -0,0 +1,77 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.assertEquals; + +import java.sql.Time; +import java.time.LocalTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; + +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class LocalTimeSqlTypeTest extends BaseTypeTest { + + private static final String TIME_COLUMN = "time"; + + @Test + public void testTime() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + LocalTime val = LocalTime.now().truncatedTo(ChronoUnit.SECONDS); + Time val2 = Time.valueOf(val); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss"); + String valStr = formatter.format(val); + TimeTable foo = new TimeTable(); + foo.time = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.LOCAL_TIME_SQL, TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + TimeTable foo = new TimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.LOCAL_TIME_SQL, TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + TimeTable.class.getDeclaredField(TIME_COLUMN), TimeTable.class); + DataType.LOCAL_TIME_SQL.getDataPersister().parseDefaultString(fieldType, "not valid time string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notTime"), + TimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.LOCAL_TIME_SQL) + String notTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class TimeTable { + @DatabaseField(columnName = TIME_COLUMN, dataType = DataType.LOCAL_TIME_SQL) + LocalTime time; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java index 25f7354c8..760dec6c6 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java @@ -2,7 +2,6 @@ import static org.junit.Assert.assertEquals; -import java.sql.SQLException; import java.time.LocalTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java index 90ef320aa..2cb6595f1 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java @@ -26,7 +26,7 @@ public void testDateTime() throws Exception { Dao dao = createDao(clazz, true); OffsetDateTime val = OffsetDateTime.now().truncatedTo(ChronoUnit.MILLIS); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSSSSSx"); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x"); String valStr = formatter.format(val); DateTimeTable foo = new DateTimeTable(); diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java new file mode 100644 index 000000000..921e107aa --- /dev/null +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java @@ -0,0 +1,77 @@ +package com.j256.ormlite.field.types; + +import static org.junit.Assert.assertEquals; + +import java.time.LocalDate; +import java.time.OffsetDateTime; +import java.time.OffsetTime; +import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeParseException; + +import org.junit.Test; + +import com.j256.ormlite.dao.Dao; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DatabaseField; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.table.DatabaseTable; + +public class OffsetTimeSqlTypeTest extends BaseTypeTest { + + private static final String TIME_COLUMN = "time"; + + @Test + public void testTime() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + OffsetTime val = OffsetTime.now(); + OffsetDateTime val2 = val.atDate(LocalDate.ofEpochDay(0)); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x"); + String valStr = formatter.format(val2); + TimeTable foo = new TimeTable(); + foo.time = val; + assertEquals(1, dao.create(foo)); + + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.OFFSET_TIME_SQL, TIME_COLUMN, false, + true, true, false, true, false, + true, false); + } + + @Test + public void testDateNull() throws Exception { + Class clazz = TimeTable.class; + Dao dao = createDao(clazz, true); + TimeTable foo = new TimeTable(); + assertEquals(1, dao.create(foo)); + testType(dao, foo, clazz, null, null, null, null, DataType.OFFSET_TIME_SQL, TIME_COLUMN, + false, true, true, false, true, + false, true, false); + } + + @Test(expected = DateTimeParseException.class) + public void testDateParseInvalid() throws Exception { + FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, + TimeTable.class.getDeclaredField(TIME_COLUMN), TimeTable.class); + DataType.OFFSET_TIME_SQL.getDataPersister().parseDefaultString(fieldType, "not valid time string"); + } + + @Test(expected = IllegalArgumentException.class) + public void testInvalidDateField() throws Exception { + FieldType.createFieldType(connectionSource, TABLE_NAME, InvalidDate.class.getDeclaredField("notTime"), + TimeTable.class); + } + + /* ============================================================================================ */ + + @DatabaseTable + protected static class InvalidDate { + @DatabaseField(dataType = DataType.OFFSET_TIME_SQL) + String notTime; + } + + @DatabaseTable(tableName = TABLE_NAME) + protected static class TimeTable { + @DatabaseField(columnName = TIME_COLUMN, dataType = DataType.OFFSET_TIME_SQL) + OffsetTime time; + } +} diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java index d33f565e3..14d6269a7 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java @@ -2,7 +2,6 @@ import static org.junit.Assert.assertEquals; -import java.sql.SQLException; import java.time.OffsetTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; @@ -24,8 +23,8 @@ public class OffsetTimeTypeTest extends BaseTypeTest { public void testTime() throws Exception { Class clazz = TimeTable.class; Dao dao = createDao(clazz, true); - OffsetTime val = OffsetTime.now().truncatedTo(ChronoUnit.SECONDS); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ssx"); + OffsetTime val = OffsetTime.now(); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS]x"); String valStr = formatter.format(val); TimeTable foo = new TimeTable(); foo.time = val; From dea9e30d7052ef54aee51558fa0f3b0d1f09619d Mon Sep 17 00:00:00 2001 From: graynk Date: Tue, 11 Dec 2018 13:51:47 +0500 Subject: [PATCH 05/12] Added workaround for H2 not supporting TIME WITH TIME ZONE The same will be done for ormlite-jdbc --- .../java/com/j256/ormlite/h2/H2DatabaseType.java | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java index 2d383f7b0..bade80dd8 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java @@ -5,7 +5,11 @@ import java.util.List; import com.j256.ormlite.db.BaseDatabaseType; +import com.j256.ormlite.field.DataPersister; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.FieldConverter; import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.types.OffsetTimeType; /** * H2 database type. @@ -71,4 +75,13 @@ protected void configureGeneratedId(String tableName, StringBuilder sb, FieldTyp public boolean isCreateIfNotExistsSupported() { return true; } + + @Override + public FieldConverter getFieldConverter(DataPersister dataPersister, FieldType fieldType) { + // H2 doesn't support TIME WITH TIME ZONE + if (fieldType.getDataPersister() instanceof OffsetTimeType) + return DataType.OFFSET_TIME_SQL.getDataPersister(); + // default is to use the dataPersister itself + return dataPersister; + } } From 51001b3125d1a0ead8cddab9e313464dd0595ef7 Mon Sep 17 00:00:00 2001 From: graynk Date: Tue, 11 Dec 2018 14:24:35 +0500 Subject: [PATCH 06/12] the way it's done in ormlite-jdbc --- src/test/java/com/j256/ormlite/h2/H2DatabaseType.java | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java index bade80dd8..a68b3e4fa 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java @@ -5,10 +5,7 @@ import java.util.List; import com.j256.ormlite.db.BaseDatabaseType; -import com.j256.ormlite.field.DataPersister; -import com.j256.ormlite.field.DataType; -import com.j256.ormlite.field.FieldConverter; -import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.*; import com.j256.ormlite.field.types.OffsetTimeType; /** @@ -79,7 +76,7 @@ public boolean isCreateIfNotExistsSupported() { @Override public FieldConverter getFieldConverter(DataPersister dataPersister, FieldType fieldType) { // H2 doesn't support TIME WITH TIME ZONE - if (fieldType.getDataPersister() instanceof OffsetTimeType) + if (dataPersister.getSqlType() == SqlType.OFFSET_TIME) return DataType.OFFSET_TIME_SQL.getDataPersister(); // default is to use the dataPersister itself return dataPersister; From 77b25d6e378f8b789fce55d93d7bad97d87f4d98 Mon Sep 17 00:00:00 2001 From: graynk Date: Wed, 12 Dec 2018 14:11:57 +0500 Subject: [PATCH 07/12] Backwards compatible with Java 6 Unfolded imports --- .../java/com/j256/ormlite/field/DataType.java | 49 ++++++++++++++++++- .../j256/ormlite/field/types/InstantType.java | 21 +++++--- .../ormlite/field/types/LocalDateSqlType.java | 13 +++-- .../field/types/LocalDateTimeSqlType.java | 13 +++-- .../field/types/LocalDateTimeType.java | 13 +++-- .../ormlite/field/types/LocalDateType.java | 13 +++-- .../ormlite/field/types/LocalTimeSqlType.java | 13 +++-- .../ormlite/field/types/LocalTimeType.java | 16 +++--- .../field/types/OffsetDateTimeType.java | 16 +++--- .../field/types/OffsetTimeSqlType.java | 16 +++--- .../ormlite/field/types/OffsetTimeType.java | 16 +++--- .../j256/ormlite/support/DatabaseResults.java | 6 ++- .../field/types/OffsetDateTimeTypeTest.java | 1 - .../field/types/OffsetTimeTypeTest.java | 1 - .../j256/ormlite/h2/H2DatabaseResults.java | 14 +++++- .../com/j256/ormlite/h2/H2DatabaseType.java | 7 ++- 16 files changed, 163 insertions(+), 65 deletions(-) diff --git a/src/main/java/com/j256/ormlite/field/DataType.java b/src/main/java/com/j256/ormlite/field/DataType.java index 596f0afc7..672678663 100644 --- a/src/main/java/com/j256/ormlite/field/DataType.java +++ b/src/main/java/com/j256/ormlite/field/DataType.java @@ -3,7 +3,54 @@ import java.math.BigDecimal; import java.math.BigInteger; -import com.j256.ormlite.field.types.*; +import com.j256.ormlite.field.types.BigDecimalNumericType; +import com.j256.ormlite.field.types.BigDecimalStringType; +import com.j256.ormlite.field.types.BigIntegerType; +import com.j256.ormlite.field.types.BooleanCharType; +import com.j256.ormlite.field.types.BooleanIntegerType; +import com.j256.ormlite.field.types.BooleanObjectType; +import com.j256.ormlite.field.types.BooleanType; +import com.j256.ormlite.field.types.ByteArrayType; +import com.j256.ormlite.field.types.ByteObjectType; +import com.j256.ormlite.field.types.ByteType; +import com.j256.ormlite.field.types.CharType; +import com.j256.ormlite.field.types.CharacterObjectType; +import com.j256.ormlite.field.types.DateLongType; +import com.j256.ormlite.field.types.DateIntegerType; +import com.j256.ormlite.field.types.DateStringType; +import com.j256.ormlite.field.types.DateTimeType; +import com.j256.ormlite.field.types.DateType; +import com.j256.ormlite.field.types.DoubleObjectType; +import com.j256.ormlite.field.types.DoubleType; +import com.j256.ormlite.field.types.EnumIntegerType; +import com.j256.ormlite.field.types.EnumStringType; +import com.j256.ormlite.field.types.EnumToStringType; +import com.j256.ormlite.field.types.FloatObjectType; +import com.j256.ormlite.field.types.FloatType; +import com.j256.ormlite.field.types.IntType; +import com.j256.ormlite.field.types.IntegerObjectType; +import com.j256.ormlite.field.types.LongObjectType; +import com.j256.ormlite.field.types.LongStringType; +import com.j256.ormlite.field.types.LongType; +import com.j256.ormlite.field.types.NativeUuidType; +import com.j256.ormlite.field.types.SerializableType; +import com.j256.ormlite.field.types.ShortObjectType; +import com.j256.ormlite.field.types.ShortType; +import com.j256.ormlite.field.types.SqlDateType; +import com.j256.ormlite.field.types.StringBytesType; +import com.j256.ormlite.field.types.StringType; +import com.j256.ormlite.field.types.TimeStampType; +import com.j256.ormlite.field.types.UuidType; +import com.j256.ormlite.field.types.LocalDateType; +import com.j256.ormlite.field.types.LocalTimeType; +import com.j256.ormlite.field.types.LocalDateTimeType; +import com.j256.ormlite.field.types.OffsetTimeType; +import com.j256.ormlite.field.types.OffsetDateTimeType; +import com.j256.ormlite.field.types.InstantType; +import com.j256.ormlite.field.types.LocalDateSqlType; +import com.j256.ormlite.field.types.LocalTimeSqlType; +import com.j256.ormlite.field.types.LocalDateTimeSqlType; +import com.j256.ormlite.field.types.OffsetTimeSqlType; /** * Data type enumeration that corresponds to a {@link DataPersister}. diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java index 12c707fee..82bdec380 100644 --- a/src/main/java/com/j256/ormlite/field/types/InstantType.java +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -4,7 +4,7 @@ import java.sql.SQLException; import java.time.Instant; import java.time.OffsetDateTime; -import java.time.ZoneOffset; +import java.time.ZoneId; import java.time.format.DateTimeFormatter; import com.j256.ormlite.field.FieldType; @@ -19,14 +19,18 @@ */ public class InstantType extends BaseDataType { - private static final InstantType singleton = new InstantType(); + private static InstantType singleton; public static InstantType getSingleton() { - try { - Class.forName("java.time.Instant", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.Instant", false, null); + singleton = new InstantType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } - return singleton; } + return singleton; + } private InstantType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { Instant.class }); } protected InstantType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @@ -54,7 +58,8 @@ public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { @Override public Object javaToSqlArg(FieldType fieldType, Object javaObject) { Instant instant = (Instant) javaObject; - return OffsetDateTime.ofInstant(instant, ZoneOffset.UTC); + // ZoneOffset.UTC is evaluated at InstantType creation, fails on Java 6. Using ZoneId.of() instead + return OffsetDateTime.ofInstant(instant, ZoneId.of("UTC")); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java index 7f555c2c4..a9799d459 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -18,12 +18,15 @@ */ public class LocalDateSqlType extends BaseDataType { - private static final LocalDateSqlType singleton = new LocalDateSqlType(); + private static LocalDateSqlType singleton; public static LocalDateSqlType getSingleton() { - try { - Class.forName("java.time.LocalDate", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.LocalDate", false, null); + singleton = new LocalDateSqlType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } return singleton; } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java index 1c1f129a3..5c5167fef 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -18,12 +18,15 @@ */ public class LocalDateTimeSqlType extends BaseDataType { - private static final LocalDateTimeSqlType singleton = new LocalDateTimeSqlType(); + private static LocalDateTimeSqlType singleton; public static LocalDateTimeSqlType getSingleton() { - try { - Class.forName("java.time.LocalDateTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.LocalDateTime", false, null); + singleton = new LocalDateTimeSqlType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } return singleton; } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index 55e04d581..07f03dbb4 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -17,12 +17,15 @@ */ public class LocalDateTimeType extends BaseDataType { - private static final LocalDateTimeType singleton = new LocalDateTimeType(); + private static LocalDateTimeType singleton; public static LocalDateTimeType getSingleton() { - try { - Class.forName("java.time.LocalDateTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.LocalDateTime", false, null); + singleton = new LocalDateTimeType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } return singleton; } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java index b6e9e3b9b..9e4742bcb 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java @@ -17,12 +17,15 @@ */ public class LocalDateType extends BaseDataType { - private static final LocalDateType singleton = new LocalDateType(); + private static LocalDateType singleton; public static LocalDateType getSingleton() { - try { - Class.forName("java.time.LocalDate", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.LocalDate", false, null); + singleton = new LocalDateType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } return singleton; } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java index bcf6c952f..490c96926 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -18,12 +18,15 @@ */ public class LocalTimeSqlType extends BaseDataType { - private static final LocalTimeSqlType singleton = new LocalTimeSqlType(); + private static LocalTimeSqlType singleton; public static LocalTimeSqlType getSingleton() { - try { - Class.forName("java.time.LocalTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.LocalTime", false, null); + singleton = new LocalTimeSqlType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } return singleton; } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index eacc3e487..13e557816 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -17,14 +17,18 @@ */ public class LocalTimeType extends BaseDataType { - private static final LocalTimeType singleton = new LocalTimeType(); + private static LocalTimeType singleton; public static LocalTimeType getSingleton() { - try { - Class.forName("java.time.LocalTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.LocalTime", false, null); + singleton = new LocalTimeType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } - return singleton; } + return singleton; + } private LocalTimeType() { super(SqlType.LOCAL_TIME, new Class[] { LocalTime.class }); } protected LocalTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java index 3d858647d..e0b4f4583 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -17,14 +17,18 @@ */ public class OffsetDateTimeType extends BaseDataType { - private static final OffsetDateTimeType singleton = new OffsetDateTimeType(); + private static OffsetDateTimeType singleton; public static OffsetDateTimeType getSingleton() { - try { - Class.forName("java.time.OffsetDateTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.OffsetDateTime", false, null); + singleton = new OffsetDateTimeType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } - return singleton; } + return singleton; + } private OffsetDateTimeType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { OffsetDateTime.class }); } protected OffsetDateTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java index 7068cae23..ffc7857f6 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java @@ -19,14 +19,18 @@ */ public class OffsetTimeSqlType extends BaseDataType { - private static final OffsetTimeSqlType singleton = new OffsetTimeSqlType(); + private static OffsetTimeSqlType singleton; public static OffsetTimeSqlType getSingleton() { - try { - Class.forName("java.time.OffsetTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.OffsetTime", false, null); + singleton = new OffsetTimeSqlType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } - return singleton; } + return singleton; + } private OffsetTimeSqlType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { OffsetTime.class }); } protected OffsetTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index 3f8299f51..0efd78f75 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -17,14 +17,18 @@ */ public class OffsetTimeType extends BaseDataType { - private static final OffsetTimeType singleton = new OffsetTimeType(); + private static OffsetTimeType singleton; public static OffsetTimeType getSingleton() { - try { - Class.forName("java.time.OffsetTime", false, null); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; + if (singleton == null) { + try { + Class.forName("java.time.OffsetTime", false, null); + singleton = new OffsetTimeType(); + } catch (ClassNotFoundException e) { + return null; // No java.time on classpath; + } } - return singleton; } + return singleton; + } private OffsetTimeType() { super(SqlType.OFFSET_TIME, new Class[] { OffsetTime.class }); } protected OffsetTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/support/DatabaseResults.java b/src/main/java/com/j256/ormlite/support/DatabaseResults.java index d6a5607bb..f96430530 100644 --- a/src/main/java/com/j256/ormlite/support/DatabaseResults.java +++ b/src/main/java/com/j256/ormlite/support/DatabaseResults.java @@ -7,7 +7,11 @@ import java.sql.SQLException; import java.sql.Time; import java.sql.Timestamp; -import java.time.*; +import java.time.LocalDate; +import java.time.LocalTime; +import java.time.LocalDateTime; +import java.time.OffsetTime; +import java.time.OffsetDateTime; import com.j256.ormlite.dao.ObjectCache; diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java index 2cb6595f1..87262e6c4 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetDateTimeTypeTest.java @@ -2,7 +2,6 @@ import static org.junit.Assert.assertEquals; -import java.sql.SQLException; import java.time.OffsetDateTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java index 14d6269a7..1f42f420a 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java @@ -5,7 +5,6 @@ import java.time.OffsetTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; -import java.time.temporal.ChronoUnit; import org.junit.Test; diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java index a44747687..c63da1fca 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseResults.java @@ -3,8 +3,18 @@ import java.io.IOException; import java.io.InputStream; import java.math.BigDecimal; -import java.sql.*; -import java.time.*; +import java.sql.Blob; +import java.sql.ResultSet; +import java.sql.ResultSetMetaData; +import java.sql.SQLException; +import java.sql.Timestamp; +import java.sql.Time; +import java.sql.Date; +import java.time.LocalDate; +import java.time.LocalTime; +import java.time.LocalDateTime; +import java.time.OffsetTime; +import java.time.OffsetDateTime; import com.j256.ormlite.dao.ObjectCache; import com.j256.ormlite.misc.IOUtils; diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java index a68b3e4fa..451970b11 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java @@ -5,8 +5,11 @@ import java.util.List; import com.j256.ormlite.db.BaseDatabaseType; -import com.j256.ormlite.field.*; -import com.j256.ormlite.field.types.OffsetTimeType; +import com.j256.ormlite.field.FieldType; +import com.j256.ormlite.field.SqlType; +import com.j256.ormlite.field.DataType; +import com.j256.ormlite.field.DataPersister; +import com.j256.ormlite.field.FieldConverter; /** * H2 database type. From 3cca49cd3f8685714da3a003eb90cfde0e0d12bb Mon Sep 17 00:00:00 2001 From: graynk Date: Thu, 13 Dec 2018 11:25:38 +0500 Subject: [PATCH 08/12] Inheritance to reduce code duplication Fixed typos, added precision to tests --- .../field/types/BaseLocalDateType.java | 19 ++++++++++++++ .../j256/ormlite/field/types/InstantType.java | 14 ++--------- .../ormlite/field/types/LocalDateSqlType.java | 23 +---------------- .../field/types/LocalDateTimeSqlType.java | 25 ++----------------- .../field/types/LocalDateTimeType.java | 12 +-------- .../ormlite/field/types/LocalDateType.java | 12 +-------- .../ormlite/field/types/LocalTimeSqlType.java | 23 +---------------- .../ormlite/field/types/LocalTimeType.java | 14 ++--------- .../field/types/OffsetDateTimeType.java | 14 ++--------- .../field/types/OffsetTimeSqlType.java | 25 ++----------------- .../ormlite/field/types/OffsetTimeType.java | 16 +++--------- .../field/types/LocalTimeTypeTest.java | 2 +- .../field/types/OffsetTimeTypeTest.java | 2 +- 13 files changed, 38 insertions(+), 163 deletions(-) create mode 100644 src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java diff --git a/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java b/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java new file mode 100644 index 000000000..4eb33d445 --- /dev/null +++ b/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java @@ -0,0 +1,19 @@ +package com.j256.ormlite.field.types; + +import com.j256.ormlite.field.SqlType; + +public abstract class BaseLocalDateType extends BaseDataType { + public BaseLocalDateType(SqlType sqlType, Class[] classes) { + super(sqlType, classes); + } + + @Override + public boolean isValidForVersion() { + return true; + } + + @Override + public boolean isArgumentHolderRequired() { + return true; + } +} diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java index 82bdec380..cd8b64f5b 100644 --- a/src/main/java/com/j256/ormlite/field/types/InstantType.java +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -17,7 +17,7 @@ * * @author graynk */ -public class InstantType extends BaseDataType { +public class InstantType extends BaseLocalDateType { private static InstantType singleton; public static InstantType getSingleton() { @@ -40,7 +40,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDateTime value: " + defaultStr, e); + " parsing default Instant value: " + defaultStr, e); } } @@ -62,11 +62,6 @@ public Object javaToSqlArg(FieldType fieldType, Object javaObject) { return OffsetDateTime.ofInstant(instant, ZoneId.of("UTC")); } - @Override - public boolean isValidForVersion() { - return true; - } - @Override public Object moveToNextValue(Object currentValue) { Instant datetime = (Instant) currentValue; @@ -77,9 +72,4 @@ public Object moveToNextValue(Object currentValue) { public boolean isValidForField(Field field) { return (field.getType() == Instant.class); } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java index a9799d459..3452f81dc 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -16,7 +16,7 @@ * * @author graynk */ -public class LocalDateSqlType extends BaseDataType { +public class LocalDateSqlType extends LocalDateType { private static LocalDateSqlType singleton; public static LocalDateSqlType getSingleton() { @@ -59,25 +59,4 @@ public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { Date date = (Date) sqlArg; return date.toLocalDate(); } - - @Override - public boolean isValidForVersion() { - return true; - } - - @Override - public Object moveToNextValue(Object currentValue) { - LocalDate date = (LocalDate) currentValue; - return date.plusDays(1); - } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } - - @Override - public boolean isValidForField(Field field) { - return (field.getType() == LocalDate.class); - } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java index 5c5167fef..17c885af9 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -16,7 +16,7 @@ * * @author graynk */ -public class LocalDateTimeSqlType extends BaseDataType { +public class LocalDateTimeSqlType extends LocalDateTimeType { private static LocalDateTimeSqlType singleton; public static LocalDateTimeSqlType getSingleton() { @@ -30,7 +30,7 @@ public static LocalDateTimeSqlType getSingleton() { } return singleton; } - private LocalDateTimeSqlType() { super(SqlType.DATE, new Class[] { LocalDateTime.class }); } + private LocalDateTimeSqlType() { super(SqlType.LOCAL_DATE_TIME, new Class[] { LocalDateTime.class }); } protected LocalDateTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override @@ -59,25 +59,4 @@ public Object javaToSqlArg(FieldType fieldType, Object javaObject) { LocalDateTime datetime = (LocalDateTime) javaObject; return Timestamp.valueOf(datetime); } - - @Override - public boolean isValidForVersion() { - return true; - } - - @Override - public Object moveToNextValue(Object currentValue) { - LocalDateTime date = (LocalDateTime) currentValue; - return date.plusNanos(1); - } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } - - @Override - public boolean isValidForField(Field field) { - return (field.getType() == LocalDateTime.class); - } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index 07f03dbb4..092a8adce 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -15,7 +15,7 @@ * * @author graynk */ -public class LocalDateTimeType extends BaseDataType { +public class LocalDateTimeType extends BaseLocalDateType { private static LocalDateTimeType singleton; public static LocalDateTimeType getSingleton() { @@ -47,11 +47,6 @@ public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int c return results.getLocalDateTime(columnPos); } - @Override - public boolean isValidForVersion() { - return true; - } - @Override public Object moveToNextValue(Object currentValue) { LocalDateTime datetime = (LocalDateTime) currentValue; @@ -62,9 +57,4 @@ public Object moveToNextValue(Object currentValue) { public boolean isValidForField(Field field) { return (field.getType() == LocalDateTime.class); } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java index 9e4742bcb..58aef8195 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java @@ -15,7 +15,7 @@ * * @author graynk */ -public class LocalDateType extends BaseDataType { +public class LocalDateType extends BaseLocalDateType { private static LocalDateType singleton; public static LocalDateType getSingleton() { @@ -47,22 +47,12 @@ public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int c return results.getLocalDate(columnPos); } - @Override - public boolean isValidForVersion() { - return true; - } - @Override public Object moveToNextValue(Object currentValue) { LocalDate date = (LocalDate) currentValue; return date.plusDays(1); } - @Override - public boolean isArgumentHolderRequired() { - return true; - } - @Override public boolean isValidForField(Field field) { return (field.getType() == LocalDate.class); diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java index 490c96926..419b86624 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -16,7 +16,7 @@ * * @author graynk */ -public class LocalTimeSqlType extends BaseDataType { +public class LocalTimeSqlType extends LocalTimeType { private static LocalTimeSqlType singleton; public static LocalTimeSqlType getSingleton() { @@ -59,25 +59,4 @@ public Object javaToSqlArg(FieldType fieldType, Object javaObject) { LocalTime time = (LocalTime) javaObject; return Time.valueOf(time); } - - @Override - public boolean isValidForVersion() { - return true; - } - - @Override - public Object moveToNextValue(Object currentValue) { - LocalTime time = (LocalTime) currentValue; - return time.plusNanos(1); - } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } - - @Override - public boolean isValidForField(Field field) { - return (field.getType() == LocalTime.class); - } } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index 13e557816..c0b2e2b4e 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -15,7 +15,7 @@ * * @author graynk */ -public class LocalTimeType extends BaseDataType { +public class LocalTimeType extends BaseLocalDateType { private static LocalTimeType singleton; public static LocalTimeType getSingleton() { @@ -38,7 +38,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDateTime value: " + defaultStr, e); + " parsing default LocalTime value: " + defaultStr, e); } } @@ -47,11 +47,6 @@ public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int c return results.getLocalTime(columnPos); } - @Override - public boolean isValidForVersion() { - return true; - } - @Override public Object moveToNextValue(Object currentValue) { LocalTime time = (LocalTime) currentValue; @@ -62,9 +57,4 @@ public Object moveToNextValue(Object currentValue) { public boolean isValidForField(Field field) { return (field.getType() == LocalTime.class); } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java index e0b4f4583..c5124943e 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -15,7 +15,7 @@ * * @author graynk */ -public class OffsetDateTimeType extends BaseDataType { +public class OffsetDateTimeType extends BaseLocalDateType { private static OffsetDateTimeType singleton; public static OffsetDateTimeType getSingleton() { @@ -38,7 +38,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDateTime value: " + defaultStr, e); + " parsing default OffsetDateTime value: " + defaultStr, e); } } @@ -47,11 +47,6 @@ public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int c return results.getOffsetDateTime(columnPos); } - @Override - public boolean isValidForVersion() { - return true; - } - @Override public Object moveToNextValue(Object currentValue) { OffsetDateTime datetime = (OffsetDateTime) currentValue; @@ -62,9 +57,4 @@ public Object moveToNextValue(Object currentValue) { public boolean isValidForField(Field field) { return (field.getType() == OffsetDateTime.class); } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java index ffc7857f6..2eed0e91c 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java @@ -17,7 +17,7 @@ * * @author graynk */ -public class OffsetTimeSqlType extends BaseDataType { +public class OffsetTimeSqlType extends OffsetTimeType { private static OffsetTimeSqlType singleton; public static OffsetTimeSqlType getSingleton() { @@ -40,7 +40,7 @@ public Object parseDefaultString(FieldType fieldType, String defaultStr) throws return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDateTime value: " + defaultStr, e); + " parsing default OffsetTime value: " + defaultStr, e); } } @@ -60,25 +60,4 @@ public Object javaToSqlArg(FieldType fieldType, Object javaObject) { OffsetTime time = (OffsetTime) javaObject; return time.atDate(LocalDate.ofEpochDay(0)); } - - @Override - public boolean isValidForVersion() { - return true; - } - - @Override - public Object moveToNextValue(Object currentValue) { - OffsetTime time = (OffsetTime) currentValue; - return time.plusNanos(1); - } - - @Override - public boolean isValidForField(Field field) { - return (field.getType() == OffsetTime.class); - } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index 0efd78f75..611a4ede6 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -15,7 +15,7 @@ * * @author graynk */ -public class OffsetTimeType extends BaseDataType { +public class OffsetTimeType extends BaseLocalDateType { private static OffsetTimeType singleton; public static OffsetTimeType getSingleton() { @@ -35,10 +35,10 @@ public static OffsetTimeType getSingleton() { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSS]x")); + return OffsetTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]x")); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDateTime value: " + defaultStr, e); + " parsing default OffsetTime value: " + defaultStr, e); } } @@ -47,11 +47,6 @@ public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int c return results.getOffsetTime(columnPos); } - @Override - public boolean isValidForVersion() { - return true; - } - @Override public Object moveToNextValue(Object currentValue) { OffsetTime time = (OffsetTime) currentValue; @@ -62,9 +57,4 @@ public Object moveToNextValue(Object currentValue) { public boolean isValidForField(Field field) { return (field.getType() == OffsetTime.class); } - - @Override - public boolean isArgumentHolderRequired() { - return true; - } } diff --git a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java index 760dec6c6..5cd8d4fb7 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java @@ -24,7 +24,7 @@ public void testTime() throws Exception { Class clazz = TimeTable.class; Dao dao = createDao(clazz, true); LocalTime val = LocalTime.now().truncatedTo(ChronoUnit.SECONDS); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss"); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss.SSSSSS"); String valStr = formatter.format(val); TimeTable foo = new TimeTable(); foo.time = val; diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java index 1f42f420a..35a1dd97c 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java @@ -23,7 +23,7 @@ public void testTime() throws Exception { Class clazz = TimeTable.class; Dao dao = createDao(clazz, true); OffsetTime val = OffsetTime.now(); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS]x"); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]x"); String valStr = formatter.format(val); TimeTable foo = new TimeTable(); foo.time = val; From f80d3e5187124a31e17ab9507f484cbc72a1486d Mon Sep 17 00:00:00 2001 From: graynk Date: Thu, 13 Dec 2018 11:29:28 +0500 Subject: [PATCH 09/12] Protected constructors for base class --- .../com/j256/ormlite/field/types/BaseLocalDateType.java | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java b/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java index 4eb33d445..27dde6f72 100644 --- a/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java @@ -3,10 +3,14 @@ import com.j256.ormlite.field.SqlType; public abstract class BaseLocalDateType extends BaseDataType { - public BaseLocalDateType(SqlType sqlType, Class[] classes) { + protected BaseLocalDateType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + protected BaseLocalDateType(SqlType sqlType) { + super(sqlType); + } + @Override public boolean isValidForVersion() { return true; From 04900ffc45f17ccf181a644f5ac6d11ec6a47637 Mon Sep 17 00:00:00 2001 From: graynk Date: Thu, 13 Dec 2018 12:29:59 +0500 Subject: [PATCH 10/12] Default persisters are now actually default --- .../java/com/j256/ormlite/field/DataType.java | 48 +++++++++---------- .../ormlite/field/types/LocalDateSqlType.java | 2 +- .../field/types/LocalDateTimeSqlType.java | 2 +- .../field/types/LocalDateTimeType.java | 1 + .../ormlite/field/types/LocalDateType.java | 1 + .../ormlite/field/types/LocalTimeSqlType.java | 2 +- .../ormlite/field/types/LocalTimeType.java | 1 + .../field/types/OffsetTimeSqlType.java | 2 +- .../ormlite/field/types/OffsetTimeType.java | 1 + 9 files changed, 32 insertions(+), 28 deletions(-) diff --git a/src/main/java/com/j256/ormlite/field/DataType.java b/src/main/java/com/j256/ormlite/field/DataType.java index 672678663..9b124f21e 100644 --- a/src/main/java/com/j256/ormlite/field/DataType.java +++ b/src/main/java/com/j256/ormlite/field/DataType.java @@ -259,30 +259,6 @@ public enum DataType { * Marker for fields that are unknown. */ UNKNOWN(null), - /** - * Persists the {@link java.time.LocalDate} Java class. By default this will use - * {@link #LOCAL_DATE} so you will need to specify this using {@link DatabaseField#dataType()}. - * - */ - LOCAL_DATE_SQL(LocalDateSqlType.getSingleton()), - /** - * Persists the {@link java.time.LocalTime} Java class. By default this will use - * {@link #LOCAL_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. - * - */ - LOCAL_TIME_SQL(LocalTimeSqlType.getSingleton()), - /** - * Persists the {@link java.time.LocalDateTime} Java class. By default this will use - * {@link #LOCAL_DATE_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. - * - */ - LOCAL_DATE_TIME_SQL(LocalDateTimeSqlType.getSingleton()), - /** - * Persists the {@link java.time.OffsetTime} Java class. By default this will use - * {@link #OFFSET_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. - * - */ - OFFSET_TIME_SQL(OffsetTimeSqlType.getSingleton()), /** * Persists the {@link java.time.LocalDate} Java class. * @@ -313,6 +289,30 @@ public enum DataType { * */ INSTANT(InstantType.getSingleton()), + /** + * Persists the {@link java.time.LocalDate} Java class. By default this will use + * {@link #LOCAL_DATE} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + LOCAL_DATE_SQL(LocalDateSqlType.getSingleton()), + /** + * Persists the {@link java.time.LocalTime} Java class. By default this will use + * {@link #LOCAL_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + LOCAL_TIME_SQL(LocalTimeSqlType.getSingleton()), + /** + * Persists the {@link java.time.LocalDateTime} Java class. By default this will use + * {@link #LOCAL_DATE_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + LOCAL_DATE_TIME_SQL(LocalDateTimeSqlType.getSingleton()), + /** + * Persists the {@link java.time.OffsetTime} Java class. By default this will use + * {@link #OFFSET_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. + * + */ + OFFSET_TIME_SQL(OffsetTimeSqlType.getSingleton()), // end ; diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java index 3452f81dc..d00567df5 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -30,7 +30,7 @@ public static LocalDateSqlType getSingleton() { } return singleton; } - private LocalDateSqlType() { super(SqlType.LOCAL_DATE, new Class[] { LocalDate.class }); } + private LocalDateSqlType() { super(SqlType.LOCAL_DATE); } protected LocalDateSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java index 17c885af9..9904715ca 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -30,7 +30,7 @@ public static LocalDateTimeSqlType getSingleton() { } return singleton; } - private LocalDateTimeSqlType() { super(SqlType.LOCAL_DATE_TIME, new Class[] { LocalDateTime.class }); } + private LocalDateTimeSqlType() { super(SqlType.LOCAL_DATE_TIME); } protected LocalDateTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index 092a8adce..3fb3f8ad3 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -31,6 +31,7 @@ public static LocalDateTimeType getSingleton() { } private LocalDateTimeType() { super(SqlType.LOCAL_DATE_TIME, new Class[] { LocalDateTime.class }); } protected LocalDateTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + protected LocalDateTimeType(SqlType sqlType) { super(sqlType); } @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java index 58aef8195..6f3aa1f63 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java @@ -30,6 +30,7 @@ public static LocalDateType getSingleton() { return singleton; } private LocalDateType() { super(SqlType.LOCAL_DATE, new Class[] { LocalDate.class }); } + protected LocalDateType(SqlType sqlType) { super(sqlType); } protected LocalDateType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java index 419b86624..41d75e5d3 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -30,7 +30,7 @@ public static LocalTimeSqlType getSingleton() { } return singleton; } - private LocalTimeSqlType() { super(SqlType.LOCAL_TIME, new Class[] { LocalTime.class }); } + private LocalTimeSqlType() { super(SqlType.LOCAL_TIME); } protected LocalTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index c0b2e2b4e..a1906eb6a 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -31,6 +31,7 @@ public static LocalTimeType getSingleton() { } private LocalTimeType() { super(SqlType.LOCAL_TIME, new Class[] { LocalTime.class }); } protected LocalTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + protected LocalTimeType(SqlType sqlType) { super(sqlType); } @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java index 2eed0e91c..cef8dd434 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java @@ -31,7 +31,7 @@ public static OffsetTimeSqlType getSingleton() { } return singleton; } - private OffsetTimeSqlType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { OffsetTime.class }); } + private OffsetTimeSqlType() { super(SqlType.OFFSET_DATE_TIME); } protected OffsetTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index 611a4ede6..f57df0676 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -31,6 +31,7 @@ public static OffsetTimeType getSingleton() { } private OffsetTimeType() { super(SqlType.OFFSET_TIME, new Class[] { OffsetTime.class }); } protected OffsetTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } + protected OffsetTimeType(SqlType sqlType) { super(sqlType); } @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { From ffb7fb9046a486408821717c96dfca09cfe7a4e9 Mon Sep 17 00:00:00 2001 From: graynk Date: Fri, 14 Dec 2018 10:31:51 +0500 Subject: [PATCH 11/12] Switched Class.forName calls to a single check for Java version Fixed javadocs, renamed OffsetTimeSql --- .../java/com/j256/ormlite/field/DataType.java | 4 +-- .../field/types/BaseLocalDateType.java | 12 +++++++++ .../j256/ormlite/field/types/InstantType.java | 18 ++++--------- .../ormlite/field/types/LocalDateSqlType.java | 17 +++--------- .../field/types/LocalDateTimeSqlType.java | 17 +++--------- .../field/types/LocalDateTimeType.java | 16 +++-------- .../ormlite/field/types/LocalDateType.java | 14 ++-------- .../ormlite/field/types/LocalTimeSqlType.java | 17 +++--------- .../ormlite/field/types/LocalTimeType.java | 16 +++-------- .../field/types/OffsetDateTimeType.java | 17 +++--------- ...SqlType.java => OffsetTimeCompatType.java} | 27 +++++++------------ .../ormlite/field/types/OffsetTimeType.java | 17 +++--------- .../field/types/LocalTimeTypeTest.java | 2 +- .../field/types/OffsetTimeSqlTypeTest.java | 10 +++---- .../com/j256/ormlite/h2/H2DatabaseType.java | 2 +- 15 files changed, 63 insertions(+), 143 deletions(-) rename src/main/java/com/j256/ormlite/field/types/{OffsetTimeSqlType.java => OffsetTimeCompatType.java} (64%) diff --git a/src/main/java/com/j256/ormlite/field/DataType.java b/src/main/java/com/j256/ormlite/field/DataType.java index 9b124f21e..bfba03396 100644 --- a/src/main/java/com/j256/ormlite/field/DataType.java +++ b/src/main/java/com/j256/ormlite/field/DataType.java @@ -50,7 +50,7 @@ import com.j256.ormlite.field.types.LocalDateSqlType; import com.j256.ormlite.field.types.LocalTimeSqlType; import com.j256.ormlite.field.types.LocalDateTimeSqlType; -import com.j256.ormlite.field.types.OffsetTimeSqlType; +import com.j256.ormlite.field.types.OffsetTimeCompatType; /** * Data type enumeration that corresponds to a {@link DataPersister}. @@ -312,7 +312,7 @@ public enum DataType { * {@link #OFFSET_TIME} so you will need to specify this using {@link DatabaseField#dataType()}. * */ - OFFSET_TIME_SQL(OffsetTimeSqlType.getSingleton()), + OFFSET_TIME_COMPAT(OffsetTimeCompatType.getSingleton()), // end ; diff --git a/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java b/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java index 27dde6f72..a9e6ec109 100644 --- a/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/BaseLocalDateType.java @@ -2,7 +2,15 @@ import com.j256.ormlite.field.SqlType; +/** + * Base class for all of the java.time class types. + * + * @author graynk + */ public abstract class BaseLocalDateType extends BaseDataType { + private static final String specificationVersion = System.getProperty("java.specification.version"); + private static final boolean javaTimeSupported = !(specificationVersion.equals("1.6") || specificationVersion.equals("1.7")); + protected BaseLocalDateType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @@ -11,6 +19,10 @@ protected BaseLocalDateType(SqlType sqlType) { super(sqlType); } + protected static boolean isJavaTimeSupported() { + return javaTimeSupported; + } + @Override public boolean isValidForVersion() { return true; diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java index cd8b64f5b..cd40d762c 100644 --- a/src/main/java/com/j256/ormlite/field/types/InstantType.java +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -13,24 +13,16 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * A custom persister that is able to store the java.time.Instant class in the database as Timestamp With Timezone object. + * This class does not have a SQL backup counter-part, the database should support JDBC 4.2 for it to be used. + * Instant is also not a part of JDBC specification, persister converts Instant to OffsetDateTime with timezone fixed at UTC * * @author graynk */ public class InstantType extends BaseLocalDateType { - private static InstantType singleton; - public static InstantType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.Instant", false, null); - singleton = new InstantType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final InstantType singleton = isJavaTimeSupported() ? new InstantType() : null; + public static InstantType getSingleton() { return singleton; } private InstantType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { Instant.class }); } protected InstantType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java index d00567df5..6b4ba06dd 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -1,6 +1,5 @@ package com.j256.ormlite.field.types; -import java.lang.reflect.Field; import java.sql.Date; import java.sql.SQLException; import java.time.LocalDate; @@ -13,23 +12,15 @@ /** * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * This class should be used only when database used does not support JDBC 4.2, since it converts java.time.LocalDate + * to java.sql.Date. * * @author graynk */ public class LocalDateSqlType extends LocalDateType { - private static LocalDateSqlType singleton; - public static LocalDateSqlType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.LocalDate", false, null); - singleton = new LocalDateSqlType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static LocalDateSqlType singleton = isJavaTimeSupported() ? new LocalDateSqlType() : null; + public static LocalDateSqlType getSingleton() { return singleton; } private LocalDateSqlType() { super(SqlType.LOCAL_DATE); } protected LocalDateSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java index 9904715ca..b2b70aa5f 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -1,6 +1,5 @@ package com.j256.ormlite.field.types; -import java.lang.reflect.Field; import java.sql.SQLException; import java.sql.Timestamp; import java.time.LocalDateTime; @@ -13,23 +12,15 @@ /** * A custom persister that is able to store the java.time.LocalDateTime class in the database as Timestamp object. + * This class should be used only when database used does not support JDBC 4.2, since it converts java.time.LocalDateTime + * to java.sql.Timestamp. * * @author graynk */ public class LocalDateTimeSqlType extends LocalDateTimeType { - private static LocalDateTimeSqlType singleton; - public static LocalDateTimeSqlType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.LocalDateTime", false, null); - singleton = new LocalDateTimeSqlType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final LocalDateTimeSqlType singleton = isJavaTimeSupported() ? new LocalDateTimeSqlType() : null; + public static LocalDateTimeSqlType getSingleton() { return singleton; } private LocalDateTimeSqlType() { super(SqlType.LOCAL_DATE_TIME); } protected LocalDateTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index 3fb3f8ad3..67797dde2 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -11,24 +11,14 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * A custom persister that is able to store the java.time.LocalDateTime class in the database as Timestamp object. * * @author graynk */ public class LocalDateTimeType extends BaseLocalDateType { - private static LocalDateTimeType singleton; - public static LocalDateTimeType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.LocalDateTime", false, null); - singleton = new LocalDateTimeType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final LocalDateTimeType singleton = isJavaTimeSupported() ? new LocalDateTimeType() : null; + public static LocalDateTimeType getSingleton() { return singleton; } private LocalDateTimeType() { super(SqlType.LOCAL_DATE_TIME, new Class[] { LocalDateTime.class }); } protected LocalDateTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } protected LocalDateTimeType(SqlType sqlType) { super(sqlType); } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java index 6f3aa1f63..55afe816f 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateType.java @@ -17,18 +17,8 @@ */ public class LocalDateType extends BaseLocalDateType { - private static LocalDateType singleton; - public static LocalDateType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.LocalDate", false, null); - singleton = new LocalDateType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final LocalDateType singleton = isJavaTimeSupported() ? new LocalDateType() : null; + public static LocalDateType getSingleton() { return singleton; } private LocalDateType() { super(SqlType.LOCAL_DATE, new Class[] { LocalDate.class }); } protected LocalDateType(SqlType sqlType) { super(sqlType); } protected LocalDateType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java index 41d75e5d3..8bc3c2f12 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -1,6 +1,5 @@ package com.j256.ormlite.field.types; -import java.lang.reflect.Field; import java.sql.SQLException; import java.sql.Time; import java.time.LocalTime; @@ -13,23 +12,15 @@ /** * A custom persister that is able to store the java.time.LocalTime class in the database as Time object. + * This class should be used only when database used does not support JDBC 4.2, since it converts java.time.LocalTime + * to java.sql.Time. * * @author graynk */ public class LocalTimeSqlType extends LocalTimeType { - private static LocalTimeSqlType singleton; - public static LocalTimeSqlType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.LocalTime", false, null); - singleton = new LocalTimeSqlType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final LocalTimeSqlType singleton = isJavaTimeSupported() ? new LocalTimeSqlType() : null; + public static LocalTimeSqlType getSingleton() { return singleton; } private LocalTimeSqlType() { super(SqlType.LOCAL_TIME); } protected LocalTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index a1906eb6a..c2caa834b 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -11,24 +11,14 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * A custom persister that is able to store the java.time.LocalTime class in the database as Time object. * * @author graynk */ public class LocalTimeType extends BaseLocalDateType { - private static LocalTimeType singleton; - public static LocalTimeType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.LocalTime", false, null); - singleton = new LocalTimeType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final LocalTimeType singleton = isJavaTimeSupported() ? new LocalTimeType() : null; + public static LocalTimeType getSingleton() { return singleton; } private LocalTimeType() { super(SqlType.LOCAL_TIME, new Class[] { LocalTime.class }); } protected LocalTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } protected LocalTimeType(SqlType sqlType) { super(sqlType); } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java index c5124943e..dcc8b406f 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -11,24 +11,15 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * A custom persister that is able to store the java.time.OffsetDateTime class in the database as Timestamp with Time Zone object. + * This class does not have a SQL backup counter-part, the database should support JDBC 4.2 for it to be used. * * @author graynk */ public class OffsetDateTimeType extends BaseLocalDateType { - private static OffsetDateTimeType singleton; - public static OffsetDateTimeType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.OffsetDateTime", false, null); - singleton = new OffsetDateTimeType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final OffsetDateTimeType singleton = isJavaTimeSupported() ? new OffsetDateTimeType() : null; + public static OffsetDateTimeType getSingleton() { return singleton; } private OffsetDateTimeType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { OffsetDateTime.class }); } protected OffsetDateTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java similarity index 64% rename from src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java rename to src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java index cef8dd434..1e25edf81 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java @@ -1,6 +1,5 @@ package com.j256.ormlite.field.types; -import java.lang.reflect.Field; import java.sql.SQLException; import java.time.LocalDate; import java.time.OffsetDateTime; @@ -13,26 +12,18 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * A custom persister that is able to store the java.time.OffsetTime class in the database as Timestamp With Time Zone object. + * This class should be used only when database used does not support Time With Time Zone, since it converts java.time.OffsetTime + * to java.time.OffsetDateTime, fixing date part at epoch, to be stored as Timestamp With Time Zone. * * @author graynk */ -public class OffsetTimeSqlType extends OffsetTimeType { - - private static OffsetTimeSqlType singleton; - public static OffsetTimeSqlType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.OffsetTime", false, null); - singleton = new OffsetTimeSqlType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } - private OffsetTimeSqlType() { super(SqlType.OFFSET_DATE_TIME); } - protected OffsetTimeSqlType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } +public class OffsetTimeCompatType extends OffsetTimeType { + + private static final OffsetTimeCompatType singleton = isJavaTimeSupported() ? new OffsetTimeCompatType() : null; + public static OffsetTimeCompatType getSingleton() { return singleton; } + private OffsetTimeCompatType() { super(SqlType.OFFSET_DATE_TIME); } + protected OffsetTimeCompatType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index f57df0676..07563aee6 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -11,24 +11,15 @@ import com.j256.ormlite.support.DatabaseResults; /** - * A custom persister that is able to store the java.time.LocalDate class in the database as Date object. + * A custom persister that is able to store the java.time.OffsetTime class in the database as Time With Time Zone object. + * This class does not have a SQL backup counter-part, the database should support JDBC 4.2 for it to be used. * * @author graynk */ public class OffsetTimeType extends BaseLocalDateType { - private static OffsetTimeType singleton; - public static OffsetTimeType getSingleton() { - if (singleton == null) { - try { - Class.forName("java.time.OffsetTime", false, null); - singleton = new OffsetTimeType(); - } catch (ClassNotFoundException e) { - return null; // No java.time on classpath; - } - } - return singleton; - } + private static final OffsetTimeType singleton = isJavaTimeSupported() ? new OffsetTimeType() : null; + public static OffsetTimeType getSingleton() { return singleton; } private OffsetTimeType() { super(SqlType.OFFSET_TIME, new Class[] { OffsetTime.class }); } protected OffsetTimeType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } protected OffsetTimeType(SqlType sqlType) { super(sqlType); } diff --git a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java index 5cd8d4fb7..760dec6c6 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalTimeTypeTest.java @@ -24,7 +24,7 @@ public void testTime() throws Exception { Class clazz = TimeTable.class; Dao dao = createDao(clazz, true); LocalTime val = LocalTime.now().truncatedTo(ChronoUnit.SECONDS); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss.SSSSSS"); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss"); String valStr = formatter.format(val); TimeTable foo = new TimeTable(); foo.time = val; diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java index 921e107aa..c4acd9fae 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java @@ -32,7 +32,7 @@ public void testTime() throws Exception { foo.time = val; assertEquals(1, dao.create(foo)); - testType(dao, foo, clazz, val, val2, val2, valStr, DataType.OFFSET_TIME_SQL, TIME_COLUMN, false, + testType(dao, foo, clazz, val, val2, val2, valStr, DataType.OFFSET_TIME_COMPAT, TIME_COLUMN, false, true, true, false, true, false, true, false); } @@ -43,7 +43,7 @@ public void testDateNull() throws Exception { Dao dao = createDao(clazz, true); TimeTable foo = new TimeTable(); assertEquals(1, dao.create(foo)); - testType(dao, foo, clazz, null, null, null, null, DataType.OFFSET_TIME_SQL, TIME_COLUMN, + testType(dao, foo, clazz, null, null, null, null, DataType.OFFSET_TIME_COMPAT, TIME_COLUMN, false, true, true, false, true, false, true, false); } @@ -52,7 +52,7 @@ public void testDateNull() throws Exception { public void testDateParseInvalid() throws Exception { FieldType fieldType = FieldType.createFieldType(connectionSource, TABLE_NAME, TimeTable.class.getDeclaredField(TIME_COLUMN), TimeTable.class); - DataType.OFFSET_TIME_SQL.getDataPersister().parseDefaultString(fieldType, "not valid time string"); + DataType.OFFSET_TIME_COMPAT.getDataPersister().parseDefaultString(fieldType, "not valid time string"); } @Test(expected = IllegalArgumentException.class) @@ -65,13 +65,13 @@ public void testInvalidDateField() throws Exception { @DatabaseTable protected static class InvalidDate { - @DatabaseField(dataType = DataType.OFFSET_TIME_SQL) + @DatabaseField(dataType = DataType.OFFSET_TIME_COMPAT) String notTime; } @DatabaseTable(tableName = TABLE_NAME) protected static class TimeTable { - @DatabaseField(columnName = TIME_COLUMN, dataType = DataType.OFFSET_TIME_SQL) + @DatabaseField(columnName = TIME_COLUMN, dataType = DataType.OFFSET_TIME_COMPAT) OffsetTime time; } } diff --git a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java index 451970b11..08ff280d7 100644 --- a/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java +++ b/src/test/java/com/j256/ormlite/h2/H2DatabaseType.java @@ -80,7 +80,7 @@ public boolean isCreateIfNotExistsSupported() { public FieldConverter getFieldConverter(DataPersister dataPersister, FieldType fieldType) { // H2 doesn't support TIME WITH TIME ZONE if (dataPersister.getSqlType() == SqlType.OFFSET_TIME) - return DataType.OFFSET_TIME_SQL.getDataPersister(); + return DataType.OFFSET_TIME_COMPAT.getDataPersister(); // default is to use the dataPersister itself return dataPersister; } From 9cabc3524fd09c189bb83d93031028d8a6d14152 Mon Sep 17 00:00:00 2001 From: graynk Date: Tue, 19 Mar 2019 14:19:14 +0500 Subject: [PATCH 12/12] Test fixes by Bo98 --- .../j256/ormlite/field/types/InstantType.java | 21 +------------------ .../ormlite/field/types/LocalDateSqlType.java | 9 +------- .../field/types/LocalDateTimeSqlType.java | 9 +------- .../field/types/LocalDateTimeType.java | 8 +++++-- .../ormlite/field/types/LocalTimeSqlType.java | 7 +------ .../ormlite/field/types/LocalTimeType.java | 14 ++++++------- .../field/types/OffsetDateTimeType.java | 9 ++++++-- .../field/types/OffsetTimeCompatType.java | 10 +++++++-- .../ormlite/field/types/OffsetTimeType.java | 9 ++++++-- .../ormlite/field/types/InstantTypeTest.java | 3 ++- .../field/types/LocalDateTimeTypeTest.java | 3 ++- ...est.java => OffsetTimeCompatTypeTest.java} | 5 +++-- .../field/types/OffsetTimeTypeTest.java | 5 +++-- 13 files changed, 48 insertions(+), 64 deletions(-) rename src/test/java/com/j256/ormlite/field/types/{OffsetTimeSqlTypeTest.java => OffsetTimeCompatTypeTest.java} (94%) diff --git a/src/main/java/com/j256/ormlite/field/types/InstantType.java b/src/main/java/com/j256/ormlite/field/types/InstantType.java index cd40d762c..3fa2767ef 100644 --- a/src/main/java/com/j256/ormlite/field/types/InstantType.java +++ b/src/main/java/com/j256/ormlite/field/types/InstantType.java @@ -1,16 +1,12 @@ package com.j256.ormlite.field.types; import java.lang.reflect.Field; -import java.sql.SQLException; import java.time.Instant; import java.time.OffsetDateTime; import java.time.ZoneId; -import java.time.format.DateTimeFormatter; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; -import com.j256.ormlite.misc.SqlExceptionUtil; -import com.j256.ormlite.support.DatabaseResults; /** * A custom persister that is able to store the java.time.Instant class in the database as Timestamp With Timezone object. @@ -19,28 +15,13 @@ * * @author graynk */ -public class InstantType extends BaseLocalDateType { +public class InstantType extends OffsetDateTimeType { private static final InstantType singleton = isJavaTimeSupported() ? new InstantType() : null; public static InstantType getSingleton() { return singleton; } private InstantType() { super(SqlType.OFFSET_DATE_TIME, new Class[] { Instant.class }); } protected InstantType(SqlType sqlType, Class[] classes) { super(sqlType, classes); } - @Override - public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { - try { - return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); - } catch (NumberFormatException e) { - throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default Instant value: " + defaultStr, e); - } - } - - @Override - public Object resultToSqlArg(FieldType fieldType, DatabaseResults results, int columnPos) throws SQLException { - return results.getOffsetDateTime(columnPos); - } - @Override public Object sqlArgToJava(FieldType fieldType, Object sqlArg, int columnPos) { OffsetDateTime value = (OffsetDateTime) sqlArg; diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java index 6b4ba06dd..7a9c39fb9 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateSqlType.java @@ -3,11 +3,9 @@ import java.sql.Date; import java.sql.SQLException; import java.time.LocalDate; -import java.time.format.DateTimeFormatter; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; -import com.j256.ormlite.misc.SqlExceptionUtil; import com.j256.ormlite.support.DatabaseResults; /** @@ -26,12 +24,7 @@ public class LocalDateSqlType extends LocalDateType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { - try { - return Date.valueOf(LocalDate.parse(defaultStr, DateTimeFormatter.ISO_LOCAL_DATE)); - } catch (NumberFormatException e) { - throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDate value: " + defaultStr, e); - } + return Date.valueOf((LocalDate) super.parseDefaultString(fieldType, defaultStr)); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java index b2b70aa5f..903c00307 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeSqlType.java @@ -3,11 +3,9 @@ import java.sql.SQLException; import java.sql.Timestamp; import java.time.LocalDateTime; -import java.time.format.DateTimeFormatter; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; -import com.j256.ormlite.misc.SqlExceptionUtil; import com.j256.ormlite.support.DatabaseResults; /** @@ -26,12 +24,7 @@ public class LocalDateTimeSqlType extends LocalDateTimeType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { - try { - return Timestamp.valueOf(LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]"))); - } catch (NumberFormatException e) { - throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalDateTime value: " + defaultStr, e); - } + return Timestamp.valueOf((LocalDateTime) super.parseDefaultString(fieldType, defaultStr)); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java index 67797dde2..bb9c52f71 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalDateTimeType.java @@ -3,7 +3,8 @@ import java.lang.reflect.Field; import java.sql.SQLException; import java.time.LocalDateTime; -import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeFormatterBuilder; +import java.time.temporal.ChronoField; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; @@ -26,7 +27,10 @@ public class LocalDateTimeType extends BaseLocalDateType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return LocalDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]")); + return LocalDateTime.parse(defaultStr, new DateTimeFormatterBuilder() + .appendPattern("yyyy-MM-dd HH:mm:ss") + .appendFraction(ChronoField.NANO_OF_SECOND, 0, 9, true) + .toFormatter()); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default LocalDateTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java index 8bc3c2f12..33b1cc507 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeSqlType.java @@ -26,12 +26,7 @@ public class LocalTimeSqlType extends LocalTimeType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { - try { - return Time.valueOf(LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]"))); - } catch (NumberFormatException e) { - throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalTime value: " + defaultStr, e); - } + return Time.valueOf((LocalTime) super.parseDefaultString(fieldType, defaultStr)); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java index c2caa834b..8f17d8630 100644 --- a/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/LocalTimeType.java @@ -3,11 +3,11 @@ import java.lang.reflect.Field; import java.sql.SQLException; import java.time.LocalTime; -import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeFormatterBuilder; +import java.time.temporal.ChronoField; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; -import com.j256.ormlite.misc.SqlExceptionUtil; import com.j256.ormlite.support.DatabaseResults; /** @@ -25,12 +25,10 @@ public class LocalTimeType extends BaseLocalDateType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { - try { - return LocalTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]")); - } catch (NumberFormatException e) { - throw SqlExceptionUtil.create("Problems with field " + fieldType + - " parsing default LocalTime value: " + defaultStr, e); - } + return LocalTime.parse(defaultStr, new DateTimeFormatterBuilder() + .appendPattern("HH:mm:ss") + .appendFraction(ChronoField.NANO_OF_SECOND, 0, 9, true) + .toFormatter()); } @Override diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java index dcc8b406f..28d05384a 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetDateTimeType.java @@ -3,7 +3,8 @@ import java.lang.reflect.Field; import java.sql.SQLException; import java.time.OffsetDateTime; -import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeFormatterBuilder; +import java.time.temporal.ChronoField; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; @@ -26,7 +27,11 @@ public class OffsetDateTimeType extends BaseLocalDateType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); + return OffsetDateTime.parse(defaultStr, new DateTimeFormatterBuilder() + .appendPattern("yyyy-MM-dd HH:mm:ss") + .appendFraction(ChronoField.NANO_OF_SECOND, 0, 9, true) + .appendPattern("x") + .toFormatter()); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default OffsetDateTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java index 1e25edf81..5bc8bfa49 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeCompatType.java @@ -4,7 +4,8 @@ import java.time.LocalDate; import java.time.OffsetDateTime; import java.time.OffsetTime; -import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeFormatterBuilder; +import java.time.temporal.ChronoField; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; @@ -28,7 +29,12 @@ public class OffsetTimeCompatType extends OffsetTimeType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetDateTime.parse(defaultStr, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x")); + OffsetTime time = OffsetTime.parse(defaultStr, new DateTimeFormatterBuilder() + .appendPattern("[yyyy-MM-dd ]HH:mm:ss") + .appendFraction(ChronoField.NANO_OF_SECOND, 0, 9, true) + .appendPattern("x") + .toFormatter()); + return javaToSqlArg(fieldType, time); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default OffsetTime value: " + defaultStr, e); diff --git a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java index 07563aee6..57ec726ca 100644 --- a/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java +++ b/src/main/java/com/j256/ormlite/field/types/OffsetTimeType.java @@ -3,7 +3,8 @@ import java.lang.reflect.Field; import java.sql.SQLException; import java.time.OffsetTime; -import java.time.format.DateTimeFormatter; +import java.time.format.DateTimeFormatterBuilder; +import java.time.temporal.ChronoField; import com.j256.ormlite.field.FieldType; import com.j256.ormlite.field.SqlType; @@ -27,7 +28,11 @@ public class OffsetTimeType extends BaseLocalDateType { @Override public Object parseDefaultString(FieldType fieldType, String defaultStr) throws SQLException { try { - return OffsetTime.parse(defaultStr, DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]x")); + return OffsetTime.parse(defaultStr, new DateTimeFormatterBuilder() + .appendPattern("HH:mm:ss") + .appendFraction(ChronoField.NANO_OF_SECOND, 0, 9, true) + .appendPattern("x") + .toFormatter()); } catch (NumberFormatException e) { throw SqlExceptionUtil.create("Problems with field " + fieldType + " parsing default OffsetTime value: " + defaultStr, e); diff --git a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java index 92740bd56..414494f07 100644 --- a/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/InstantTypeTest.java @@ -7,6 +7,7 @@ import java.time.ZoneOffset; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; import org.junit.Test; @@ -24,7 +25,7 @@ public class InstantTypeTest extends BaseTypeTest { public void testInstant() throws Exception { Class clazz = InstantTable.class; Dao dao = createDao(clazz, true); - Instant val = Instant.now(); + Instant val = Instant.now().truncatedTo(ChronoUnit.MILLIS); DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x"); OffsetDateTime val2 = OffsetDateTime.ofInstant(val, ZoneOffset.UTC); String valStr = formatter.format(val2); diff --git a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java index 75badfe04..19244ac51 100644 --- a/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/LocalDateTimeTypeTest.java @@ -5,6 +5,7 @@ import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; import org.junit.Test; @@ -22,7 +23,7 @@ public class LocalDateTimeTypeTest extends BaseTypeTest { public void testDateTime() throws Exception { Class clazz = DateTimeTable.class; Dao dao = createDao(clazz, true); - LocalDateTime val = LocalDateTime.now(); + LocalDateTime val = LocalDateTime.now().truncatedTo(ChronoUnit.MILLIS); DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS"); String valStr = formatter.format(val); DateTimeTable foo = new DateTimeTable(); diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeCompatTypeTest.java similarity index 94% rename from src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java rename to src/test/java/com/j256/ormlite/field/types/OffsetTimeCompatTypeTest.java index c4acd9fae..8a8dac7f1 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeSqlTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeCompatTypeTest.java @@ -7,6 +7,7 @@ import java.time.OffsetTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; import org.junit.Test; @@ -16,7 +17,7 @@ import com.j256.ormlite.field.FieldType; import com.j256.ormlite.table.DatabaseTable; -public class OffsetTimeSqlTypeTest extends BaseTypeTest { +public class OffsetTimeCompatTypeTest extends BaseTypeTest { private static final String TIME_COLUMN = "time"; @@ -24,7 +25,7 @@ public class OffsetTimeSqlTypeTest extends BaseTypeTest { public void testTime() throws Exception { Class clazz = TimeTable.class; Dao dao = createDao(clazz, true); - OffsetTime val = OffsetTime.now(); + OffsetTime val = OffsetTime.now().truncatedTo(ChronoUnit.MILLIS); OffsetDateTime val2 = val.atDate(LocalDate.ofEpochDay(0)); DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss[.SSS]x"); String valStr = formatter.format(val2); diff --git a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java index 35a1dd97c..0bfc6032a 100644 --- a/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java +++ b/src/test/java/com/j256/ormlite/field/types/OffsetTimeTypeTest.java @@ -5,6 +5,7 @@ import java.time.OffsetTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeParseException; +import java.time.temporal.ChronoUnit; import org.junit.Test; @@ -22,8 +23,8 @@ public class OffsetTimeTypeTest extends BaseTypeTest { public void testTime() throws Exception { Class clazz = TimeTable.class; Dao dao = createDao(clazz, true); - OffsetTime val = OffsetTime.now(); - DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSSSSS]x"); + OffsetTime val = OffsetTime.now().truncatedTo(ChronoUnit.MILLIS); + DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS]x"); String valStr = formatter.format(val); TimeTable foo = new TimeTable(); foo.time = val;