THRIFT-4944 Field IDs > 255 fail with compact protocol
Cliwent: Delphi
Patch: Jens Geyer
diff --git a/lib/delphi/src/Thrift.Protocol.Compact.pas b/lib/delphi/src/Thrift.Protocol.Compact.pas
index 1c1b3da..07cab9a 100644
--- a/lib/delphi/src/Thrift.Protocol.Compact.pas
+++ b/lib/delphi/src/Thrift.Protocol.Compact.pas
@@ -686,7 +686,8 @@
// Read a field header off the wire.
function TCompactProtocolImpl.ReadFieldBegin: TThriftField;
var type_ : Byte;
- fieldId, modifier : ShortInt;
+ modifier : ShortInt;
+ fieldId : SmallInt;
begin
type_ := Byte( ReadByte);
@@ -700,7 +701,7 @@
modifier := ShortInt( (type_ and $F0) shr 4);
if (modifier = 0)
then fieldId := ReadI16 // not a delta. look ahead for the zigzag varint field id.
- else fieldId := ShortInt( lastFieldId_ + modifier); // add the delta to the last Read field id.
+ else fieldId := SmallInt( lastFieldId_ + modifier); // add the delta to the last Read field id.
Init( result, '', getTType(Byte(type_ and $0F)), fieldId);
diff --git a/lib/delphi/test/serializer/TestSerializer.Data.pas b/lib/delphi/test/serializer/TestSerializer.Data.pas
index 5fc0070..2420e9a 100644
--- a/lib/delphi/test/serializer/TestSerializer.Data.pas
+++ b/lib/delphi/test/serializer/TestSerializer.Data.pas
@@ -336,6 +336,10 @@
result.Byte_set_map := TDebugProtoTestConstants.COMPACT_TEST.Byte_set_map;
result.Byte_list_map := TDebugProtoTestConstants.COMPACT_TEST.Byte_list_map;
+ result.Field500 := 500;
+ result.Field5000 := 5000;
+ result.Field20000 := 20000;
+
{$IF cDebugProtoTest_Option_AnsiStr_Binary}
result.A_binary := AnsiString( #0#1#2#3#4#5#6#7#8);
{$ELSE}
diff --git a/lib/delphi/test/serializer/TestSerializer.dpr b/lib/delphi/test/serializer/TestSerializer.dpr
index 1f5ae8b..56d0d15 100644
--- a/lib/delphi/test/serializer/TestSerializer.dpr
+++ b/lib/delphi/test/serializer/TestSerializer.dpr
@@ -29,6 +29,7 @@
Thrift.Transport in '..\..\src\Thrift.Transport.pas',
Thrift.Protocol in '..\..\src\Thrift.Protocol.pas',
Thrift.Protocol.JSON in '..\..\src\Thrift.Protocol.JSON.pas',
+ Thrift.Protocol.Compact in '..\..\src\Thrift.Protocol.Compact.pas',
Thrift.Collections in '..\..\src\Thrift.Collections.pas',
Thrift.Server in '..\..\src\Thrift.Server.pas',
Thrift.Utils in '..\..\src\Thrift.Utils.pas',
@@ -44,6 +45,12 @@
type
TTestSerializer = class //extends TestCase {
+ private type
+ TMethod = (
+ mt_Bytes,
+ mt_Stream
+ );
+
private
FProtocols : TList< IProtocolFactory>;
@@ -53,6 +60,8 @@
class procedure Deserialize( const input : TStream; const target : IBase; const factory : IProtocolFactory); overload;
procedure Test_Serializer_Deserializer;
+ procedure Test_OneOfEach( const method : TMethod; const factory : IProtocolFactory; const stream : TFileStream);
+ procedure Test_CompactStruct( const method : TMethod; const factory : IProtocolFactory; const stream : TFileStream);
public
constructor Create;
@@ -70,7 +79,7 @@
inherited Create;
FProtocols := TList< IProtocolFactory>.Create;
FProtocols.Add( TBinaryProtocolImpl.TFactory.Create);
- //FProtocols.Add( TCompactProtocolImpl.TFactory.Create);
+ FProtocols.Add( TCompactProtocolImpl.TFactory.Create);
FProtocols.Add( TJSONProtocolImpl.TFactory.Create);
end;
@@ -84,73 +93,115 @@
end;
end;
-type TMethod = (mt_Bytes, mt_Stream);
+
+procedure TTestSerializer.Test_OneOfEach( const method : TMethod; const factory : IProtocolFactory; const stream : TFileStream);
+var tested, correct : IOneOfEach;
+ bytes : TBytes;
+ i : Integer;
+begin
+ // write
+ tested := Fixtures.CreateOneOfEach;
+ case method of
+ mt_Bytes: bytes := Serialize( tested, factory);
+ mt_Stream: begin
+ stream.Size := 0;
+ Serialize( tested, factory, stream);
+ end
+ else
+ ASSERT( FALSE);
+ end;
+
+ // init + read
+ tested := TOneOfEachImpl.Create;
+ case method of
+ mt_Bytes: Deserialize( bytes, tested, factory);
+ mt_Stream: begin
+ stream.Position := 0;
+ Deserialize( stream, tested, factory);
+ end
+ else
+ ASSERT( FALSE);
+ end;
+
+ // check
+ correct := Fixtures.CreateOneOfEach;
+ ASSERT( tested.Im_true = correct.Im_true);
+ ASSERT( tested.Im_false = correct.Im_false);
+ ASSERT( tested.A_bite = correct.A_bite);
+ ASSERT( tested.Integer16 = correct.Integer16);
+ ASSERT( tested.Integer32 = correct.Integer32);
+ ASSERT( tested.Integer64 = correct.Integer64);
+ ASSERT( Abs( tested.Double_precision - correct.Double_precision) < 1E-12);
+ ASSERT( tested.Some_characters = correct.Some_characters);
+ ASSERT( tested.Zomg_unicode = correct.Zomg_unicode);
+ ASSERT( tested.What_who = correct.What_who);
+
+ ASSERT( Length(tested.Base64) = Length(correct.Base64));
+ ASSERT( CompareMem( @tested.Base64[0], @correct.Base64[0], Length(correct.Base64)));
+
+ ASSERT( tested.Byte_list.Count = correct.Byte_list.Count);
+ for i := 0 to tested.Byte_list.Count-1
+ do ASSERT( tested.Byte_list[i] = correct.Byte_list[i]);
+
+ ASSERT( tested.I16_list.Count = correct.I16_list.Count);
+ for i := 0 to tested.I16_list.Count-1
+ do ASSERT( tested.I16_list[i] = correct.I16_list[i]);
+
+ ASSERT( tested.I64_list.Count = correct.I64_list.Count);
+ for i := 0 to tested.I64_list.Count-1
+ do ASSERT( tested.I64_list[i] = correct.I64_list[i]);
+end;
+
+
+procedure TTestSerializer.Test_CompactStruct( const method : TMethod; const factory : IProtocolFactory; const stream : TFileStream);
+var tested, correct : ICompactProtoTestStruct;
+ bytes : TBytes;
+begin
+ // write
+ tested := Fixtures.CreateCompactProtoTestStruct;
+ case method of
+ mt_Bytes: bytes := Serialize( tested, factory);
+ mt_Stream: begin
+ stream.Size := 0;
+ Serialize( tested, factory, stream);
+ end
+ else
+ ASSERT( FALSE);
+ end;
+
+ // init + read
+ correct := TCompactProtoTestStructImpl.Create;
+ case method of
+ mt_Bytes: Deserialize( bytes, tested, factory);
+ mt_Stream: begin
+ stream.Position := 0;
+ Deserialize( stream, tested, factory);
+ end
+ else
+ ASSERT( FALSE);
+ end;
+
+ // check
+ correct := Fixtures.CreateCompactProtoTestStruct;
+ ASSERT( correct.Field500 = tested.Field500);
+ ASSERT( correct.Field5000 = tested.Field5000);
+ ASSERT( correct.Field20000 = tested.Field20000);
+end;
procedure TTestSerializer.Test_Serializer_Deserializer;
-var level3ooe, correct : IOneOfEach;
- factory : IProtocolFactory;
- bytes : TBytes;
+var factory : IProtocolFactory;
stream : TFileStream;
- i : Integer;
method : TMethod;
begin
- correct := Fixtures.CreateOneOfEach;
stream := TFileStream.Create( 'TestSerializer.dat', fmCreate);
try
for method in [Low(TMethod)..High(TMethod)] do begin
for factory in FProtocols do begin
- // write
- level3ooe := Fixtures.CreateOneOfEach;
- case method of
- mt_Bytes: bytes := Serialize( level3ooe, factory);
- mt_Stream: begin
- stream.Size := 0;
- Serialize( level3ooe, factory, stream);
- end
- else
- ASSERT( FALSE);
- end;
-
- // init + read
- level3ooe := TOneOfEachImpl.Create;
- case method of
- mt_Bytes: Deserialize( bytes, level3ooe, factory);
- mt_Stream: begin
- stream.Position := 0;
- Deserialize( stream, level3ooe, factory);
- end
- else
- ASSERT( FALSE);
- end;
-
-
- // check
- ASSERT( level3ooe.Im_true = correct.Im_true);
- ASSERT( level3ooe.Im_false = correct.Im_false);
- ASSERT( level3ooe.A_bite = correct.A_bite);
- ASSERT( level3ooe.Integer16 = correct.Integer16);
- ASSERT( level3ooe.Integer32 = correct.Integer32);
- ASSERT( level3ooe.Integer64 = correct.Integer64);
- ASSERT( Abs( level3ooe.Double_precision - correct.Double_precision) < 1E-12);
- ASSERT( level3ooe.Some_characters = correct.Some_characters);
- ASSERT( level3ooe.Zomg_unicode = correct.Zomg_unicode);
- ASSERT( level3ooe.What_who = correct.What_who);
- ASSERT( level3ooe.Base64 = correct.Base64);
-
- ASSERT( level3ooe.Byte_list.Count = correct.Byte_list.Count);
- for i := 0 to level3ooe.Byte_list.Count-1
- do ASSERT( level3ooe.Byte_list[i] = correct.Byte_list[i]);
-
- ASSERT( level3ooe.I16_list.Count = correct.I16_list.Count);
- for i := 0 to level3ooe.I16_list.Count-1
- do ASSERT( level3ooe.I16_list[i] = correct.I16_list[i]);
-
- ASSERT( level3ooe.I64_list.Count = correct.I64_list.Count);
- for i := 0 to level3ooe.I64_list.Count-1
- do ASSERT( level3ooe.I64_list[i] = correct.I64_list[i]);
+ Test_OneOfEach( method, factory, stream);
+ Test_CompactStruct( method, factory, stream);
end;
end;