blob: 45c880d32f8fd444e41662f06697cacff72683e1 [file] [log] [blame]
Girish Gowdra5d7d6442020-09-08 17:03:11 -07001/*
2 * Licensed to the Apache Software Foundation (ASF) under one
3 * or more contributor license agreements. See the NOTICE file
4 * distributed with this work for additional information
5 * regarding copyright ownership. The ASF licenses this file
6 * to you under the Apache License, Version 2.0 (the
7 * "License"); you may not use this file except in compliance
8 * with the License. You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing,
13 * software distributed under the License is distributed on an
14 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 * KIND, either express or implied. See the License for the
16 * specific language governing permissions and limitations
17 * under the License.
18 */
19
20package thrift
21
22import (
23 "bytes"
Girish Gowdra390f12f2021-07-01 15:53:49 -070024 "context"
Girish Gowdra5d7d6442020-09-08 17:03:11 -070025 "encoding/binary"
26 "errors"
27 "fmt"
28 "io"
29 "math"
30)
31
32type TBinaryProtocol struct {
33 trans TRichTransport
34 origTransport TTransport
Girish Gowdra390f12f2021-07-01 15:53:49 -070035 cfg *TConfiguration
Girish Gowdra5d7d6442020-09-08 17:03:11 -070036 buffer [64]byte
37}
38
39type TBinaryProtocolFactory struct {
Girish Gowdra390f12f2021-07-01 15:53:49 -070040 cfg *TConfiguration
Girish Gowdra5d7d6442020-09-08 17:03:11 -070041}
42
Girish Gowdra390f12f2021-07-01 15:53:49 -070043// Deprecated: Use NewTBinaryProtocolConf instead.
Girish Gowdra5d7d6442020-09-08 17:03:11 -070044func NewTBinaryProtocolTransport(t TTransport) *TBinaryProtocol {
Girish Gowdra390f12f2021-07-01 15:53:49 -070045 return NewTBinaryProtocolConf(t, &TConfiguration{
46 noPropagation: true,
47 })
Girish Gowdra5d7d6442020-09-08 17:03:11 -070048}
49
Girish Gowdra390f12f2021-07-01 15:53:49 -070050// Deprecated: Use NewTBinaryProtocolConf instead.
Girish Gowdra5d7d6442020-09-08 17:03:11 -070051func NewTBinaryProtocol(t TTransport, strictRead, strictWrite bool) *TBinaryProtocol {
Girish Gowdra390f12f2021-07-01 15:53:49 -070052 return NewTBinaryProtocolConf(t, &TConfiguration{
53 TBinaryStrictRead: &strictRead,
54 TBinaryStrictWrite: &strictWrite,
55
56 noPropagation: true,
57 })
58}
59
60func NewTBinaryProtocolConf(t TTransport, conf *TConfiguration) *TBinaryProtocol {
61 PropagateTConfiguration(t, conf)
62 p := &TBinaryProtocol{
63 origTransport: t,
64 cfg: conf,
65 }
Girish Gowdra5d7d6442020-09-08 17:03:11 -070066 if et, ok := t.(TRichTransport); ok {
67 p.trans = et
68 } else {
69 p.trans = NewTRichTransport(t)
70 }
Girish Gowdra5d7d6442020-09-08 17:03:11 -070071 return p
72}
73
Girish Gowdra390f12f2021-07-01 15:53:49 -070074// Deprecated: Use NewTBinaryProtocolFactoryConf instead.
Girish Gowdra5d7d6442020-09-08 17:03:11 -070075func NewTBinaryProtocolFactoryDefault() *TBinaryProtocolFactory {
Girish Gowdra390f12f2021-07-01 15:53:49 -070076 return NewTBinaryProtocolFactoryConf(&TConfiguration{
77 noPropagation: true,
78 })
Girish Gowdra5d7d6442020-09-08 17:03:11 -070079}
80
Girish Gowdra390f12f2021-07-01 15:53:49 -070081// Deprecated: Use NewTBinaryProtocolFactoryConf instead.
Girish Gowdra5d7d6442020-09-08 17:03:11 -070082func NewTBinaryProtocolFactory(strictRead, strictWrite bool) *TBinaryProtocolFactory {
Girish Gowdra390f12f2021-07-01 15:53:49 -070083 return NewTBinaryProtocolFactoryConf(&TConfiguration{
84 TBinaryStrictRead: &strictRead,
85 TBinaryStrictWrite: &strictWrite,
86
87 noPropagation: true,
88 })
89}
90
91func NewTBinaryProtocolFactoryConf(conf *TConfiguration) *TBinaryProtocolFactory {
92 return &TBinaryProtocolFactory{
93 cfg: conf,
94 }
Girish Gowdra5d7d6442020-09-08 17:03:11 -070095}
96
97func (p *TBinaryProtocolFactory) GetProtocol(t TTransport) TProtocol {
Girish Gowdra390f12f2021-07-01 15:53:49 -070098 return NewTBinaryProtocolConf(t, p.cfg)
99}
100
101func (p *TBinaryProtocolFactory) SetTConfiguration(conf *TConfiguration) {
102 p.cfg = conf
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700103}
104
105/**
106 * Writing Methods
107 */
108
Girish Gowdra390f12f2021-07-01 15:53:49 -0700109func (p *TBinaryProtocol) WriteMessageBegin(ctx context.Context, name string, typeId TMessageType, seqId int32) error {
110 if p.cfg.GetTBinaryStrictWrite() {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700111 version := uint32(VERSION_1) | uint32(typeId)
Girish Gowdra390f12f2021-07-01 15:53:49 -0700112 e := p.WriteI32(ctx, int32(version))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700113 if e != nil {
114 return e
115 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700116 e = p.WriteString(ctx, name)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700117 if e != nil {
118 return e
119 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700120 e = p.WriteI32(ctx, seqId)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700121 return e
122 } else {
Girish Gowdra390f12f2021-07-01 15:53:49 -0700123 e := p.WriteString(ctx, name)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700124 if e != nil {
125 return e
126 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700127 e = p.WriteByte(ctx, int8(typeId))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700128 if e != nil {
129 return e
130 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700131 e = p.WriteI32(ctx, seqId)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700132 return e
133 }
134 return nil
135}
136
Girish Gowdra390f12f2021-07-01 15:53:49 -0700137func (p *TBinaryProtocol) WriteMessageEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700138 return nil
139}
140
Girish Gowdra390f12f2021-07-01 15:53:49 -0700141func (p *TBinaryProtocol) WriteStructBegin(ctx context.Context, name string) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700142 return nil
143}
144
Girish Gowdra390f12f2021-07-01 15:53:49 -0700145func (p *TBinaryProtocol) WriteStructEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700146 return nil
147}
148
Girish Gowdra390f12f2021-07-01 15:53:49 -0700149func (p *TBinaryProtocol) WriteFieldBegin(ctx context.Context, name string, typeId TType, id int16) error {
150 e := p.WriteByte(ctx, int8(typeId))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700151 if e != nil {
152 return e
153 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700154 e = p.WriteI16(ctx, id)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700155 return e
156}
157
Girish Gowdra390f12f2021-07-01 15:53:49 -0700158func (p *TBinaryProtocol) WriteFieldEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700159 return nil
160}
161
Girish Gowdra390f12f2021-07-01 15:53:49 -0700162func (p *TBinaryProtocol) WriteFieldStop(ctx context.Context) error {
163 e := p.WriteByte(ctx, STOP)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700164 return e
165}
166
Girish Gowdra390f12f2021-07-01 15:53:49 -0700167func (p *TBinaryProtocol) WriteMapBegin(ctx context.Context, keyType TType, valueType TType, size int) error {
168 e := p.WriteByte(ctx, int8(keyType))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700169 if e != nil {
170 return e
171 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700172 e = p.WriteByte(ctx, int8(valueType))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700173 if e != nil {
174 return e
175 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700176 e = p.WriteI32(ctx, int32(size))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700177 return e
178}
179
Girish Gowdra390f12f2021-07-01 15:53:49 -0700180func (p *TBinaryProtocol) WriteMapEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700181 return nil
182}
183
Girish Gowdra390f12f2021-07-01 15:53:49 -0700184func (p *TBinaryProtocol) WriteListBegin(ctx context.Context, elemType TType, size int) error {
185 e := p.WriteByte(ctx, int8(elemType))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700186 if e != nil {
187 return e
188 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700189 e = p.WriteI32(ctx, int32(size))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700190 return e
191}
192
Girish Gowdra390f12f2021-07-01 15:53:49 -0700193func (p *TBinaryProtocol) WriteListEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700194 return nil
195}
196
Girish Gowdra390f12f2021-07-01 15:53:49 -0700197func (p *TBinaryProtocol) WriteSetBegin(ctx context.Context, elemType TType, size int) error {
198 e := p.WriteByte(ctx, int8(elemType))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700199 if e != nil {
200 return e
201 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700202 e = p.WriteI32(ctx, int32(size))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700203 return e
204}
205
Girish Gowdra390f12f2021-07-01 15:53:49 -0700206func (p *TBinaryProtocol) WriteSetEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700207 return nil
208}
209
Girish Gowdra390f12f2021-07-01 15:53:49 -0700210func (p *TBinaryProtocol) WriteBool(ctx context.Context, value bool) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700211 if value {
Girish Gowdra390f12f2021-07-01 15:53:49 -0700212 return p.WriteByte(ctx, 1)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700213 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700214 return p.WriteByte(ctx, 0)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700215}
216
Girish Gowdra390f12f2021-07-01 15:53:49 -0700217func (p *TBinaryProtocol) WriteByte(ctx context.Context, value int8) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700218 e := p.trans.WriteByte(byte(value))
219 return NewTProtocolException(e)
220}
221
Girish Gowdra390f12f2021-07-01 15:53:49 -0700222func (p *TBinaryProtocol) WriteI16(ctx context.Context, value int16) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700223 v := p.buffer[0:2]
224 binary.BigEndian.PutUint16(v, uint16(value))
Girish Gowdra390f12f2021-07-01 15:53:49 -0700225 _, e := p.trans.Write(v)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700226 return NewTProtocolException(e)
227}
228
Girish Gowdra390f12f2021-07-01 15:53:49 -0700229func (p *TBinaryProtocol) WriteI32(ctx context.Context, value int32) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700230 v := p.buffer[0:4]
231 binary.BigEndian.PutUint32(v, uint32(value))
Girish Gowdra390f12f2021-07-01 15:53:49 -0700232 _, e := p.trans.Write(v)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700233 return NewTProtocolException(e)
234}
235
Girish Gowdra390f12f2021-07-01 15:53:49 -0700236func (p *TBinaryProtocol) WriteI64(ctx context.Context, value int64) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700237 v := p.buffer[0:8]
238 binary.BigEndian.PutUint64(v, uint64(value))
Girish Gowdra390f12f2021-07-01 15:53:49 -0700239 _, err := p.trans.Write(v)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700240 return NewTProtocolException(err)
241}
242
Girish Gowdra390f12f2021-07-01 15:53:49 -0700243func (p *TBinaryProtocol) WriteDouble(ctx context.Context, value float64) error {
244 return p.WriteI64(ctx, int64(math.Float64bits(value)))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700245}
246
Girish Gowdra390f12f2021-07-01 15:53:49 -0700247func (p *TBinaryProtocol) WriteString(ctx context.Context, value string) error {
248 e := p.WriteI32(ctx, int32(len(value)))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700249 if e != nil {
250 return e
251 }
252 _, err := p.trans.WriteString(value)
253 return NewTProtocolException(err)
254}
255
Girish Gowdra390f12f2021-07-01 15:53:49 -0700256func (p *TBinaryProtocol) WriteBinary(ctx context.Context, value []byte) error {
257 e := p.WriteI32(ctx, int32(len(value)))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700258 if e != nil {
259 return e
260 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700261 _, err := p.trans.Write(value)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700262 return NewTProtocolException(err)
263}
264
265/**
266 * Reading methods
267 */
268
Girish Gowdra390f12f2021-07-01 15:53:49 -0700269func (p *TBinaryProtocol) ReadMessageBegin(ctx context.Context) (name string, typeId TMessageType, seqId int32, err error) {
270 size, e := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700271 if e != nil {
272 return "", typeId, 0, NewTProtocolException(e)
273 }
274 if size < 0 {
275 typeId = TMessageType(size & 0x0ff)
276 version := int64(int64(size) & VERSION_MASK)
277 if version != VERSION_1 {
278 return name, typeId, seqId, NewTProtocolExceptionWithType(BAD_VERSION, fmt.Errorf("Bad version in ReadMessageBegin"))
279 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700280 name, e = p.ReadString(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700281 if e != nil {
282 return name, typeId, seqId, NewTProtocolException(e)
283 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700284 seqId, e = p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700285 if e != nil {
286 return name, typeId, seqId, NewTProtocolException(e)
287 }
288 return name, typeId, seqId, nil
289 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700290 if p.cfg.GetTBinaryStrictRead() {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700291 return name, typeId, seqId, NewTProtocolExceptionWithType(BAD_VERSION, fmt.Errorf("Missing version in ReadMessageBegin"))
292 }
293 name, e2 := p.readStringBody(size)
294 if e2 != nil {
295 return name, typeId, seqId, e2
296 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700297 b, e3 := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700298 if e3 != nil {
299 return name, typeId, seqId, e3
300 }
301 typeId = TMessageType(b)
Girish Gowdra390f12f2021-07-01 15:53:49 -0700302 seqId, e4 := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700303 if e4 != nil {
304 return name, typeId, seqId, e4
305 }
306 return name, typeId, seqId, nil
307}
308
Girish Gowdra390f12f2021-07-01 15:53:49 -0700309func (p *TBinaryProtocol) ReadMessageEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700310 return nil
311}
312
Girish Gowdra390f12f2021-07-01 15:53:49 -0700313func (p *TBinaryProtocol) ReadStructBegin(ctx context.Context) (name string, err error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700314 return
315}
316
Girish Gowdra390f12f2021-07-01 15:53:49 -0700317func (p *TBinaryProtocol) ReadStructEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700318 return nil
319}
320
Girish Gowdra390f12f2021-07-01 15:53:49 -0700321func (p *TBinaryProtocol) ReadFieldBegin(ctx context.Context) (name string, typeId TType, seqId int16, err error) {
322 t, err := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700323 typeId = TType(t)
324 if err != nil {
325 return name, typeId, seqId, err
326 }
327 if t != STOP {
Girish Gowdra390f12f2021-07-01 15:53:49 -0700328 seqId, err = p.ReadI16(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700329 }
330 return name, typeId, seqId, err
331}
332
Girish Gowdra390f12f2021-07-01 15:53:49 -0700333func (p *TBinaryProtocol) ReadFieldEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700334 return nil
335}
336
337var invalidDataLength = NewTProtocolExceptionWithType(INVALID_DATA, errors.New("Invalid data length"))
338
Girish Gowdra390f12f2021-07-01 15:53:49 -0700339func (p *TBinaryProtocol) ReadMapBegin(ctx context.Context) (kType, vType TType, size int, err error) {
340 k, e := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700341 if e != nil {
342 err = NewTProtocolException(e)
343 return
344 }
345 kType = TType(k)
Girish Gowdra390f12f2021-07-01 15:53:49 -0700346 v, e := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700347 if e != nil {
348 err = NewTProtocolException(e)
349 return
350 }
351 vType = TType(v)
Girish Gowdra390f12f2021-07-01 15:53:49 -0700352 size32, e := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700353 if e != nil {
354 err = NewTProtocolException(e)
355 return
356 }
357 if size32 < 0 {
358 err = invalidDataLength
359 return
360 }
361 size = int(size32)
362 return kType, vType, size, nil
363}
364
Girish Gowdra390f12f2021-07-01 15:53:49 -0700365func (p *TBinaryProtocol) ReadMapEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700366 return nil
367}
368
Girish Gowdra390f12f2021-07-01 15:53:49 -0700369func (p *TBinaryProtocol) ReadListBegin(ctx context.Context) (elemType TType, size int, err error) {
370 b, e := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700371 if e != nil {
372 err = NewTProtocolException(e)
373 return
374 }
375 elemType = TType(b)
Girish Gowdra390f12f2021-07-01 15:53:49 -0700376 size32, e := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700377 if e != nil {
378 err = NewTProtocolException(e)
379 return
380 }
381 if size32 < 0 {
382 err = invalidDataLength
383 return
384 }
385 size = int(size32)
386
387 return
388}
389
Girish Gowdra390f12f2021-07-01 15:53:49 -0700390func (p *TBinaryProtocol) ReadListEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700391 return nil
392}
393
Girish Gowdra390f12f2021-07-01 15:53:49 -0700394func (p *TBinaryProtocol) ReadSetBegin(ctx context.Context) (elemType TType, size int, err error) {
395 b, e := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700396 if e != nil {
397 err = NewTProtocolException(e)
398 return
399 }
400 elemType = TType(b)
Girish Gowdra390f12f2021-07-01 15:53:49 -0700401 size32, e := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700402 if e != nil {
403 err = NewTProtocolException(e)
404 return
405 }
406 if size32 < 0 {
407 err = invalidDataLength
408 return
409 }
410 size = int(size32)
411 return elemType, size, nil
412}
413
Girish Gowdra390f12f2021-07-01 15:53:49 -0700414func (p *TBinaryProtocol) ReadSetEnd(ctx context.Context) error {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700415 return nil
416}
417
Girish Gowdra390f12f2021-07-01 15:53:49 -0700418func (p *TBinaryProtocol) ReadBool(ctx context.Context) (bool, error) {
419 b, e := p.ReadByte(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700420 v := true
421 if b != 1 {
422 v = false
423 }
424 return v, e
425}
426
Girish Gowdra390f12f2021-07-01 15:53:49 -0700427func (p *TBinaryProtocol) ReadByte(ctx context.Context) (int8, error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700428 v, err := p.trans.ReadByte()
429 return int8(v), err
430}
431
Girish Gowdra390f12f2021-07-01 15:53:49 -0700432func (p *TBinaryProtocol) ReadI16(ctx context.Context) (value int16, err error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700433 buf := p.buffer[0:2]
Girish Gowdra390f12f2021-07-01 15:53:49 -0700434 err = p.readAll(ctx, buf)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700435 value = int16(binary.BigEndian.Uint16(buf))
436 return value, err
437}
438
Girish Gowdra390f12f2021-07-01 15:53:49 -0700439func (p *TBinaryProtocol) ReadI32(ctx context.Context) (value int32, err error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700440 buf := p.buffer[0:4]
Girish Gowdra390f12f2021-07-01 15:53:49 -0700441 err = p.readAll(ctx, buf)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700442 value = int32(binary.BigEndian.Uint32(buf))
443 return value, err
444}
445
Girish Gowdra390f12f2021-07-01 15:53:49 -0700446func (p *TBinaryProtocol) ReadI64(ctx context.Context) (value int64, err error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700447 buf := p.buffer[0:8]
Girish Gowdra390f12f2021-07-01 15:53:49 -0700448 err = p.readAll(ctx, buf)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700449 value = int64(binary.BigEndian.Uint64(buf))
450 return value, err
451}
452
Girish Gowdra390f12f2021-07-01 15:53:49 -0700453func (p *TBinaryProtocol) ReadDouble(ctx context.Context) (value float64, err error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700454 buf := p.buffer[0:8]
Girish Gowdra390f12f2021-07-01 15:53:49 -0700455 err = p.readAll(ctx, buf)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700456 value = math.Float64frombits(binary.BigEndian.Uint64(buf))
457 return value, err
458}
459
Girish Gowdra390f12f2021-07-01 15:53:49 -0700460func (p *TBinaryProtocol) ReadString(ctx context.Context) (value string, err error) {
461 size, e := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700462 if e != nil {
463 return "", e
464 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700465 err = checkSizeForProtocol(size, p.cfg)
466 if err != nil {
467 return
468 }
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700469 if size < 0 {
470 err = invalidDataLength
471 return
472 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700473 if size == 0 {
474 return "", nil
475 }
476 if size < int32(len(p.buffer)) {
477 // Avoid allocation on small reads
478 buf := p.buffer[:size]
479 read, e := io.ReadFull(p.trans, buf)
480 return string(buf[:read]), NewTProtocolException(e)
481 }
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700482
483 return p.readStringBody(size)
484}
485
Girish Gowdra390f12f2021-07-01 15:53:49 -0700486func (p *TBinaryProtocol) ReadBinary(ctx context.Context) ([]byte, error) {
487 size, e := p.ReadI32(ctx)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700488 if e != nil {
489 return nil, e
490 }
Girish Gowdra390f12f2021-07-01 15:53:49 -0700491 if err := checkSizeForProtocol(size, p.cfg); err != nil {
492 return nil, err
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700493 }
494
Girish Gowdra390f12f2021-07-01 15:53:49 -0700495 buf, err := safeReadBytes(size, p.trans)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700496 return buf, NewTProtocolException(err)
497}
498
Girish Gowdra390f12f2021-07-01 15:53:49 -0700499func (p *TBinaryProtocol) Flush(ctx context.Context) (err error) {
500 return NewTProtocolException(p.trans.Flush(ctx))
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700501}
502
Girish Gowdra390f12f2021-07-01 15:53:49 -0700503func (p *TBinaryProtocol) Skip(ctx context.Context, fieldType TType) (err error) {
504 return SkipDefaultDepth(ctx, p, fieldType)
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700505}
506
507func (p *TBinaryProtocol) Transport() TTransport {
508 return p.origTransport
509}
510
Girish Gowdra390f12f2021-07-01 15:53:49 -0700511func (p *TBinaryProtocol) readAll(ctx context.Context, buf []byte) (err error) {
512 var read int
513 _, deadlineSet := ctx.Deadline()
514 for {
515 read, err = io.ReadFull(p.trans, buf)
516 if deadlineSet && read == 0 && isTimeoutError(err) && ctx.Err() == nil {
517 // This is I/O timeout without anything read,
518 // and we still have time left, keep retrying.
519 continue
520 }
521 // For anything else, don't retry
522 break
523 }
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700524 return NewTProtocolException(err)
525}
526
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700527func (p *TBinaryProtocol) readStringBody(size int32) (value string, err error) {
Girish Gowdra390f12f2021-07-01 15:53:49 -0700528 buf, err := safeReadBytes(size, p.trans)
529 return string(buf), NewTProtocolException(err)
530}
531
532func (p *TBinaryProtocol) SetTConfiguration(conf *TConfiguration) {
533 PropagateTConfiguration(p.trans, conf)
534 PropagateTConfiguration(p.origTransport, conf)
535 p.cfg = conf
536}
537
538var (
539 _ TConfigurationSetter = (*TBinaryProtocolFactory)(nil)
540 _ TConfigurationSetter = (*TBinaryProtocol)(nil)
541)
542
543// This function is shared between TBinaryProtocol and TCompactProtocol.
544//
545// It tries to read size bytes from trans, in a way that prevents large
546// allocations when size is insanely large (mostly caused by malformed message).
547func safeReadBytes(size int32, trans io.Reader) ([]byte, error) {
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700548 if size < 0 {
Girish Gowdra390f12f2021-07-01 15:53:49 -0700549 return nil, nil
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700550 }
551
Girish Gowdra390f12f2021-07-01 15:53:49 -0700552 buf := new(bytes.Buffer)
553 _, err := io.CopyN(buf, trans, int64(size))
554 return buf.Bytes(), err
Girish Gowdra5d7d6442020-09-08 17:03:11 -0700555}