Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Algorithmen, Datenstrukturen und Klassendesign (https://www.delphipraxis.net/78-algorithmen-datenstrukturen-und-klassendesign/)
-   -   TJSONMarshal / TJSONUnMarshal böse Falle (https://www.delphipraxis.net/193981-tjsonmarshal-tjsonunmarshal-boese-falle.html)

Rollo62 1. Okt 2017 20:40


TJSONMarshal / TJSONUnMarshal böse Falle
 
Ich frage mich seit geraumen warum eine App plötzlich Timestamps falsch berechnet.
Der Grund war das Encoding / Decoding mit TJSONMarshal / TJSONUnMarshal.

Das hat bis dato gut funktioniert, Object zu String, und zurück, aber seit womöglich 10.1 Berlin gab es da plötzlich Probleme.
Normalerweise debugge ich nicht tief in die System Sourcen, aber hier musste es mal wieder sein :(

Der Grund war das TJSONMarshal / TJSONUnMarshal anscheinend neuerdings unterschiedliche IsDateTimeUTC settings per Default eingestellt haben.
Kann jetzt nicht genau sagen seit wann ...

Die Lösung war das explizit mit anzugeben, um die Konvertierungen kompatibel zu machen, also
Delphi-Quellcode:
LMar.DateTimeIsUTC := True;
für Marshal/Unmarshal:

Delphi-Quellcode:
function TParcel.InternalMarshal_ToString : String;
var
  LMar : TJSONMarshal;
begin

  LMar := TJSONMarshal.Create();   // This doesn't correctly set DateTimeUTC by default

  try
      LMar.DateTimeIsUTC := True; // !!! Important to ensure same Coding/Decoding

      Result := LMar.Marshal(Self).ToString; // as TJSonObject;

  finally
      LMar.Free;
  end;

end;



function TParcel.InternalUnmarshal_FromObject(value: TJSONObject): TParcel;

var
  LMar: TJSONUnMarshal;

begin
    Result := nil;

    if not Assigned( value ) then
        Exit;

    LMar := TJSONUnMarshal.Create(); // This sets DateTimeIsUTC internally by default

    try
        LMar.DateTimeIsUTC := True; // !!! Important to ensure same Coding/Decoding

        ...
        ...
        ...


    finally
        LMar.Free;
    end;
end;

Wenn es nicht gleich gesetzt wird kann es zu einer Zeitdifferenz dei Encode/Decode kommen von 2:00h.
Das ist definitiv nicht das was ich erwarten würde :stupid:

Rollo

Der schöne Günther 2. Okt 2017 07:39

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Traue niemals der Delphi-RTL, schreibe Unit-Tests! :warn:

Zumindest in 10 Seattle ergibt ein
Delphi-Quellcode:
EncodeDateTime(1988, 10, 21, 17, 45, 39, 999)
im JSON noch ein
Delphi-Quellcode:
'1988-10-21T17:45:39.999Z'
.

Uwe Raabe 2. Okt 2017 09:18

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Kannst du dafür noch einen Eintrag in QP erstellen?

Der schöne Günther 2. Okt 2017 09:36

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Ich verstehe das Problem noch nicht. Ein TDateTime ist lokale Zeit, nicht UTC. Von TObject nach JSON nehmen wir kein UTC an, von Json nach TObject schon. Ich denke, das ist doch so richtig, oder?

Delphi-Quellcode:
program UTCMarshalTest;

{$APPTYPE CONSOLE}
{$R *.res}

uses
  System.SysUtils,
  System.DateUtils,
  System.Types,
  System.Json,
  REST.JsonReflect,
  Unit1 in 'Unit1.pas';

procedure p();
var
   marshal: TJsonMarshal;
   unmarshal: TJSONUnMarshal;
   myObject: TMyObject;
   timestamp: TDateTime;
   myObject2: TMyObject;
begin
   marshal := nil; unmarshal := nil;
   myObject := nil; myObject2 := nil;
   try
      marshal := TJSONMarshal.Create();
      unmarshal := TJSONUnMarshal.Create();

      myObject := TMyObject.Create();
      timestamp := EncodeDateTime(2017, 10, 02, 10, 28, 44, 123);
      myObject.FTimeStamp := timestamp;

      myObject2 := unmarshal.CreateObject(
         TMyObject,
         marshal.Marshal(myObject) as TJsonObject
      ) as TMyObject;

      Assert(
         CompareDateTime(myObject.FTimeStamp, myObject2.FTimeStamp)
         =
         System.Types.EqualsValue
      );
   finally
      marshal.Free(); unmarshal.Free();
      myObject.Free(); myObject2.Free();
   end;
end;

begin
   try
      p();
   except
      on E: Exception do
         Writeln(E.ClassName, ': ', E.Message);
   end;
   WriteLn(sLineBreak, 'end.');
   ReadLn;
end.

Uwe Raabe 2. Okt 2017 09:50

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Das Problem ist, daß dein Code unter Tokyo eine EAssertionFailed-Exception auslöst.

Der schöne Günther 2. Okt 2017 09:55

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Oh, ok. Unter Seattle läuft alles korrekt durch. Ich dachte er sei verwirrt dass
Delphi-Quellcode:
marshal
und
Delphi-Quellcode:
unmarshal
nicht die gleiche Datums/UTC-Eigenschaften hatten.

Rollo62 2. Okt 2017 09:57

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Hallo Uwe,

Zitat:

Kannst du dafür noch einen Eintrag in QP erstellen?
Bin mir nicht sicher ob das jetzt ein Bug ist oder nicht.
Man muss es halt "vollständig" konfigurieren.

Rollo

Rollo62 2. Okt 2017 10:04

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Hallo Günther,

Zitat:

ch verstehe das Problem noch nicht. Ein TDateTime ist lokale Zeit, nicht UTC. Von TObject nach JSON nehmen wir kein UTC an, von Json nach TObject schon. Ich denke, das ist doch so richtig, oder?
Das Problem ist ich ein Objekt zu JSON mit Marshal encodiere , und
dann dieses JSON wieder zu einem Objekt mit Unmarshal dekodiere.

Wenn da im Objekt ein TDateTime Feld drin ist wird mit Marshal eine UTC Korrektur gemacht
und dann bei Unmarshal aber nicht (oder umgekehrt, habs jetzt gerade nicht nachgesehen).
Standardmässig wird das ISOEncode benutzt, welches UTC berücksichtigt.

Das Ergebnis-Objekt hat dann danach zum Ausgangs-Objekt je nach Zeitzone eine Differenz von z.B. +2:00.

Es muss im Marshal/Umarshal DateTimeIsUTC = True/False; expliziert gesetzt werden das ich mit oder ohne Korrektur arbeiten möchte, jedenfalls das es bei Beiden gleich sein sollte.

Rollo

Uwe Raabe 2. Okt 2017 10:20

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Zitat:

Zitat von Rollo62 (Beitrag 1382445)
Bin mir nicht sicher ob das jetzt ein Bug ist oder nicht.
Man muss es halt "vollständig" konfigurieren.

Das Verhalten unter Tokyo ist zumindest ein Anderes als unter Seattle und Berlin. Das ist in jedem Fall einen QP-Eintrag wert (Regression). Der Fehler kommt durch einen Bugfix in
Delphi-Quellcode:
TISODateTimeInterceptor
zustande, der eine entsprechende Anpassung in
Delphi-Quellcode:
TJSONMarshal.Create
erfordert. Insofern ist dein Anwendungsfall ein wichtiger Testcase.

Berlin:
Delphi-Quellcode:
constructor TISODateTimeInterceptor.Create(ADateTimeIsUTC: Boolean);
begin
  ConverterType := ctString;
  ReverterType := rtString;
  FDateTimeIsUTC := true;
end;
Tokyo:
Delphi-Quellcode:
constructor TISODateTimeInterceptor.Create(ADateTimeIsUTC: Boolean);
begin
  ConverterType := ctString;
  ReverterType := rtString;
  FDateTimeIsUTC := ADateTimeIsUTC;
end;

Rollo62 2. Okt 2017 17:02

AW: TJSONMarshal / TJSONUnMarshal böse Falle
 
Hallo Uwe,

hier der QP-Eintrag, hab mal ne freie Minute gefunden.

Rollo


Alle Zeitangaben in WEZ +1. Es ist jetzt 20:12 Uhr.
Seite 1 von 2  1 2      

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz