| (* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| *) |
| |
| {$SCOPEDENUMS ON} |
| |
| unit Thrift.Protocol; |
| |
| interface |
| |
| uses |
| Classes, |
| SysUtils, |
| Contnrs, |
| Thrift.Stream, |
| Thrift.Collections, |
| Thrift.Transport; |
| |
| type |
| |
| TType = ( |
| Stop = 0, |
| Void = 1, |
| Bool_ = 2, |
| Byte_ = 3, |
| Double_ = 4, |
| I16 = 6, |
| I32 = 8, |
| I64 = 10, |
| String_ = 11, |
| Struct = 12, |
| Map = 13, |
| Set_ = 14, |
| List = 15 |
| ); |
| |
| TMessageType = ( |
| Call = 1, |
| Reply = 2, |
| Exception = 3, |
| Oneway = 4 |
| ); |
| |
| IProtocol = interface; |
| IStruct = interface; |
| |
| IProtocolFactory = interface |
| ['{7CD64A10-4E9F-4E99-93BF-708A31F4A67B}'] |
| function GetProtocol( const trans: ITransport): IProtocol; |
| end; |
| |
| TThriftStringBuilder = class( TStringBuilder) |
| public |
| function Append(const Value: TBytes): TStringBuilder; overload; |
| function Append(const Value: IThriftContainer): TStringBuilder; overload; |
| end; |
| |
| TProtocolException = class( Exception ) |
| public |
| const // TODO(jensg): change into enum |
| UNKNOWN : Integer = 0; |
| INVALID_DATA : Integer = 1; |
| NEGATIVE_SIZE : Integer = 2; |
| SIZE_LIMIT : Integer = 3; |
| BAD_VERSION : Integer = 4; |
| NOT_IMPLEMENTED : Integer = 5; |
| DEPTH_LIMIT : Integer = 6; |
| protected |
| FType : Integer; |
| public |
| constructor Create; overload; |
| constructor Create( type_: Integer ); overload; |
| constructor Create( type_: Integer; const msg: string); overload; |
| end; |
| |
| IMap = interface |
| ['{30531D97-7E06-4233-B800-C3F53CCD23E7}'] |
| function GetKeyType: TType; |
| procedure SetKeyType( Value: TType); |
| function GetValueType: TType; |
| procedure SetValueType( Value: TType); |
| function GetCount: Integer; |
| procedure SetCount( Value: Integer); |
| property KeyType: TType read GetKeyType write SetKeyType; |
| property ValueType: TType read GetValueType write SetValueType; |
| property Count: Integer read GetCount write SetCount; |
| end; |
| |
| TMapImpl = class( TInterfacedObject, IMap) |
| private |
| FValueType: TType; |
| FKeyType: TType; |
| FCount: Integer; |
| protected |
| function GetKeyType: TType; |
| procedure SetKeyType( Value: TType); |
| function GetValueType: TType; |
| procedure SetValueType( Value: TType); |
| function GetCount: Integer; |
| procedure SetCount( Value: Integer); |
| public |
| constructor Create( AValueType: TType; AKeyType: TType; ACount: Integer); overload; |
| constructor Create; overload; |
| end; |
| |
| IList = interface |
| ['{6763E1EA-A934-4472-904F-0083980B9B87}'] |
| function GetElementType: TType; |
| procedure SetElementType( Value: TType); |
| function GetCount: Integer; |
| procedure SetCount( Value: Integer); |
| property ElementType: TType read GetElementType write SetElementType; |
| property Count: Integer read GetCount write SetCount; |
| end; |
| |
| TListImpl = class( TInterfacedObject, IList) |
| private |
| FElementType: TType; |
| FCount : Integer; |
| protected |
| function GetElementType: TType; |
| procedure SetElementType( Value: TType); |
| function GetCount: Integer; |
| procedure SetCount( Value: Integer); |
| public |
| constructor Create( AElementType: TType; ACount: Integer); overload; |
| constructor Create; overload; |
| end; |
| |
| ISet = interface |
| ['{A8671700-7514-4C1E-8A05-62786872005F}'] |
| function GetElementType: TType; |
| procedure SetElementType( Value: TType); |
| function GetCount: Integer; |
| procedure SetCount( Value: Integer); |
| property ElementType: TType read GetElementType write SetElementType; |
| property Count: Integer read GetCount write SetCount; |
| end; |
| |
| TSetImpl = class( TInterfacedObject, ISet) |
| private |
| FCount: Integer; |
| FElementType: TType; |
| protected |
| function GetElementType: TType; |
| procedure SetElementType( Value: TType); |
| function GetCount: Integer; |
| procedure SetCount( Value: Integer); |
| public |
| constructor Create( AElementType: TType; ACount: Integer); overload; |
| constructor Create; overload; |
| end; |
| |
| IMessage = interface |
| ['{9E368B4A-B1FA-43E7-8CF5-56C66D256CA7}'] |
| function GetName: string; |
| procedure SetName( const Value: string); |
| function GetType: TMessageType; |
| procedure SetType( Value: TMessageType); |
| function GetSeqID: Integer; |
| procedure SetSeqID( Value: Integer); |
| property Name: string read GetName write SetName; |
| property Type_: TMessageType read GetType write SetType; |
| property SeqID: Integer read GetSeqID write SetSeqID; |
| end; |
| |
| TMessageImpl = class( TInterfacedObject, IMessage ) |
| private |
| FName: string; |
| FMessageType: TMessageType; |
| FSeqID: Integer; |
| protected |
| function GetName: string; |
| procedure SetName( const Value: string); |
| function GetType: TMessageType; |
| procedure SetType( Value: TMessageType); |
| function GetSeqID: Integer; |
| procedure SetSeqID( Value: Integer); |
| public |
| property Name: string read FName write FName; |
| property Type_: TMessageType read FMessageType write FMessageType; |
| property SeqID: Integer read FSeqID write FSeqID; |
| constructor Create( AName: string; AMessageType: TMessageType; ASeqID: Integer); overload; |
| constructor Create; overload; |
| end; |
| |
| IField = interface |
| ['{F0D43BE5-7883-442E-83FF-0580CC632B72}'] |
| function GetName: string; |
| procedure SetName( const Value: string); |
| function GetType: TType; |
| procedure SetType( Value: TType); |
| function GetId: SmallInt; |
| procedure SetId( Value: SmallInt); |
| property Name: string read GetName write SetName; |
| property Type_: TType read GetType write SetType; |
| property Id: SmallInt read GetId write SetId; |
| end; |
| |
| TFieldImpl = class( TInterfacedObject, IField) |
| private |
| FName : string; |
| FType : TType; |
| FId : SmallInt; |
| protected |
| function GetName: string; |
| procedure SetName( const Value: string); |
| function GetType: TType; |
| procedure SetType( Value: TType); |
| function GetId: SmallInt; |
| procedure SetId( Value: SmallInt); |
| public |
| constructor Create( const AName: string; const AType: TType; AId: SmallInt); overload; |
| constructor Create; overload; |
| end; |
| |
| TProtocolUtil = class |
| public |
| class procedure Skip( prot: IProtocol; type_: TType); |
| end; |
| |
| IProtocol = interface |
| ['{FD95C151-1527-4C96-8134-B902BFC4B4FC}'] |
| function GetTransport: ITransport; |
| procedure WriteMessageBegin( const msg: IMessage); |
| procedure WriteMessageEnd; |
| procedure WriteStructBegin( const struc: IStruct); |
| procedure WriteStructEnd; |
| procedure WriteFieldBegin( const field: IField); |
| procedure WriteFieldEnd; |
| procedure WriteFieldStop; |
| procedure WriteMapBegin( const map: IMap); |
| procedure WriteMapEnd; |
| procedure WriteListBegin( const list: IList); |
| procedure WriteListEnd(); |
| procedure WriteSetBegin( const set_: ISet ); |
| procedure WriteSetEnd(); |
| procedure WriteBool( b: Boolean); |
| procedure WriteByte( b: ShortInt); |
| procedure WriteI16( i16: SmallInt); |
| procedure WriteI32( i32: Integer); |
| procedure WriteI64( const i64: Int64); |
| procedure WriteDouble( const d: Double); |
| procedure WriteString( const s: string ); |
| procedure WriteAnsiString( const s: AnsiString); |
| procedure WriteBinary( const b: TBytes); |
| |
| function ReadMessageBegin: IMessage; |
| procedure ReadMessageEnd(); |
| function ReadStructBegin: IStruct; |
| procedure ReadStructEnd; |
| function ReadFieldBegin: IField; |
| procedure ReadFieldEnd(); |
| function ReadMapBegin: IMap; |
| procedure ReadMapEnd(); |
| function ReadListBegin: IList; |
| procedure ReadListEnd(); |
| function ReadSetBegin: ISet; |
| procedure ReadSetEnd(); |
| function ReadBool: Boolean; |
| function ReadByte: ShortInt; |
| function ReadI16: SmallInt; |
| function ReadI32: Integer; |
| function ReadI64: Int64; |
| function ReadDouble:Double; |
| function ReadBinary: TBytes; |
| function ReadString: string; |
| function ReadAnsiString: AnsiString; |
| property Transport: ITransport read GetTransport; |
| end; |
| |
| TProtocolImpl = class abstract( TInterfacedObject, IProtocol) |
| protected |
| FTrans : ITransport; |
| function GetTransport: ITransport; |
| public |
| procedure WriteMessageBegin( const msg: IMessage); virtual; abstract; |
| procedure WriteMessageEnd; virtual; abstract; |
| procedure WriteStructBegin( const struc: IStruct); virtual; abstract; |
| procedure WriteStructEnd; virtual; abstract; |
| procedure WriteFieldBegin( const field: IField); virtual; abstract; |
| procedure WriteFieldEnd; virtual; abstract; |
| procedure WriteFieldStop; virtual; abstract; |
| procedure WriteMapBegin( const map: IMap); virtual; abstract; |
| procedure WriteMapEnd; virtual; abstract; |
| procedure WriteListBegin( const list: IList); virtual; abstract; |
| procedure WriteListEnd(); virtual; abstract; |
| procedure WriteSetBegin( const set_: ISet ); virtual; abstract; |
| procedure WriteSetEnd(); virtual; abstract; |
| procedure WriteBool( b: Boolean); virtual; abstract; |
| procedure WriteByte( b: ShortInt); virtual; abstract; |
| procedure WriteI16( i16: SmallInt); virtual; abstract; |
| procedure WriteI32( i32: Integer); virtual; abstract; |
| procedure WriteI64( const i64: Int64); virtual; abstract; |
| procedure WriteDouble( const d: Double); virtual; abstract; |
| procedure WriteString( const s: string ); virtual; |
| procedure WriteAnsiString( const s: AnsiString); virtual; |
| procedure WriteBinary( const b: TBytes); virtual; abstract; |
| |
| function ReadMessageBegin: IMessage; virtual; abstract; |
| procedure ReadMessageEnd(); virtual; abstract; |
| function ReadStructBegin: IStruct; virtual; abstract; |
| procedure ReadStructEnd; virtual; abstract; |
| function ReadFieldBegin: IField; virtual; abstract; |
| procedure ReadFieldEnd(); virtual; abstract; |
| function ReadMapBegin: IMap; virtual; abstract; |
| procedure ReadMapEnd(); virtual; abstract; |
| function ReadListBegin: IList; virtual; abstract; |
| procedure ReadListEnd(); virtual; abstract; |
| function ReadSetBegin: ISet; virtual; abstract; |
| procedure ReadSetEnd(); virtual; abstract; |
| function ReadBool: Boolean; virtual; abstract; |
| function ReadByte: ShortInt; virtual; abstract; |
| function ReadI16: SmallInt; virtual; abstract; |
| function ReadI32: Integer; virtual; abstract; |
| function ReadI64: Int64; virtual; abstract; |
| function ReadDouble:Double; virtual; abstract; |
| function ReadBinary: TBytes; virtual; abstract; |
| function ReadString: string; virtual; |
| function ReadAnsiString: AnsiString; virtual; |
| |
| property Transport: ITransport read GetTransport; |
| |
| constructor Create( trans: ITransport ); |
| end; |
| |
| IBase = interface |
| ['{08D9BAA8-5EAA-410F-B50B-AC2E6E5E4155}'] |
| function ToString: string; |
| procedure Read( const iprot: IProtocol); |
| procedure Write( const iprot: IProtocol); |
| end; |
| |
| IStruct = interface |
| ['{5DCE39AA-C916-4BC7-A79B-96A0C36B2220}'] |
| procedure SetName(const Value: string); |
| function GetName: string; |
| property Name: string read GetName write SetName; |
| end; |
| |
| TStructImpl = class( TInterfacedObject, IStruct ) |
| private |
| FName: string; |
| protected |
| function GetName: string; |
| procedure SetName(const Value: string); |
| public |
| constructor Create( const AName: string); |
| end; |
| |
| TBinaryProtocolImpl = class( TProtocolImpl ) |
| protected |
| const |
| VERSION_MASK : Cardinal = $ffff0000; |
| VERSION_1 : Cardinal = $80010000; |
| protected |
| FStrictRead : Boolean; |
| FStrictWrite : Boolean; |
| |
| private |
| function ReadAll( var buf: TBytes; off: Integer; len: Integer ): Integer; |
| function ReadStringBody( size: Integer): string; |
| |
| public |
| |
| type |
| TFactory = class( TInterfacedObject, IProtocolFactory) |
| protected |
| FStrictRead : Boolean; |
| FStrictWrite : Boolean; |
| public |
| function GetProtocol( const trans: ITransport): IProtocol; |
| constructor Create( AStrictRead, AStrictWrite: Boolean ); overload; |
| constructor Create; overload; |
| end; |
| |
| constructor Create( const trans: ITransport); overload; |
| constructor Create( const trans: ITransport; strictRead: Boolean; strictWrite: Boolean); overload; |
| |
| procedure WriteMessageBegin( const msg: IMessage); override; |
| procedure WriteMessageEnd; override; |
| procedure WriteStructBegin( const struc: IStruct); override; |
| procedure WriteStructEnd; override; |
| procedure WriteFieldBegin( const field: IField); override; |
| procedure WriteFieldEnd; override; |
| procedure WriteFieldStop; override; |
| procedure WriteMapBegin( const map: IMap); override; |
| procedure WriteMapEnd; override; |
| procedure WriteListBegin( const list: IList); override; |
| procedure WriteListEnd(); override; |
| procedure WriteSetBegin( const set_: ISet ); override; |
| procedure WriteSetEnd(); override; |
| procedure WriteBool( b: Boolean); override; |
| procedure WriteByte( b: ShortInt); override; |
| procedure WriteI16( i16: SmallInt); override; |
| procedure WriteI32( i32: Integer); override; |
| procedure WriteI64( const i64: Int64); override; |
| procedure WriteDouble( const d: Double); override; |
| procedure WriteBinary( const b: TBytes); override; |
| |
| function ReadMessageBegin: IMessage; override; |
| procedure ReadMessageEnd(); override; |
| function ReadStructBegin: IStruct; override; |
| procedure ReadStructEnd; override; |
| function ReadFieldBegin: IField; override; |
| procedure ReadFieldEnd(); override; |
| function ReadMapBegin: IMap; override; |
| procedure ReadMapEnd(); override; |
| function ReadListBegin: IList; override; |
| procedure ReadListEnd(); override; |
| function ReadSetBegin: ISet; override; |
| procedure ReadSetEnd(); override; |
| function ReadBool: Boolean; override; |
| function ReadByte: ShortInt; override; |
| function ReadI16: SmallInt; override; |
| function ReadI32: Integer; override; |
| function ReadI64: Int64; override; |
| function ReadDouble:Double; override; |
| function ReadBinary: TBytes; override; |
| |
| end; |
| |
| |
| { TProtocolDecorator forwards all requests to an enclosed TProtocol instance, |
| providing a way to author concise concrete decorator subclasses. The decorator |
| does not (and should not) modify the behaviour of the enclosed TProtocol |
| |
| See p.175 of Design Patterns (by Gamma et al.) |
| } |
| TProtocolDecorator = class( TProtocolImpl) |
| private |
| FWrappedProtocol : IProtocol; |
| |
| public |
| // Encloses the specified protocol. |
| // All operations will be forward to the given protocol. Must be non-null. |
| constructor Create( const aProtocol : IProtocol); |
| |
| procedure WriteMessageBegin( const msg: IMessage); override; |
| procedure WriteMessageEnd; override; |
| procedure WriteStructBegin( const struc: IStruct); override; |
| procedure WriteStructEnd; override; |
| procedure WriteFieldBegin( const field: IField); override; |
| procedure WriteFieldEnd; override; |
| procedure WriteFieldStop; override; |
| procedure WriteMapBegin( const map: IMap); override; |
| procedure WriteMapEnd; override; |
| procedure WriteListBegin( const list: IList); override; |
| procedure WriteListEnd(); override; |
| procedure WriteSetBegin( const set_: ISet ); override; |
| procedure WriteSetEnd(); override; |
| procedure WriteBool( b: Boolean); override; |
| procedure WriteByte( b: ShortInt); override; |
| procedure WriteI16( i16: SmallInt); override; |
| procedure WriteI32( i32: Integer); override; |
| procedure WriteI64( const i64: Int64); override; |
| procedure WriteDouble( const d: Double); override; |
| procedure WriteString( const s: string ); override; |
| procedure WriteAnsiString( const s: AnsiString); override; |
| procedure WriteBinary( const b: TBytes); override; |
| |
| function ReadMessageBegin: IMessage; override; |
| procedure ReadMessageEnd(); override; |
| function ReadStructBegin: IStruct; override; |
| procedure ReadStructEnd; override; |
| function ReadFieldBegin: IField; override; |
| procedure ReadFieldEnd(); override; |
| function ReadMapBegin: IMap; override; |
| procedure ReadMapEnd(); override; |
| function ReadListBegin: IList; override; |
| procedure ReadListEnd(); override; |
| function ReadSetBegin: ISet; override; |
| procedure ReadSetEnd(); override; |
| function ReadBool: Boolean; override; |
| function ReadByte: ShortInt; override; |
| function ReadI16: SmallInt; override; |
| function ReadI32: Integer; override; |
| function ReadI64: Int64; override; |
| function ReadDouble:Double; override; |
| function ReadBinary: TBytes; override; |
| function ReadString: string; override; |
| function ReadAnsiString: AnsiString; override; |
| end; |
| |
| |
| type |
| IRequestEvents = interface |
| ['{F926A26A-5B00-4560-86FA-2CAE3BA73DAF}'] |
| // Called before reading arguments. |
| procedure PreRead; |
| // Called between reading arguments and calling the handler. |
| procedure PostRead; |
| // Called between calling the handler and writing the response. |
| procedure PreWrite; |
| // Called after writing the response. |
| procedure PostWrite; |
| // Called when an oneway (async) function call completes successfully. |
| procedure OnewayComplete; |
| // Called if the handler throws an undeclared exception. |
| procedure UnhandledError( const e : Exception); |
| // Called when a client has finished request-handling to clean up |
| procedure CleanupContext; |
| end; |
| |
| |
| IProcessorEvents = interface |
| ['{A8661119-657C-447D-93C5-512E36162A45}'] |
| // Called when a client is about to call the processor. |
| procedure Processing( const transport : ITransport); |
| // Called on any service function invocation |
| function CreateRequestContext( const aFunctionName : string) : IRequestEvents; |
| // Called when a client has finished request-handling to clean up |
| procedure CleanupContext; |
| end; |
| |
| |
| IProcessor = interface |
| ['{7BAE92A5-46DA-4F13-B6EA-0EABE233EE5F}'] |
| function Process( const iprot :IProtocol; const oprot: IProtocol; const events : IProcessorEvents = nil): Boolean; |
| end; |
| |
| |
| |
| implementation |
| |
| function ConvertInt64ToDouble( const n: Int64): Double; |
| begin |
| ASSERT( SizeOf(n) = SizeOf(Result)); |
| System.Move( n, Result, SizeOf(Result)); |
| end; |
| |
| function ConvertDoubleToInt64( const d: Double): Int64; |
| begin |
| ASSERT( SizeOf(d) = SizeOf(Result)); |
| System.Move( d, Result, SizeOf(Result)); |
| end; |
| |
| { TFieldImpl } |
| |
| constructor TFieldImpl.Create(const AName: string; const AType: TType; |
| AId: SmallInt); |
| begin |
| inherited Create; |
| FName := AName; |
| FType := AType; |
| FId := AId; |
| end; |
| |
| constructor TFieldImpl.Create; |
| begin |
| inherited Create; |
| FName := ''; |
| FType := Low(TType); |
| FId := 0; |
| end; |
| |
| function TFieldImpl.GetId: SmallInt; |
| begin |
| Result := FId; |
| end; |
| |
| function TFieldImpl.GetName: string; |
| begin |
| Result := FName; |
| end; |
| |
| function TFieldImpl.GetType: TType; |
| begin |
| Result := FType; |
| end; |
| |
| procedure TFieldImpl.SetId(Value: SmallInt); |
| begin |
| FId := Value; |
| end; |
| |
| procedure TFieldImpl.SetName(const Value: string); |
| begin |
| FName := Value; |
| end; |
| |
| procedure TFieldImpl.SetType(Value: TType); |
| begin |
| FType := Value; |
| end; |
| |
| { TProtocolImpl } |
| |
| constructor TProtocolImpl.Create(trans: ITransport); |
| begin |
| inherited Create; |
| FTrans := trans; |
| end; |
| |
| function TProtocolImpl.GetTransport: ITransport; |
| begin |
| Result := FTrans; |
| end; |
| |
| function TProtocolImpl.ReadAnsiString: AnsiString; |
| var |
| b : TBytes; |
| len : Integer; |
| begin |
| Result := ''; |
| b := ReadBinary; |
| len := Length( b ); |
| if len > 0 then |
| begin |
| SetLength( Result, len); |
| System.Move( b[0], Pointer(Result)^, len ); |
| end; |
| end; |
| |
| function TProtocolImpl.ReadString: string; |
| begin |
| Result := TEncoding.UTF8.GetString( ReadBinary ); |
| end; |
| |
| procedure TProtocolImpl.WriteAnsiString(const s: AnsiString); |
| var |
| b : TBytes; |
| len : Integer; |
| begin |
| len := Length(s); |
| SetLength( b, len); |
| if len > 0 then |
| begin |
| System.Move( Pointer(s)^, b[0], len ); |
| end; |
| WriteBinary( b ); |
| end; |
| |
| procedure TProtocolImpl.WriteString(const s: string); |
| var |
| b : TBytes; |
| begin |
| b := TEncoding.UTF8.GetBytes(s); |
| WriteBinary( b ); |
| end; |
| |
| { TProtocolUtil } |
| |
| class procedure TProtocolUtil.Skip( prot: IProtocol; type_: TType); |
| var field : IField; |
| map : IMap; |
| set_ : ISet; |
| list : IList; |
| i : Integer; |
| begin |
| case type_ of |
| // simple types |
| TType.Bool_ : prot.ReadBool(); |
| TType.Byte_ : prot.ReadByte(); |
| TType.I16 : prot.ReadI16(); |
| TType.I32 : prot.ReadI32(); |
| TType.I64 : prot.ReadI64(); |
| TType.Double_ : prot.ReadDouble(); |
| TType.String_ : prot.ReadBinary();// Don't try to decode the string, just skip it. |
| |
| // structured types |
| TType.Struct : begin |
| prot.ReadStructBegin(); |
| while TRUE do begin |
| field := prot.ReadFieldBegin(); |
| if (field.Type_ = TType.Stop) then Break; |
| Skip(prot, field.Type_); |
| prot.ReadFieldEnd(); |
| end; |
| prot.ReadStructEnd(); |
| end; |
| |
| TType.Map : begin |
| map := prot.ReadMapBegin(); |
| for i := 0 to map.Count-1 do begin |
| Skip(prot, map.KeyType); |
| Skip(prot, map.ValueType); |
| end; |
| prot.ReadMapEnd(); |
| end; |
| |
| TType.Set_ : begin |
| set_ := prot.ReadSetBegin(); |
| for i := 0 to set_.Count-1 |
| do Skip( prot, set_.ElementType); |
| prot.ReadSetEnd(); |
| end; |
| |
| TType.List : begin |
| list := prot.ReadListBegin(); |
| for i := 0 to list.Count-1 |
| do Skip( prot, list.ElementType); |
| prot.ReadListEnd(); |
| end; |
| |
| else |
| ASSERT( FALSE); // any new types? |
| end; |
| end; |
| |
| { TStructImpl } |
| |
| constructor TStructImpl.Create(const AName: string); |
| begin |
| inherited Create; |
| FName := AName; |
| end; |
| |
| function TStructImpl.GetName: string; |
| begin |
| Result := FName; |
| end; |
| |
| procedure TStructImpl.SetName(const Value: string); |
| begin |
| FName := Value; |
| end; |
| |
| { TMapImpl } |
| |
| constructor TMapImpl.Create(AValueType, AKeyType: TType; ACount: Integer); |
| begin |
| inherited Create; |
| FValueType := AValueType; |
| FKeyType := AKeyType; |
| FCount := ACount; |
| end; |
| |
| constructor TMapImpl.Create; |
| begin |
| inherited Create; |
| end; |
| |
| function TMapImpl.GetCount: Integer; |
| begin |
| Result := FCount; |
| end; |
| |
| function TMapImpl.GetKeyType: TType; |
| begin |
| Result := FKeyType; |
| end; |
| |
| function TMapImpl.GetValueType: TType; |
| begin |
| Result := FValueType; |
| end; |
| |
| procedure TMapImpl.SetCount(Value: Integer); |
| begin |
| FCount := Value; |
| end; |
| |
| procedure TMapImpl.SetKeyType(Value: TType); |
| begin |
| FKeyType := Value; |
| end; |
| |
| procedure TMapImpl.SetValueType(Value: TType); |
| begin |
| FValueType := Value; |
| end; |
| |
| { IMessage } |
| |
| constructor TMessageImpl.Create(AName: string; AMessageType: TMessageType; |
| ASeqID: Integer); |
| begin |
| inherited Create; |
| FName := AName; |
| FMessageType := AMessageType; |
| FSeqID := ASeqID; |
| end; |
| |
| constructor TMessageImpl.Create; |
| begin |
| inherited; |
| end; |
| |
| function TMessageImpl.GetName: string; |
| begin |
| Result := FName; |
| end; |
| |
| function TMessageImpl.GetSeqID: Integer; |
| begin |
| Result := FSeqID; |
| end; |
| |
| function TMessageImpl.GetType: TMessageType; |
| begin |
| Result := FMessageType; |
| end; |
| |
| procedure TMessageImpl.SetName(const Value: string); |
| begin |
| FName := Value; |
| end; |
| |
| procedure TMessageImpl.SetSeqID(Value: Integer); |
| begin |
| FSeqID := Value; |
| end; |
| |
| procedure TMessageImpl.SetType(Value: TMessageType); |
| begin |
| FMessageType := Value; |
| end; |
| |
| { ISet } |
| |
| constructor TSetImpl.Create( AElementType: TType; ACount: Integer); |
| begin |
| inherited Create; |
| FCount := ACount; |
| FElementType := AElementType; |
| end; |
| |
| constructor TSetImpl.Create; |
| begin |
| inherited Create; |
| end; |
| |
| function TSetImpl.GetCount: Integer; |
| begin |
| Result := FCount; |
| end; |
| |
| function TSetImpl.GetElementType: TType; |
| begin |
| Result := FElementType; |
| end; |
| |
| procedure TSetImpl.SetCount(Value: Integer); |
| begin |
| FCount := Value; |
| end; |
| |
| procedure TSetImpl.SetElementType(Value: TType); |
| begin |
| FElementType := Value; |
| end; |
| |
| { IList } |
| |
| constructor TListImpl.Create( AElementType: TType; ACount: Integer); |
| begin |
| inherited Create; |
| FCount := ACount; |
| FElementType := AElementType; |
| end; |
| |
| constructor TListImpl.Create; |
| begin |
| inherited Create; |
| end; |
| |
| function TListImpl.GetCount: Integer; |
| begin |
| Result := FCount; |
| end; |
| |
| function TListImpl.GetElementType: TType; |
| begin |
| Result := FElementType; |
| end; |
| |
| procedure TListImpl.SetCount(Value: Integer); |
| begin |
| FCount := Value; |
| end; |
| |
| procedure TListImpl.SetElementType(Value: TType); |
| begin |
| FElementType := Value; |
| end; |
| |
| { TBinaryProtocolImpl } |
| |
| constructor TBinaryProtocolImpl.Create( const trans: ITransport); |
| begin |
| //no inherited |
| Create( trans, False, True); |
| end; |
| |
| constructor TBinaryProtocolImpl.Create( const trans: ITransport; strictRead, |
| strictWrite: Boolean); |
| begin |
| inherited Create( trans ); |
| FStrictRead := strictRead; |
| FStrictWrite := strictWrite; |
| end; |
| |
| function TBinaryProtocolImpl.ReadAll( var buf: TBytes; off, |
| len: Integer): Integer; |
| begin |
| Result := FTrans.ReadAll( buf, off, len ); |
| end; |
| |
| function TBinaryProtocolImpl.ReadBinary: TBytes; |
| var |
| size : Integer; |
| buf : TBytes; |
| begin |
| size := ReadI32; |
| SetLength( buf, size ); |
| FTrans.ReadAll( buf, 0, size); |
| Result := buf; |
| end; |
| |
| function TBinaryProtocolImpl.ReadBool: Boolean; |
| begin |
| Result := ReadByte = 1; |
| end; |
| |
| function TBinaryProtocolImpl.ReadByte: ShortInt; |
| var |
| bin : TBytes; |
| begin |
| SetLength( bin, 1); |
| ReadAll( bin, 0, 1 ); |
| Result := ShortInt( bin[0]); |
| end; |
| |
| function TBinaryProtocolImpl.ReadDouble: Double; |
| begin |
| Result := ConvertInt64ToDouble( ReadI64 ) |
| end; |
| |
| function TBinaryProtocolImpl.ReadFieldBegin: IField; |
| var |
| field : IField; |
| begin |
| field := TFieldImpl.Create; |
| field.Type_ := TType( ReadByte); |
| if ( field.Type_ <> TType.Stop ) then |
| begin |
| field.Id := ReadI16; |
| end; |
| Result := field; |
| end; |
| |
| procedure TBinaryProtocolImpl.ReadFieldEnd; |
| begin |
| |
| end; |
| |
| function TBinaryProtocolImpl.ReadI16: SmallInt; |
| var |
| i16in : TBytes; |
| begin |
| SetLength( i16in, 2 ); |
| ReadAll( i16in, 0, 2); |
| Result := SmallInt(((i16in[0] and $FF) shl 8) or (i16in[1] and $FF)); |
| end; |
| |
| function TBinaryProtocolImpl.ReadI32: Integer; |
| var |
| i32in : TBytes; |
| begin |
| SetLength( i32in, 4 ); |
| ReadAll( i32in, 0, 4); |
| |
| Result := Integer( |
| ((i32in[0] and $FF) shl 24) or |
| ((i32in[1] and $FF) shl 16) or |
| ((i32in[2] and $FF) shl 8) or |
| (i32in[3] and $FF)); |
| |
| end; |
| |
| function TBinaryProtocolImpl.ReadI64: Int64; |
| var |
| i64in : TBytes; |
| begin |
| SetLength( i64in, 8); |
| ReadAll( i64in, 0, 8); |
| Result := |
| (Int64( i64in[0] and $FF) shl 56) or |
| (Int64( i64in[1] and $FF) shl 48) or |
| (Int64( i64in[2] and $FF) shl 40) or |
| (Int64( i64in[3] and $FF) shl 32) or |
| (Int64( i64in[4] and $FF) shl 24) or |
| (Int64( i64in[5] and $FF) shl 16) or |
| (Int64( i64in[6] and $FF) shl 8) or |
| (Int64( i64in[7] and $FF)); |
| end; |
| |
| function TBinaryProtocolImpl.ReadListBegin: IList; |
| var |
| list : IList; |
| begin |
| list := TListImpl.Create; |
| list.ElementType := TType( ReadByte ); |
| list.Count := ReadI32; |
| Result := list; |
| end; |
| |
| procedure TBinaryProtocolImpl.ReadListEnd; |
| begin |
| |
| end; |
| |
| function TBinaryProtocolImpl.ReadMapBegin: IMap; |
| var |
| map : IMap; |
| begin |
| map := TMapImpl.Create; |
| map.KeyType := TType( ReadByte ); |
| map.ValueType := TType( ReadByte ); |
| map.Count := ReadI32; |
| Result := map; |
| end; |
| |
| procedure TBinaryProtocolImpl.ReadMapEnd; |
| begin |
| |
| end; |
| |
| function TBinaryProtocolImpl.ReadMessageBegin: IMessage; |
| var |
| size : Integer; |
| version : Integer; |
| message : IMessage; |
| begin |
| message := TMessageImpl.Create; |
| size := ReadI32; |
| if (size < 0) then |
| begin |
| version := size and Integer( VERSION_MASK); |
| if ( version <> Integer( VERSION_1)) then |
| begin |
| raise TProtocolException.Create(TProtocolException.BAD_VERSION, 'Bad version in ReadMessageBegin: ' + IntToStr(version) ); |
| end; |
| message.Type_ := TMessageType( size and $000000ff); |
| message.Name := ReadString; |
| message.SeqID := ReadI32; |
| end else |
| begin |
| if FStrictRead then |
| begin |
| raise TProtocolException.Create( TProtocolException.BAD_VERSION, 'Missing version in readMessageBegin, old client?' ); |
| end; |
| message.Name := ReadStringBody( size ); |
| message.Type_ := TMessageType( ReadByte ); |
| message.SeqID := ReadI32; |
| end; |
| Result := message; |
| end; |
| |
| procedure TBinaryProtocolImpl.ReadMessageEnd; |
| begin |
| inherited; |
| |
| end; |
| |
| function TBinaryProtocolImpl.ReadSetBegin: ISet; |
| var |
| set_ : ISet; |
| begin |
| set_ := TSetImpl.Create; |
| set_.ElementType := TType( ReadByte ); |
| set_.Count := ReadI32; |
| Result := set_; |
| end; |
| |
| procedure TBinaryProtocolImpl.ReadSetEnd; |
| begin |
| |
| end; |
| |
| function TBinaryProtocolImpl.ReadStringBody( size: Integer): string; |
| var |
| buf : TBytes; |
| begin |
| SetLength( buf, size ); |
| FTrans.ReadAll( buf, 0, size ); |
| Result := TEncoding.UTF8.GetString( buf); |
| end; |
| |
| function TBinaryProtocolImpl.ReadStructBegin: IStruct; |
| begin |
| Result := TStructImpl.Create(''); |
| end; |
| |
| procedure TBinaryProtocolImpl.ReadStructEnd; |
| begin |
| inherited; |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteBinary( const b: TBytes); |
| var iLen : Integer; |
| begin |
| iLen := Length(b); |
| WriteI32( iLen); |
| if iLen > 0 then FTrans.Write(b, 0, iLen); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteBool(b: Boolean); |
| begin |
| if b then |
| begin |
| WriteByte( 1 ); |
| end else |
| begin |
| WriteByte( 0 ); |
| end; |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteByte(b: ShortInt); |
| var |
| a : TBytes; |
| begin |
| SetLength( a, 1); |
| a[0] := Byte( b ); |
| FTrans.Write( a, 0, 1 ); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteDouble( const d: Double); |
| begin |
| WriteI64(ConvertDoubleToInt64(d)); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteFieldBegin( const field: IField); |
| begin |
| WriteByte(ShortInt(field.Type_)); |
| WriteI16(field.ID); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteFieldEnd; |
| begin |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteFieldStop; |
| begin |
| WriteByte(ShortInt(TType.Stop)); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteI16(i16: SmallInt); |
| var |
| i16out : TBytes; |
| begin |
| SetLength( i16out, 2); |
| i16out[0] := Byte($FF and (i16 shr 8)); |
| i16out[1] := Byte($FF and i16); |
| FTrans.Write( i16out ); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteI32(i32: Integer); |
| var |
| i32out : TBytes; |
| begin |
| SetLength( i32out, 4); |
| i32out[0] := Byte($FF and (i32 shr 24)); |
| i32out[1] := Byte($FF and (i32 shr 16)); |
| i32out[2] := Byte($FF and (i32 shr 8)); |
| i32out[3] := Byte($FF and i32); |
| FTrans.Write( i32out, 0, 4); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteI64( const i64: Int64); |
| var |
| i64out : TBytes; |
| begin |
| SetLength( i64out, 8); |
| i64out[0] := Byte($FF and (i64 shr 56)); |
| i64out[1] := Byte($FF and (i64 shr 48)); |
| i64out[2] := Byte($FF and (i64 shr 40)); |
| i64out[3] := Byte($FF and (i64 shr 32)); |
| i64out[4] := Byte($FF and (i64 shr 24)); |
| i64out[5] := Byte($FF and (i64 shr 16)); |
| i64out[6] := Byte($FF and (i64 shr 8)); |
| i64out[7] := Byte($FF and i64); |
| FTrans.Write( i64out, 0, 8); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteListBegin( const list: IList); |
| begin |
| WriteByte(ShortInt(list.ElementType)); |
| WriteI32(list.Count); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteListEnd; |
| begin |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteMapBegin( const map: IMap); |
| begin |
| WriteByte(ShortInt(map.KeyType)); |
| WriteByte(ShortInt(map.ValueType)); |
| WriteI32(map.Count); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteMapEnd; |
| begin |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteMessageBegin( const msg: IMessage); |
| var |
| version : Cardinal; |
| begin |
| if FStrictWrite then |
| begin |
| version := VERSION_1 or Cardinal( msg.Type_); |
| WriteI32( Integer( version) ); |
| WriteString( msg.Name); |
| WriteI32( msg.SeqID); |
| end else |
| begin |
| WriteString( msg.Name); |
| WriteByte(ShortInt( msg.Type_)); |
| WriteI32( msg.SeqID); |
| end; |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteMessageEnd; |
| begin |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteSetBegin( const set_: ISet); |
| begin |
| WriteByte(ShortInt(set_.ElementType)); |
| WriteI32(set_.Count); |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteSetEnd; |
| begin |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteStructBegin( const struc: IStruct); |
| begin |
| |
| end; |
| |
| procedure TBinaryProtocolImpl.WriteStructEnd; |
| begin |
| |
| end; |
| |
| { TProtocolException } |
| |
| constructor TProtocolException.Create; |
| begin |
| inherited Create(''); |
| FType := UNKNOWN; |
| end; |
| |
| constructor TProtocolException.Create(type_: Integer); |
| begin |
| inherited Create(''); |
| FType := type_; |
| end; |
| |
| constructor TProtocolException.Create(type_: Integer; const msg: string); |
| begin |
| inherited Create( msg ); |
| FType := type_; |
| end; |
| |
| { TThriftStringBuilder } |
| |
| function TThriftStringBuilder.Append(const Value: TBytes): TStringBuilder; |
| begin |
| Result := Append( string( RawByteString(Value)) ); |
| end; |
| |
| function TThriftStringBuilder.Append( |
| const Value: IThriftContainer): TStringBuilder; |
| begin |
| Result := Append( Value.ToString ); |
| end; |
| |
| { TBinaryProtocolImpl.TFactory } |
| |
| constructor TBinaryProtocolImpl.TFactory.Create(AStrictRead, AStrictWrite: Boolean); |
| begin |
| inherited Create; |
| FStrictRead := AStrictRead; |
| FStrictWrite := AStrictWrite; |
| end; |
| |
| constructor TBinaryProtocolImpl.TFactory.Create; |
| begin |
| //no inherited; |
| Create( False, True ) |
| end; |
| |
| function TBinaryProtocolImpl.TFactory.GetProtocol( const trans: ITransport): IProtocol; |
| begin |
| Result := TBinaryProtocolImpl.Create( trans, FStrictRead, FStrictWrite); |
| end; |
| |
| |
| { TProtocolDecorator } |
| |
| constructor TProtocolDecorator.Create( const aProtocol : IProtocol); |
| begin |
| ASSERT( aProtocol <> nil); |
| inherited Create( aProtocol.Transport); |
| FWrappedProtocol := aProtocol; |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteMessageBegin( const msg: IMessage); |
| begin |
| FWrappedProtocol.WriteMessageBegin( msg); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteMessageEnd; |
| begin |
| FWrappedProtocol.WriteMessageEnd; |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteStructBegin( const struc: IStruct); |
| begin |
| FWrappedProtocol.WriteStructBegin( struc); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteStructEnd; |
| begin |
| FWrappedProtocol.WriteStructEnd; |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteFieldBegin( const field: IField); |
| begin |
| FWrappedProtocol.WriteFieldBegin( field); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteFieldEnd; |
| begin |
| FWrappedProtocol.WriteFieldEnd; |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteFieldStop; |
| begin |
| FWrappedProtocol.WriteFieldStop; |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteMapBegin( const map: IMap); |
| begin |
| FWrappedProtocol.WriteMapBegin( map); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteMapEnd; |
| begin |
| FWrappedProtocol.WriteMapEnd; |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteListBegin( const list: IList); |
| begin |
| FWrappedProtocol.WriteListBegin( list); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteListEnd(); |
| begin |
| FWrappedProtocol.WriteListEnd(); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteSetBegin( const set_: ISet ); |
| begin |
| FWrappedProtocol.WriteSetBegin( set_); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteSetEnd(); |
| begin |
| FWrappedProtocol.WriteSetEnd(); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteBool( b: Boolean); |
| begin |
| FWrappedProtocol.WriteBool( b); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteByte( b: ShortInt); |
| begin |
| FWrappedProtocol.WriteByte( b); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteI16( i16: SmallInt); |
| begin |
| FWrappedProtocol.WriteI16( i16); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteI32( i32: Integer); |
| begin |
| FWrappedProtocol.WriteI32( i32); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteI64( const i64: Int64); |
| begin |
| FWrappedProtocol.WriteI64( i64); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteDouble( const d: Double); |
| begin |
| FWrappedProtocol.WriteDouble( d); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteString( const s: string ); |
| begin |
| FWrappedProtocol.WriteString( s); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteAnsiString( const s: AnsiString); |
| begin |
| FWrappedProtocol.WriteAnsiString( s); |
| end; |
| |
| |
| procedure TProtocolDecorator.WriteBinary( const b: TBytes); |
| begin |
| FWrappedProtocol.WriteBinary( b); |
| end; |
| |
| |
| function TProtocolDecorator.ReadMessageBegin: IMessage; |
| begin |
| result := FWrappedProtocol.ReadMessageBegin; |
| end; |
| |
| |
| procedure TProtocolDecorator.ReadMessageEnd(); |
| begin |
| FWrappedProtocol.ReadMessageEnd(); |
| end; |
| |
| |
| function TProtocolDecorator.ReadStructBegin: IStruct; |
| begin |
| result := FWrappedProtocol.ReadStructBegin; |
| end; |
| |
| |
| procedure TProtocolDecorator.ReadStructEnd; |
| begin |
| FWrappedProtocol.ReadStructEnd; |
| end; |
| |
| |
| function TProtocolDecorator.ReadFieldBegin: IField; |
| begin |
| result := FWrappedProtocol.ReadFieldBegin; |
| end; |
| |
| |
| procedure TProtocolDecorator.ReadFieldEnd(); |
| begin |
| FWrappedProtocol.ReadFieldEnd(); |
| end; |
| |
| |
| function TProtocolDecorator.ReadMapBegin: IMap; |
| begin |
| result := FWrappedProtocol.ReadMapBegin; |
| end; |
| |
| |
| procedure TProtocolDecorator.ReadMapEnd(); |
| begin |
| FWrappedProtocol.ReadMapEnd(); |
| end; |
| |
| |
| function TProtocolDecorator.ReadListBegin: IList; |
| begin |
| result := FWrappedProtocol.ReadListBegin; |
| end; |
| |
| |
| procedure TProtocolDecorator.ReadListEnd(); |
| begin |
| FWrappedProtocol.ReadListEnd(); |
| end; |
| |
| |
| function TProtocolDecorator.ReadSetBegin: ISet; |
| begin |
| result := FWrappedProtocol.ReadSetBegin; |
| end; |
| |
| |
| procedure TProtocolDecorator.ReadSetEnd(); |
| begin |
| FWrappedProtocol.ReadSetEnd(); |
| end; |
| |
| |
| function TProtocolDecorator.ReadBool: Boolean; |
| begin |
| result := FWrappedProtocol.ReadBool; |
| end; |
| |
| |
| function TProtocolDecorator.ReadByte: ShortInt; |
| begin |
| result := FWrappedProtocol.ReadByte; |
| end; |
| |
| |
| function TProtocolDecorator.ReadI16: SmallInt; |
| begin |
| result := FWrappedProtocol.ReadI16; |
| end; |
| |
| |
| function TProtocolDecorator.ReadI32: Integer; |
| begin |
| result := FWrappedProtocol.ReadI32; |
| end; |
| |
| |
| function TProtocolDecorator.ReadI64: Int64; |
| begin |
| result := FWrappedProtocol.ReadI64; |
| end; |
| |
| |
| function TProtocolDecorator.ReadDouble:Double; |
| begin |
| result := FWrappedProtocol.ReadDouble; |
| end; |
| |
| |
| function TProtocolDecorator.ReadBinary: TBytes; |
| begin |
| result := FWrappedProtocol.ReadBinary; |
| end; |
| |
| |
| function TProtocolDecorator.ReadString: string; |
| begin |
| result := FWrappedProtocol.ReadString; |
| end; |
| |
| |
| function TProtocolDecorator.ReadAnsiString: AnsiString; |
| begin |
| result := FWrappedProtocol.ReadAnsiString; |
| end; |
| |
| |
| |
| end. |
| |