Flutter parsing json with DateTime from Golang RFC3339: FormatException: Invalid date format

5,684

Solution 1

Because the documentation doesn't cover this sufficiently it took me a day of researching the flutter sources and trial & error of different things to solve it. So may as well share it.

Golang by default encodes the time.Time in RFC3339 when serializing to Json (like in the given example). Flutter explicitly supports RFC3339, so why doesn't it work? The answer is a small difference in how the seconds fraction part is supported. While Golang produces a precision of 7 digits Dart only supports up to 6 digits and does not gracefully handle violations. So if the example is corrected to only have 6 digits of precision it will parse just fine in Dart:

{
    ...
    "dateCreated": "2018-09-29T19:51:57.413978-07:00",
    ...
}

In order to solve this in a generic way you have two options: 1. to truncate the additional precision from the string, or 2. implement your own parsing. Let's assume we extend the DateTime class and create your own CustomDateTime. The new class has the parse method overridden to remove all excess after 6 digits before handing it to the parent class' parse method.

Now we can use the CustomDateTime in our Dart classes. For example:

@JsonSerializable()
class MyObj {

  CustomDateTime dateCreated;

  MyObj( this.dateCreated);

  factory MyObj.fromJson(Map<String, dynamic> json) => _$MyObjFromJson(json);  
  Map<String, dynamic> toJson() => _$MyObjToJson(this); 
}

But of course now the code generation is broken and we get the following error:

Error running JsonSerializableGenerator
Could not generate 'toJson' code for 'dateCreated'.
None of the provided 'TypeHelper' instances support the defined type.

Luckily the json_annotation package now has an easy solution for us - The JsonConverter. Here is how to use it in our example:

First define a converter that explains to the code generator how to convert our CustomDateTime type:

class CustomDateTimeConverter implements JsonConverter<CustomDateTime, String> {
  const CustomDateTimeConverter();

  @override
  CustomDateTime fromJson(String json) =>
      json == null ? null : CustomDateTime.parse(json);

  @override
  String toJson(CustomDateTime object) => object.toIso8601String();
}

Second we just annotate this converter to every class that is using our CustomDateTime data type:

@JsonSerializable()
@CustomDateTimeConverter()
class MyObj {

  CustomDateTime dateCreated;

  MyObj( this.dateCreated);

  factory MyObj.fromJson(Map<String, dynamic> json) => _$MyObjFromJson(json);  
  Map<String, dynamic> toJson() => _$MyObjToJson(this); 
}

This satisfies the code generator and Voila! We can read json with RFC3339 timestamps that come from golang time.Time.

Solution 2

I havd the same problem. I found a very simple solution. We can use a custom converter with JsonConverter. For more explanation you can use my article.

import 'package:json_annotation/json_annotation.dart';

class CustomDateTimeConverter implements JsonConverter<DateTime, String> {
  const CustomDateTimeConverter();

  @override
  DateTime fromJson(String json) {
    if (json.contains(".")) {
      json = json.substring(0, json.length - 1);
    }

    return DateTime.parse(json);
  }

  @override
  String toJson(DateTime json) => json.toIso8601String();
}

import 'package:json_annotation/json_annotation.dart';
import 'package:my_app/shared/helpers/custom_datetime.dart';

part 'publication_document.g.dart';

@JsonSerializable()
@CustomDateTimeConverter()
class PublicationDocument {
  final int id;
  final int publicationId;

  final DateTime publicationDate;
  final DateTime createTime;
  final DateTime updateTime;
  final bool isFree;

  PublicationDocument({
    this.id,
    this.publicationId,
    this.publicationDate,
    this.createTime,
    this.updateTime,
    this.isFree,
  });

  factory PublicationDocument.fromJson(Map<String, dynamic> json) =>
      _$PublicationDocumentFromJson(json);
  Map<String, dynamic> toJson() => _$PublicationDocumentToJson(this);
}
Share:
5,684
Oswin Noetzelmann
Author by

Oswin Noetzelmann

Updated on December 07, 2022

Comments

  • Oswin Noetzelmann
    Oswin Noetzelmann over 1 year

    When trying to read json files generated with golangs json package in Dart / Flutter I noticed that parsing dates produce an error:

    FormatException: Invalid date format
    

    An example is the following json generated on the Go server:

    {
        ...
        "dateCreated": "2018-09-29T19:51:57.4139787-07:00",
        ...
    }
    

    I am using the code generation approach for json (de-)serialization to avoid writing all the boiler plate code. The json_serializable package is a standard package available for this purpose. So my code looks like the following:

    @JsonSerializable()
    class MyObj {
    
      DateTime dateCreated;
    
      MyObj( this.dateCreated);
    
      factory MyObj.fromJson(Map<String, dynamic> json) => _$MyObjFromJson(json);  
      Map<String, dynamic> toJson() => _$MyObjToJson(this); 
    }
    
  • mezoni
    mezoni almost 5 years
    But how you suggest to deal with timezones?
  • Oswin Noetzelmann
    Oswin Noetzelmann over 4 years
    @mezoni - RFC3339 typically include the timezone (notice the -07:00 at the end of the example in my answer "2018-09-29T19:51:57.413978-07:00"). However if they don't contain this timezone offset, you would assume GMT timezone (unless you know better because of your applications' details).