Search code examples
javajsonjacksonjodatime

Jackson loses time offset from dates when deserializing to JodaTime


I'm trying to serialize and then deserialize a Joda DateTime object using Jackson, but it doesn't deserialize the object fully. It looks like timezone information is lost.

This code:

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());
mapper.configure(com.fasterxml.jackson.databind.SerializationFeature.WRITE_DATES_AS_TIMESTAMPS , false);

DateTime dt = DateTime.now();
String j = mapper.writeValueAsString(dt);
DateTime dt2 = mapper.readValue(j, DateTime.class);

System.out.println("json: " + j);
System.out.println("eq? " + (dt.equals(dt2)));
System.out.println("dates:\n" + dt + "\n" + dt2);

outputs this:

json: "2013-10-18T14:10:52.458-07:00"
eq? false
dates:
2013-10-18T14:10:52.458-07:00
2013-10-18T21:10:52.458Z

Is this by design? Is there anything I can do here, short of writing my own serializer/deserializer? I've seen a few questions about this on SO, but none that deal with this aspect specifically.

I'm using Joda 2.1 and Jackson 2.1


Solution

  • Yes, this is by design. JodaTime DateTimeSerializer use standard toString() method. According to JodaTime official guide toString() returns - the standard ISO8601 string for the DateTime. Also, standard DateTimeDeserializer always creates UTC datetimes.

    To store TimeZone you need to store it separately with same json and use .withZone() method after deserialization or just create serializer and deserializer.

    UPDATE

    Version 2.2.3 have a bit extended behaviour - DateTimeDeserializer creates DateTime with timeZone taken from DeserializationContext. it may be changed with ObjectMapper.setTimeZone(). Default is TimeZone.getTimeZone("GMT")