HHH-13266 Test OffsetDateTime serialization, in particular around 1900-01-01

This commit is contained in:
Yoann Rodière 2019-03-01 13:03:00 +01:00 committed by gbadner
parent c409c3305f
commit 08bb8e149f
1 changed files with 240 additions and 106 deletions

View File

@ -6,160 +6,294 @@
*/ */
package org.hibernate.test.type; package org.hibernate.test.type;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.Timestamp;
import java.time.LocalDateTime;
import java.time.OffsetDateTime; import java.time.OffsetDateTime;
import java.time.ZoneId;
import java.time.ZoneOffset; import java.time.ZoneOffset;
import java.util.Arrays;
import java.util.GregorianCalendar;
import java.util.List; import java.util.List;
import java.util.TimeZone;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.function.Consumer;
import javax.persistence.Basic;
import javax.persistence.Column; import javax.persistence.Column;
import javax.persistence.Entity; import javax.persistence.Entity;
import javax.persistence.Id; import javax.persistence.Id;
import javax.persistence.Table; import javax.persistence.Table;
import org.hibernate.Query; import org.hibernate.Query;
import org.hibernate.Session; import org.hibernate.dialect.Dialect;
import org.hibernate.resource.transaction.spi.TransactionStatus; import org.hibernate.dialect.PostgreSQL81Dialect;
import org.hibernate.type.OffsetDateTimeType; import org.hibernate.type.OffsetDateTimeType;
import org.hibernate.testing.TestForIssue; import org.hibernate.testing.TestForIssue;
import org.hibernate.testing.junit4.BaseNonConfigCoreFunctionalTestCase; import org.hibernate.testing.junit4.BaseNonConfigCoreFunctionalTestCase;
import org.hibernate.testing.junit4.CustomParameterized;
import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import static org.hamcrest.core.Is.is; import static org.hamcrest.core.Is.is;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertThat; import static org.junit.Assert.assertThat;
import static org.junit.Assert.fail; import static org.junit.Assert.assertTrue;
/** /**
* @author Andrea Boriero * @author Andrea Boriero
*/ */
@RunWith(CustomParameterized.class)
@TestForIssue(jiraKey = "HHH-10372") @TestForIssue(jiraKey = "HHH-10372")
public class OffsetDateTimeTest extends BaseNonConfigCoreFunctionalTestCase { public class OffsetDateTimeTest extends BaseNonConfigCoreFunctionalTestCase {
private static Dialect DIALECT;
private static Dialect determineDialect() {
try {
return Dialect.getDialect();
}
catch (Exception e) {
return new Dialect() {
};
}
}
/*
* The default timezone affects conversions done using java.util,
* which is why we take it into account even when testing OffsetDateTime.
*/
@Parameterized.Parameters(name = "{0}-{1}-{2}T{3}:{4}:{5}.{6}[{7}] [JVM TZ: {8}]")
public static List<Object[]> data() {
DIALECT = determineDialect();
return Arrays.asList(
// Not affected by HHH-13266
data( 2017, 11, 6, 19, 19, 1, 0, "+10:00", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+07:00", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+01:30", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+01:00", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+00:30", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "-02:00", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "-06:00", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "-08:00", ZoneId.of( "UTC-8" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+10:00", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+07:00", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+01:30", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 500, "+01:00", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+01:00", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "+00:30", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "-02:00", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "-06:00", ZoneId.of( "Europe/Paris" ) ),
data( 2017, 11, 6, 19, 19, 1, 0, "-08:00", ZoneId.of( "Europe/Paris" ) ),
data( 1970, 1, 1, 0, 0, 0, 0, "+01:00", ZoneId.of( "GMT" ) ),
data( 1970, 1, 1, 0, 0, 0, 0, "+00:00", ZoneId.of( "GMT" ) ),
data( 1970, 1, 1, 0, 0, 0, 0, "-01:00", ZoneId.of( "GMT" ) ),
data( 1900, 1, 1, 0, 0, 0, 0, "+01:00", ZoneId.of( "GMT" ) ),
data( 1900, 1, 1, 0, 0, 0, 0, "+00:00", ZoneId.of( "GMT" ) ),
data( 1900, 1, 1, 0, 0, 0, 0, "-01:00", ZoneId.of( "GMT" ) ),
data( 1900, 1, 1, 0, 0, 0, 0, "+00:00", ZoneId.of( "Europe/Oslo" ) ),
data( 1900, 1, 1, 0, 9, 21, 0, "+00:09:21", ZoneId.of( "Europe/Paris" ) ),
data( 1900, 1, 1, 0, 19, 32, 0, "+00:19:32", ZoneId.of( "Europe/Paris" ) ),
data( 1900, 1, 1, 0, 19, 32, 0, "+00:19:32", ZoneId.of( "Europe/Amsterdam" ) ),
// Affected by HHH-13266
data( 1892, 1, 1, 0, 0, 0, 0, "+00:00", ZoneId.of( "Europe/Oslo" ) ),
data( 1900, 1, 1, 0, 9, 20, 0, "+00:09:21", ZoneId.of( "Europe/Paris" ) ),
data( 1900, 1, 1, 0, 19, 31, 0, "+00:19:32", ZoneId.of( "Europe/Paris" ) ),
data( 1900, 1, 1, 0, 19, 31, 0, "+00:19:32", ZoneId.of( "Europe/Amsterdam" ) ),
data( 1600, 1, 1, 0, 0, 0, 0, "+00:19:32", ZoneId.of( "Europe/Amsterdam" ) )
);
}
private static Object[] data(int year, int month, int day,
int hour, int minute, int second, int nanosecond, String offset, ZoneId defaultTimeZone) {
if ( DIALECT instanceof PostgreSQL81Dialect ) {
// PostgreSQL apparently doesn't support nanosecond precision correctly
nanosecond = 0;
}
return new Object[] { year, month, day, hour, minute, second, nanosecond, offset, defaultTimeZone };
}
private final int year;
private final int month;
private final int day;
private final int hour;
private final int minute;
private final int second;
private final int nanosecond;
private final String offset;
private final ZoneId defaultTimeZone;
public OffsetDateTimeTest(int year, int month, int day,
int hour, int minute, int second, int nanosecond, String offset, ZoneId defaultTimeZone) {
this.year = year;
this.month = month;
this.day = day;
this.hour = hour;
this.minute = minute;
this.second = second;
this.nanosecond = nanosecond;
this.offset = offset;
this.defaultTimeZone = defaultTimeZone;
}
private OffsetDateTime getOriginalOffsetDateTime() {
return OffsetDateTime.of( year, month, day, hour, minute, second, nanosecond, ZoneOffset.of( offset ) );
}
private OffsetDateTime getExpectedOffsetDateTime() {
return getOriginalOffsetDateTime().atZoneSameInstant( ZoneId.systemDefault() ).toOffsetDateTime();
}
private Timestamp getExpectedTimestamp() {
LocalDateTime dateTimeInDefaultTimeZone = getOriginalOffsetDateTime().atZoneSameInstant( ZoneId.systemDefault() )
.toLocalDateTime();
return new Timestamp(
dateTimeInDefaultTimeZone.getYear() - 1900, dateTimeInDefaultTimeZone.getMonthValue() - 1,
dateTimeInDefaultTimeZone.getDayOfMonth(),
dateTimeInDefaultTimeZone.getHour(), dateTimeInDefaultTimeZone.getMinute(),
dateTimeInDefaultTimeZone.getSecond(),
dateTimeInDefaultTimeZone.getNano()
);
}
@Override @Override
protected Class[] getAnnotatedClasses() { protected Class<?>[] getAnnotatedClasses() {
return new Class[] {OffsetDateTimeEvent.class}; return new Class[] { EntityWithOffsetDateTime.class };
}
@Before
public void cleanup() {
inTransaction( session -> {
session.createNativeQuery( "DELETE FROM theentity" ).executeUpdate();
} );
} }
@Test @Test
public void testOffsetDateTimeWithHoursZoneOffset() { @TestForIssue(jiraKey = "HHH-13266")
final OffsetDateTime expectedStartDate = OffsetDateTime.of( public void writeThenRead() {
2015, withDefaultTimeZone( defaultTimeZone, () -> {
1, inTransaction( session -> {
1, session.persist( new EntityWithOffsetDateTime( 1, getOriginalOffsetDateTime() ) );
0, } );
0, inTransaction( session -> {
0, OffsetDateTime read = session.find( org.hibernate.test.type.OffsetDateTimeTest.EntityWithOffsetDateTime.class, 1 ).value;
0, assertEquals(
ZoneOffset.ofHours( 5 ) "Writing then reading a value should return the original value",
); getExpectedOffsetDateTime(), read
);
saveOffsetDateTimeEventWithStartDate( expectedStartDate ); assertTrue(
getExpectedOffsetDateTime().isEqual( read )
checkSavedOffsetDateTimeIsEqual( expectedStartDate ); );
compareSavedOffsetDateTimeWith( expectedStartDate ); assertEquals(
0,
OffsetDateTimeType.INSTANCE.getComparator().compare( getExpectedOffsetDateTime(), read )
);
} );
} );
} }
@Test @Test
public void testOffsetDateTimeWithUTCZoneOffset() { @TestForIssue(jiraKey = "HHH-13266")
final OffsetDateTime expectedStartDate = OffsetDateTime.of( public void writeThenNativeRead() {
1, withDefaultTimeZone( defaultTimeZone, () -> {
1, inTransaction( session -> {
1, session.persist( new EntityWithOffsetDateTime( 1, getOriginalOffsetDateTime() ) );
0, } );
0, inTransaction( session -> {
0, session.doWork( connection -> {
0, final PreparedStatement statement = connection.prepareStatement(
ZoneOffset.UTC "SELECT thevalue FROM theentity WHERE theid = ?"
); );
statement.setInt( 1, 1 );
saveOffsetDateTimeEventWithStartDate( expectedStartDate ); statement.execute();
final ResultSet resultSet = statement.getResultSet();
checkSavedOffsetDateTimeIsEqual( expectedStartDate ); resultSet.next();
compareSavedOffsetDateTimeWith( expectedStartDate ); Timestamp nativeRead = resultSet.getTimestamp( 1 );
assertEquals(
"Raw values written in database should match the original value (same instant)",
getExpectedTimestamp(),
nativeRead
);
} );
} );
} );
} }
@Test @Test
public void testRetrievingEntityByOffsetDateTime() { public void testRetrievingEntityByOffsetDateTime() {
withDefaultTimeZone( defaultTimeZone, () -> {
final OffsetDateTime startDate = OffsetDateTime.of( inTransaction( session -> {
1, session.persist( new EntityWithOffsetDateTime( 1, getOriginalOffsetDateTime() ) );
1, } );
1, Consumer<OffsetDateTime> checkOneMatch = expected -> inSession( s -> {
0, Query query = s.createQuery( "from EntityWithOffsetDateTime o where o.value = :date" );
0, query.setParameter( "date", expected, OffsetDateTimeType.INSTANCE );
0, List<EntityWithOffsetDateTime> list = query.list();
0, assertThat( list.size(), is( 1 ) );
ZoneOffset.ofHours( 3 ) } );
); checkOneMatch.accept( getOriginalOffsetDateTime() );
checkOneMatch.accept( getExpectedOffsetDateTime() );
saveOffsetDateTimeEventWithStartDate( startDate ); checkOneMatch.accept( getExpectedOffsetDateTime().withOffsetSameInstant( ZoneOffset.UTC ) );
} );
final Session s = openSession();
try {
Query query = s.createQuery( "from OffsetDateTimeEvent o where o.startDate = :date" );
query.setParameter( "date", startDate, OffsetDateTimeType.INSTANCE );
List<OffsetDateTimeEvent> list = query.list();
assertThat( list.size(), is( 1 ) );
}
finally {
s.close();
}
} }
private void checkSavedOffsetDateTimeIsEqual(OffsetDateTime startdate) { private static void withDefaultTimeZone(ZoneId zoneId, Runnable runnable) {
final Session s = openSession(); TimeZone timeZoneBefore = TimeZone.getDefault();
TimeZone.setDefault( TimeZone.getTimeZone( zoneId ) );
/*
* Run the code in a new thread, because some libraries (looking at you, h2 JDBC driver)
* cache data dependent on the default timezone in thread local variables,
* and we want this data to be reinitialized with the new default time zone.
*/
try { try {
final OffsetDateTimeEvent offsetDateEvent = s.get( OffsetDateTimeEvent.class, 1L ); ExecutorService executor = Executors.newSingleThreadExecutor();
assertThat( offsetDateEvent.startDate.isEqual( startdate ), is( true ) ); Future<?> future = executor.submit( runnable );
executor.shutdown();
future.get();
} }
finally { catch (InterruptedException e) {
s.close(); throw new IllegalStateException( "Interrupted while testing", e );
} }
} catch (ExecutionException e) {
Throwable cause = e.getCause();
private void compareSavedOffsetDateTimeWith(OffsetDateTime startdate) { if ( cause instanceof RuntimeException ) {
final Session s = openSession(); throw (RuntimeException) cause;
try { }
final OffsetDateTimeEvent offsetDateEvent = s.get( OffsetDateTimeEvent.class, 1L ); else if ( cause instanceof Error ) {
assertThat( throw (Error) cause;
OffsetDateTimeType.INSTANCE.getComparator().compare( offsetDateEvent.startDate, startdate ), }
is( 0 ) else {
); throw new IllegalStateException( "Unexpected exception while testing", cause );
}
finally {
s.close();
}
}
private void saveOffsetDateTimeEventWithStartDate(OffsetDateTime startdate) {
final OffsetDateTimeEvent event = new OffsetDateTimeEvent();
event.id = 1L;
event.startDate = startdate;
final Session s = openSession();
s.getTransaction().begin();
try {
s.save( event );
s.getTransaction().commit();
}
catch (Exception e) {
if ( s.getTransaction() != null && s.getTransaction().getStatus() == TransactionStatus.ACTIVE ) {
s.getTransaction().rollback();
} }
fail( e.getMessage() );
} }
finally { finally {
s.close(); TimeZone.setDefault( timeZoneBefore );
} }
} }
@Entity(name = "OffsetDateTimeEvent") @Entity(name = "EntityWithOffsetDateTime")
@Table(name = "OFFSET_DATE_TIME_EVENT") @Table(name = "theentity")
public static class OffsetDateTimeEvent { private static final class EntityWithOffsetDateTime {
@Id @Id
private Long id; @Column(name = "theid")
private Integer id;
@Column(name = "START_DATE") @Basic
private OffsetDateTime startDate; @Column(name = "thevalue")
private OffsetDateTime value;
protected EntityWithOffsetDateTime() {
}
private EntityWithOffsetDateTime(int id, OffsetDateTime value) {
this.id = id;
this.value = value;
}
} }
@Override @Override