-
Notifications
You must be signed in to change notification settings - Fork 571
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
SerializerUtils support OffsetDateTime and Instant #2116
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -23,10 +23,7 @@ | |
import java.net.Inet4Address; | ||
import java.net.Inet6Address; | ||
import java.sql.Timestamp; | ||
import java.time.LocalDate; | ||
import java.time.LocalDateTime; | ||
import java.time.ZoneId; | ||
import java.time.ZonedDateTime; | ||
import java.time.*; | ||
import java.util.Arrays; | ||
import java.util.HashMap; | ||
import java.util.HashSet; | ||
|
@@ -639,16 +636,22 @@ public static void writeDateTime32(OutputStream output, Object value, ZoneId tar | |
} | ||
|
||
public static void writeDateTime(OutputStream output, Object value, ZoneId targetTz) throws IOException { | ||
long ts = 0; | ||
long ts; | ||
if (value instanceof LocalDateTime) { | ||
LocalDateTime dt = (LocalDateTime) value; | ||
ts = dt.atZone(targetTz).toEpochSecond(); | ||
} else if (value instanceof ZonedDateTime) { | ||
ZonedDateTime dt = (ZonedDateTime) value; | ||
ts = dt.withZoneSameInstant(targetTz).toEpochSecond(); | ||
ts = dt.toEpochSecond(); | ||
} else if (value instanceof Timestamp) { | ||
Timestamp t = (Timestamp) value; | ||
ts = t.toLocalDateTime().atZone(targetTz).toEpochSecond(); | ||
} else if(value instanceof OffsetDateTime) { | ||
OffsetDateTime dt = (OffsetDateTime) value; | ||
ts = dt.toEpochSecond(); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think we should convert to a target timezone. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hi @chernser . This is the same reason as above Epoch seconds itself implies that the time zone is UTC, so the epoch seconds calculated in any time zone are always the same. |
||
} else if (value instanceof Instant) { | ||
Instant dt = (Instant) value; | ||
ts = dt.getEpochSecond(); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. what about timezone? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hi @chernser . Instant now = Instant.now(); // now --> 2025-01-31T11:21:58.887640603Z
ZonedDateTime zdt = now.atZone(ZoneId.systemDefault()); // zdt --> 2025-01-31T19:21:58.887640603+08:00[Asia/Shanghai]
// Yes, correct datetime in my time zone There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So to summary, as long as the datetime type contains the time zone information (like This is why so many ORM frameworks (Like mybatis, hibernate) recommends using |
||
} else { | ||
throw new IllegalArgumentException("Cannot convert " + value + " to DataTime"); | ||
} | ||
|
@@ -661,20 +664,30 @@ public static void writeDateTime64(OutputStream output, Object value, int scale, | |
throw new IllegalArgumentException("Invalid scale value '" + scale + "'"); | ||
} | ||
|
||
long ts = 0; | ||
long nano = 0; | ||
long ts; | ||
long nano; | ||
if (value instanceof LocalDateTime) { | ||
ZonedDateTime dt = ((LocalDateTime) value).atZone(targetTz); | ||
ts = dt.toEpochSecond(); | ||
nano = dt.getNano(); | ||
LocalDateTime dt = (LocalDateTime) value; | ||
ZonedDateTime zdt = dt.atZone(targetTz); | ||
ts = zdt.toEpochSecond(); | ||
nano = zdt.getNano(); | ||
} else if (value instanceof ZonedDateTime) { | ||
ZonedDateTime dt = ((ZonedDateTime) value).withZoneSameInstant(targetTz); | ||
ZonedDateTime dt = (ZonedDateTime) value; | ||
ts = dt.toEpochSecond(); | ||
nano = dt.getNano(); | ||
} else if (value instanceof Timestamp) { | ||
ZonedDateTime dt = ((Timestamp) value).toLocalDateTime().atZone(targetTz); | ||
Timestamp dt = (Timestamp) value; | ||
ZonedDateTime zdt = dt.toLocalDateTime().atZone(targetTz); | ||
ts = zdt.toEpochSecond(); | ||
nano = zdt.getNano(); | ||
} else if (value instanceof OffsetDateTime) { | ||
OffsetDateTime dt = (OffsetDateTime) value; | ||
ts = dt.toEpochSecond(); | ||
nano = dt.getNano(); | ||
} else if (value instanceof Instant) { | ||
Instant dt = (Instant) value; | ||
ts = dt.getEpochSecond(); | ||
nano = dt.getNano(); | ||
} else { | ||
throw new IllegalArgumentException("Cannot convert " + value + " to DataTime"); | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
seems missing
.withZoneSameInstante(targetTz)
?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @chernser
withZoneSameInstant
is useless, see comments inwithZoneSameInstante
:So this function change the
time zone
only, not change the real datetime.We don't really need to use this method, because
ZonedDateTime
itself contains thetime zone
, so using it to convert to epoch time is always in line with expectationsA simple test: