Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 1 | %% |
| 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 | |
| 20 | -module(thrift_compact_protocol). |
| 21 | |
| 22 | -behaviour(thrift_protocol). |
| 23 | |
| 24 | -include("thrift_constants.hrl"). |
| 25 | -include("thrift_protocol.hrl"). |
| 26 | |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 27 | -export([ |
| 28 | new/1, new/2, |
| 29 | read/2, |
| 30 | write/2, |
| 31 | flush_transport/1, |
| 32 | close_transport/1, |
| 33 | new_protocol_factory/2 |
| 34 | ]). |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 35 | |
| 36 | -define(ID_NONE, 16#10000). |
| 37 | -define(CBOOL_NONE, 0). |
Vince Foley | d150f8b | 2016-07-14 14:21:00 -0700 | [diff] [blame] | 38 | -define(CBOOL_TRUE, 1). |
| 39 | -define(CBOOL_FALSE, 2). |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 40 | |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 41 | -type cbool() :: ?CBOOL_NONE | ?CBOOL_TRUE | ?CBOOL_FALSE. |
| 42 | |
| 43 | -record(t_compact, { |
| 44 | transport :: term(), |
| 45 | % state for pending boolean fields |
| 46 | read_stack = [] :: iodata(), |
| 47 | read_value = ?CBOOL_NONE :: cbool(), |
| 48 | write_stack = [] :: iodata(), |
| 49 | write_id = ?ID_NONE :: non_neg_integer() |
| 50 | }). |
| 51 | |
| 52 | -type t_compact() :: #t_compact{}. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 53 | |
| 54 | -define(PROTOCOL_ID, 16#82). |
| 55 | -define(VERSION_MASK, 16#1f). |
| 56 | -define(VERSION_1, 16#01). |
| 57 | -define(TYPE_MASK, 16#E0). |
| 58 | -define(TYPE_BITS, 16#07). |
| 59 | -define(TYPE_SHIFT_AMOUNT, 5). |
| 60 | |
| 61 | typeid_to_compact(?tType_STOP) -> 16#0; |
Nobuaki Sukegawa | e1b8540 | 2016-07-18 21:27:27 +0900 | [diff] [blame] | 62 | typeid_to_compact(?tType_BOOL) -> 16#2; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 63 | typeid_to_compact(?tType_I8) -> 16#3; |
| 64 | typeid_to_compact(?tType_I16) -> 16#4; |
| 65 | typeid_to_compact(?tType_I32) -> 16#5; |
| 66 | typeid_to_compact(?tType_I64) -> 16#6; |
Nobuaki Sukegawa | e1b8540 | 2016-07-18 21:27:27 +0900 | [diff] [blame] | 67 | typeid_to_compact(?tType_DOUBLE) -> 16#7; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 68 | typeid_to_compact(?tType_STRING) -> 16#8; |
| 69 | typeid_to_compact(?tType_STRUCT) -> 16#C; |
| 70 | typeid_to_compact(?tType_MAP) -> 16#B; |
| 71 | typeid_to_compact(?tType_SET) -> 16#A; |
| 72 | typeid_to_compact(?tType_LIST) -> 16#9. |
| 73 | |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 74 | compact_to_typeid(16#0) -> ?tType_STOP; |
| 75 | compact_to_typeid(?CBOOL_FALSE) -> ?tType_BOOL; |
| 76 | compact_to_typeid(?CBOOL_TRUE) -> ?tType_BOOL; |
| 77 | compact_to_typeid(16#7) -> ?tType_DOUBLE; |
| 78 | compact_to_typeid(16#3) -> ?tType_I8; |
| 79 | compact_to_typeid(16#4) -> ?tType_I16; |
| 80 | compact_to_typeid(16#5) -> ?tType_I32; |
| 81 | compact_to_typeid(16#6) -> ?tType_I64; |
| 82 | compact_to_typeid(16#8) -> ?tType_STRING; |
| 83 | compact_to_typeid(16#C) -> ?tType_STRUCT; |
| 84 | compact_to_typeid(16#B) -> ?tType_MAP; |
| 85 | compact_to_typeid(16#A) -> ?tType_SET; |
| 86 | compact_to_typeid(16#9) -> ?tType_LIST. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 87 | |
| 88 | bool_to_cbool(Value) when Value -> ?CBOOL_TRUE; |
| 89 | bool_to_cbool(_) -> ?CBOOL_FALSE. |
| 90 | cbool_to_bool(Value) -> Value =:= ?CBOOL_TRUE. |
| 91 | |
| 92 | new(Transport) -> new(Transport, _Options = []). |
| 93 | |
| 94 | new(Transport, _Options) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 95 | State = #t_compact{transport = Transport}, |
| 96 | thrift_protocol:new(?MODULE, State). |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 97 | |
| 98 | flush_transport(This = #t_compact{transport = Transport}) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 99 | {NewTransport, Result} = thrift_transport:flush(Transport), |
| 100 | {This#t_compact{transport = NewTransport}, Result}. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 101 | |
| 102 | close_transport(This = #t_compact{transport = Transport}) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 103 | {NewTransport, Result} = thrift_transport:close(Transport), |
| 104 | {This#t_compact{transport = NewTransport}, Result}. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 105 | |
| 106 | %%% |
| 107 | %%% instance methods |
| 108 | %%% |
| 109 | |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 110 | write_field_begin(This0 = #t_compact{write_stack = [LastId | T]}, CompactType, Id) -> |
| 111 | IdDiff = Id - LastId, |
| 112 | This1 = This0#t_compact{write_stack = [Id | T]}, |
| 113 | case (IdDiff > 0) and (IdDiff < 16) of |
| 114 | true -> |
| 115 | write(This1, {byte, (IdDiff bsl 4) bor CompactType}); |
| 116 | false -> |
| 117 | {This2, ok} = write(This1, {byte, CompactType}), |
| 118 | write(This2, {i16, Id}) |
| 119 | end. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 120 | |
| 121 | -spec to_zigzag(integer()) -> non_neg_integer(). |
Nobuaki Sukegawa | e1b8540 | 2016-07-18 21:27:27 +0900 | [diff] [blame] | 122 | to_zigzag(Value) -> 16#FFFFFFFFFFFFFFFF band ((Value bsl 1) bxor (Value bsr 63)). |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 123 | |
| 124 | -spec from_zigzag(non_neg_integer()) -> integer(). |
| 125 | from_zigzag(Value) -> (Value bsr 1) bxor -(Value band 1). |
| 126 | |
| 127 | -spec to_varint(non_neg_integer(), iolist()) -> iolist(). |
| 128 | to_varint(Value, Acc) when (Value < 16#80) -> [Acc, Value]; |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 129 | to_varint(Value, Acc) -> to_varint(Value bsr 7, [Acc, ((Value band 16#7F) bor 16#80)]). |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 130 | |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 131 | -spec read_varint(t_compact(), non_neg_integer(), non_neg_integer()) -> |
| 132 | {t_compact(), {'ok', integer()}}. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 133 | read_varint(This0, Acc, Count) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 134 | {This1, {ok, Byte}} = read(This0, byte), |
| 135 | case (Byte band 16#80) of |
| 136 | 0 -> {This1, {ok, (Byte bsl (7 * Count)) + Acc}}; |
| 137 | _ -> read_varint(This1, ((Byte band 16#7f) bsl (7 * Count)) + Acc, Count + 1) |
| 138 | end. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 139 | |
| 140 | write(This0, #protocol_message_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 141 | name = Name, |
| 142 | type = Type, |
| 143 | seqid = Seqid |
| 144 | }) -> |
| 145 | {This1, ok} = write(This0, {byte, ?PROTOCOL_ID}), |
| 146 | {This2, ok} = write( |
| 147 | This1, {byte, (?VERSION_1 band ?VERSION_MASK) bor (Type bsl ?TYPE_SHIFT_AMOUNT)} |
| 148 | ), |
| 149 | {This3, ok} = write(This2, {ui32, Seqid}), |
| 150 | {This4, ok} = write(This3, {string, Name}), |
| 151 | {This4, ok}; |
| 152 | write(This, message_end) -> |
| 153 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 154 | write(This0, #protocol_field_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 155 | name = _Name, |
| 156 | type = Type, |
| 157 | id = Id |
| 158 | }) when |
| 159 | (Type =:= ?tType_BOOL) |
| 160 | -> |
| 161 | {This0#t_compact{write_id = Id}, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 162 | write(This0, #protocol_field_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 163 | name = _Name, |
| 164 | type = Type, |
| 165 | id = Id |
| 166 | }) -> |
| 167 | write_field_begin(This0, typeid_to_compact(Type), Id); |
| 168 | write(This, field_stop) -> |
| 169 | write(This, {byte, ?tType_STOP}); |
| 170 | write(This, field_end) -> |
| 171 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 172 | write(This0, #protocol_map_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 173 | ktype = _Ktype, |
| 174 | vtype = _Vtype, |
| 175 | size = Size |
| 176 | }) when |
| 177 | Size =:= 0 |
| 178 | -> |
| 179 | write(This0, {byte, 0}); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 180 | write(This0, #protocol_map_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 181 | ktype = Ktype, |
| 182 | vtype = Vtype, |
| 183 | size = Size |
| 184 | }) -> |
| 185 | {This1, ok} = write(This0, {ui32, Size}), |
| 186 | write(This1, {byte, (typeid_to_compact(Ktype) bsl 4) bor typeid_to_compact(Vtype)}); |
| 187 | write(This, map_end) -> |
| 188 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 189 | write(This0, #protocol_list_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 190 | etype = Etype, |
| 191 | size = Size |
| 192 | }) when |
| 193 | Size < 16#f |
| 194 | -> |
| 195 | write(This0, {byte, (Size bsl 4) bor typeid_to_compact(Etype)}); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 196 | write(This0, #protocol_list_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 197 | etype = Etype, |
| 198 | size = Size |
| 199 | }) -> |
| 200 | {This1, ok} = write(This0, {byte, 16#f0 bor typeid_to_compact(Etype)}), |
| 201 | write(This1, {ui32, Size}); |
| 202 | write(This, list_end) -> |
| 203 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 204 | write(This0, #protocol_set_begin{ |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 205 | etype = Etype, |
| 206 | size = Size |
| 207 | }) -> |
| 208 | write(This0, #protocol_list_begin{etype = Etype, size = Size}); |
| 209 | write(This, set_end) -> |
| 210 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 211 | write(This = #t_compact{write_stack = Stack}, #protocol_struct_begin{}) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 212 | {This#t_compact{write_stack = [0 | Stack]}, ok}; |
| 213 | write(This = #t_compact{write_stack = [_ | T]}, struct_end) -> |
| 214 | {This#t_compact{write_stack = T}, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 215 | write(This = #t_compact{write_id = ?ID_NONE}, {bool, Value}) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 216 | write(This, {byte, bool_to_cbool(Value)}); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 217 | write(This0 = #t_compact{write_id = Id}, {bool, Value}) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 218 | {This1, ok} = write_field_begin(This0, bool_to_cbool(Value), Id), |
| 219 | {This1#t_compact{write_id = ?ID_NONE}, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 220 | write(This, {byte, Value}) when is_integer(Value) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 221 | write(This, <<Value:8/big-signed>>); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 222 | write(This, {i16, Value}) when is_integer(Value) -> write(This, to_varint(to_zigzag(Value), [])); |
| 223 | write(This, {ui32, Value}) when is_integer(Value) -> write(This, to_varint(Value, [])); |
| 224 | write(This, {i32, Value}) when is_integer(Value) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 225 | write(This, to_varint(to_zigzag(Value), [])); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 226 | write(This, {i64, Value}) when is_integer(Value) -> write(This, to_varint(to_zigzag(Value), [])); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 227 | write(This, {double, Double}) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 228 | write(This, <<Double:64/float-signed-little>>); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 229 | write(This0, {string, Str}) when is_list(Str) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 230 | % TODO: limit length |
| 231 | {This1, ok} = write(This0, {ui32, length(Str)}), |
| 232 | {This2, ok} = write(This1, list_to_binary(Str)), |
| 233 | {This2, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 234 | write(This0, {string, Bin}) when is_binary(Bin) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 235 | % TODO: limit length |
| 236 | {This1, ok} = write(This0, {ui32, size(Bin)}), |
| 237 | {This2, ok} = write(This1, Bin), |
| 238 | {This2, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 239 | %% Data :: iolist() |
| 240 | write(This = #t_compact{transport = Trans}, Data) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 241 | {NewTransport, Result} = thrift_transport:write(Trans, Data), |
| 242 | {This#t_compact{transport = NewTransport}, Result}. |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 243 | |
| 244 | %% |
| 245 | %% |
| 246 | |
| 247 | read(This0, message_begin) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 248 | {This1, {ok, ?PROTOCOL_ID}} = read(This0, ubyte), |
| 249 | {This2, {ok, VerAndType}} = read(This1, ubyte), |
| 250 | ?VERSION_1 = VerAndType band ?VERSION_MASK, |
| 251 | {This3, {ok, SeqId}} = read(This2, ui32), |
| 252 | {This4, {ok, Name}} = read(This3, string), |
| 253 | {This4, #protocol_message_begin{ |
| 254 | name = binary_to_list(Name), |
| 255 | type = (VerAndType bsr ?TYPE_SHIFT_AMOUNT) band ?TYPE_BITS, |
| 256 | seqid = SeqId |
| 257 | }}; |
| 258 | read(This, message_end) -> |
| 259 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 260 | read(This = #t_compact{read_stack = Stack}, struct_begin) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 261 | {This#t_compact{read_stack = [0 | Stack]}, ok}; |
| 262 | read(This = #t_compact{read_stack = [_H | T]}, struct_end) -> |
| 263 | {This#t_compact{read_stack = T}, ok}; |
| 264 | read(This0 = #t_compact{read_stack = [LastId | T]}, field_begin) -> |
| 265 | {This1, {ok, Byte}} = read(This0, ubyte), |
| 266 | case Byte band 16#f of |
| 267 | CompactType = ?tType_STOP -> |
| 268 | {This1, #protocol_field_begin{type = CompactType}}; |
| 269 | CompactType -> |
| 270 | {This2, {ok, Id}} = |
| 271 | case Byte bsr 4 of |
| 272 | 0 -> read(This1, i16); |
| 273 | IdDiff -> {This1, {ok, LastId + IdDiff}} |
| 274 | end, |
| 275 | case compact_to_typeid(CompactType) of |
| 276 | ?tType_BOOL -> |
| 277 | { |
| 278 | This2#t_compact{ |
| 279 | read_stack = [Id | T], read_value = CompactType |
| 280 | }, |
| 281 | #protocol_field_begin{type = ?tType_BOOL, id = Id} |
| 282 | }; |
| 283 | Type -> |
| 284 | {This2#t_compact{read_stack = [Id | T]}, #protocol_field_begin{ |
| 285 | type = Type, id = Id |
| 286 | }} |
| 287 | end |
| 288 | end; |
| 289 | read(This, field_end) -> |
| 290 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 291 | read(This0, map_begin) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 292 | {This1, {ok, Size}} = read(This0, ui32), |
| 293 | {This2, {ok, KV}} = |
| 294 | case Size of |
| 295 | 0 -> {This1, {ok, 0}}; |
| 296 | _ -> read(This1, ubyte) |
| 297 | end, |
| 298 | {This2, #protocol_map_begin{ |
| 299 | ktype = compact_to_typeid(KV bsr 4), |
| 300 | vtype = compact_to_typeid(KV band 16#f), |
| 301 | size = Size |
| 302 | }}; |
| 303 | read(This, map_end) -> |
| 304 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 305 | read(This0, list_begin) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 306 | {This1, {ok, SizeAndType}} = read(This0, ubyte), |
| 307 | {This2, {ok, Size}} = |
| 308 | case (SizeAndType bsr 4) band 16#f of |
| 309 | 16#f -> read(This1, ui32); |
| 310 | Else -> {This1, {ok, Else}} |
| 311 | end, |
| 312 | {This2, #protocol_list_begin{ |
| 313 | etype = compact_to_typeid(SizeAndType band 16#f), |
| 314 | size = Size |
| 315 | }}; |
| 316 | read(This, list_end) -> |
| 317 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 318 | read(This0, set_begin) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 319 | {This1, {ok, SizeAndType}} = read(This0, ubyte), |
| 320 | {This2, {ok, Size}} = |
| 321 | case (SizeAndType bsr 4) band 16#f of |
| 322 | 16#f -> read(This1, ui32); |
| 323 | Else -> {This1, {ok, Else}} |
| 324 | end, |
| 325 | {This2, #protocol_set_begin{ |
| 326 | etype = compact_to_typeid(SizeAndType band 16#f), |
| 327 | size = Size |
| 328 | }}; |
| 329 | read(This, set_end) -> |
| 330 | {This, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 331 | read(This0, field_stop) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 332 | {This1, {ok, ?tType_STOP}} = read(This0, ubyte), |
| 333 | {This1, ok}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 334 | %% |
| 335 | |
| 336 | read(This0 = #t_compact{read_value = ?CBOOL_NONE}, bool) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 337 | {This1, {ok, Byte}} = read(This0, ubyte), |
| 338 | {This1, {ok, cbool_to_bool(Byte)}}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 339 | read(This0 = #t_compact{read_value = Bool}, bool) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 340 | {This0#t_compact{read_value = ?CBOOL_NONE}, {ok, cbool_to_bool(Bool)}}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 341 | read(This0, ubyte) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 342 | {This1, {ok, <<Val:8/integer-unsigned-big, _/binary>>}} = read_data(This0, 1), |
| 343 | {This1, {ok, Val}}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 344 | read(This0, byte) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 345 | {This1, Bytes} = read_data(This0, 1), |
| 346 | case Bytes of |
| 347 | {ok, <<Val:8/integer-signed-big, _/binary>>} -> {This1, {ok, Val}}; |
| 348 | Else -> {This1, Else} |
| 349 | end; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 350 | read(This0, i16) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 351 | {This1, {ok, Zigzag}} = read_varint(This0, 0, 0), |
| 352 | {This1, {ok, from_zigzag(Zigzag)}}; |
| 353 | read(This0, ui32) -> |
| 354 | read_varint(This0, 0, 0); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 355 | read(This0, i32) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 356 | {This1, {ok, Zigzag}} = read_varint(This0, 0, 0), |
| 357 | {This1, {ok, from_zigzag(Zigzag)}}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 358 | read(This0, i64) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 359 | {This1, {ok, Zigzag}} = read_varint(This0, 0, 0), |
| 360 | {This1, {ok, from_zigzag(Zigzag)}}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 361 | read(This0, double) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 362 | {This1, Bytes} = read_data(This0, 8), |
| 363 | case Bytes of |
| 364 | {ok, <<Val:64/float-signed-little, _/binary>>} -> {This1, {ok, Val}}; |
| 365 | Else -> {This1, Else} |
| 366 | end; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 367 | % returns a binary directly, call binary_to_list if necessary |
| 368 | read(This0, string) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 369 | {This1, {ok, Sz}} = read(This0, ui32), |
| 370 | read_data(This1, Sz). |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 371 | |
| 372 | -spec read_data(#t_compact{}, non_neg_integer()) -> |
| 373 | {#t_compact{}, {ok, binary()} | {error, _Reason}}. |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 374 | read_data(This, 0) -> |
| 375 | {This, {ok, <<>>}}; |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 376 | read_data(This = #t_compact{transport = Trans}, Len) when is_integer(Len) andalso Len > 0 -> |
| 377 | {NewTransport, Result} = thrift_transport:read(Trans, Len), |
| 378 | {This#t_compact{transport = NewTransport}, Result}. |
| 379 | |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 380 | %%%% FACTORY GENERATION %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 381 | |
| 382 | %% returns a (fun() -> thrift_protocol()) |
| 383 | new_protocol_factory(TransportFactory, _Options) -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 384 | F = fun() -> |
| 385 | case TransportFactory() of |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 386 | {ok, Transport} -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 387 | thrift_compact_protocol:new( |
| 388 | Transport, |
| 389 | [] |
| 390 | ); |
Nobuaki Sukegawa | b31f090 | 2015-11-01 17:00:34 +0900 | [diff] [blame] | 391 | {error, Error} -> |
Sergei Elin | 4576409 | 2022-09-23 23:21:31 +0300 | [diff] [blame] | 392 | {error, Error} |
| 393 | end |
| 394 | end, |
| 395 | {ok, F}. |