| %% |
| %% 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. |
| %% |
| |
| -module(thrift_compact_protocol). |
| |
| -behaviour(thrift_protocol). |
| |
| -include("thrift_constants.hrl"). |
| -include("thrift_protocol.hrl"). |
| |
| -export([ |
| new/1, new/2, |
| read/2, |
| write/2, |
| flush_transport/1, |
| close_transport/1, |
| new_protocol_factory/2 |
| ]). |
| |
| -define(ID_NONE, 16#10000). |
| -define(CBOOL_NONE, 0). |
| -define(CBOOL_TRUE, 1). |
| -define(CBOOL_FALSE, 2). |
| |
| -type cbool() :: ?CBOOL_NONE | ?CBOOL_TRUE | ?CBOOL_FALSE. |
| |
| -record(t_compact, { |
| transport :: term(), |
| % state for pending boolean fields |
| read_stack = [] :: iodata(), |
| read_value = ?CBOOL_NONE :: cbool(), |
| write_stack = [] :: iodata(), |
| write_id = ?ID_NONE :: non_neg_integer() |
| }). |
| |
| -type t_compact() :: #t_compact{}. |
| |
| -define(PROTOCOL_ID, 16#82). |
| -define(VERSION_MASK, 16#1f). |
| -define(VERSION_1, 16#01). |
| -define(TYPE_MASK, 16#E0). |
| -define(TYPE_BITS, 16#07). |
| -define(TYPE_SHIFT_AMOUNT, 5). |
| |
| typeid_to_compact(?tType_STOP) -> 16#0; |
| typeid_to_compact(?tType_BOOL) -> 16#2; |
| typeid_to_compact(?tType_I8) -> 16#3; |
| typeid_to_compact(?tType_I16) -> 16#4; |
| typeid_to_compact(?tType_I32) -> 16#5; |
| typeid_to_compact(?tType_I64) -> 16#6; |
| typeid_to_compact(?tType_DOUBLE) -> 16#7; |
| typeid_to_compact(?tType_STRING) -> 16#8; |
| typeid_to_compact(?tType_STRUCT) -> 16#C; |
| typeid_to_compact(?tType_MAP) -> 16#B; |
| typeid_to_compact(?tType_SET) -> 16#A; |
| typeid_to_compact(?tType_LIST) -> 16#9. |
| |
| compact_to_typeid(16#0) -> ?tType_STOP; |
| compact_to_typeid(?CBOOL_FALSE) -> ?tType_BOOL; |
| compact_to_typeid(?CBOOL_TRUE) -> ?tType_BOOL; |
| compact_to_typeid(16#7) -> ?tType_DOUBLE; |
| compact_to_typeid(16#3) -> ?tType_I8; |
| compact_to_typeid(16#4) -> ?tType_I16; |
| compact_to_typeid(16#5) -> ?tType_I32; |
| compact_to_typeid(16#6) -> ?tType_I64; |
| compact_to_typeid(16#8) -> ?tType_STRING; |
| compact_to_typeid(16#C) -> ?tType_STRUCT; |
| compact_to_typeid(16#B) -> ?tType_MAP; |
| compact_to_typeid(16#A) -> ?tType_SET; |
| compact_to_typeid(16#9) -> ?tType_LIST. |
| |
| bool_to_cbool(Value) when Value -> ?CBOOL_TRUE; |
| bool_to_cbool(_) -> ?CBOOL_FALSE. |
| cbool_to_bool(Value) -> Value =:= ?CBOOL_TRUE. |
| |
| new(Transport) -> new(Transport, _Options = []). |
| |
| new(Transport, _Options) -> |
| State = #t_compact{transport = Transport}, |
| thrift_protocol:new(?MODULE, State). |
| |
| flush_transport(This = #t_compact{transport = Transport}) -> |
| {NewTransport, Result} = thrift_transport:flush(Transport), |
| {This#t_compact{transport = NewTransport}, Result}. |
| |
| close_transport(This = #t_compact{transport = Transport}) -> |
| {NewTransport, Result} = thrift_transport:close(Transport), |
| {This#t_compact{transport = NewTransport}, Result}. |
| |
| %%% |
| %%% instance methods |
| %%% |
| |
| write_field_begin(This0 = #t_compact{write_stack = [LastId | T]}, CompactType, Id) -> |
| IdDiff = Id - LastId, |
| This1 = This0#t_compact{write_stack = [Id | T]}, |
| case (IdDiff > 0) and (IdDiff < 16) of |
| true -> |
| write(This1, {byte, (IdDiff bsl 4) bor CompactType}); |
| false -> |
| {This2, ok} = write(This1, {byte, CompactType}), |
| write(This2, {i16, Id}) |
| end. |
| |
| -spec to_zigzag(integer()) -> non_neg_integer(). |
| to_zigzag(Value) -> 16#FFFFFFFFFFFFFFFF band ((Value bsl 1) bxor (Value bsr 63)). |
| |
| -spec from_zigzag(non_neg_integer()) -> integer(). |
| from_zigzag(Value) -> (Value bsr 1) bxor -(Value band 1). |
| |
| -spec to_varint(non_neg_integer(), iolist()) -> iolist(). |
| to_varint(Value, Acc) when (Value < 16#80) -> [Acc, Value]; |
| to_varint(Value, Acc) -> to_varint(Value bsr 7, [Acc, ((Value band 16#7F) bor 16#80)]). |
| |
| -spec read_varint(t_compact(), non_neg_integer(), non_neg_integer()) -> |
| {t_compact(), {'ok', integer()}}. |
| read_varint(This0, Acc, Count) -> |
| {This1, {ok, Byte}} = read(This0, byte), |
| case (Byte band 16#80) of |
| 0 -> {This1, {ok, (Byte bsl (7 * Count)) + Acc}}; |
| _ -> read_varint(This1, ((Byte band 16#7f) bsl (7 * Count)) + Acc, Count + 1) |
| end. |
| |
| write(This0, #protocol_message_begin{ |
| name = Name, |
| type = Type, |
| seqid = Seqid |
| }) -> |
| {This1, ok} = write(This0, {byte, ?PROTOCOL_ID}), |
| {This2, ok} = write( |
| This1, {byte, (?VERSION_1 band ?VERSION_MASK) bor (Type bsl ?TYPE_SHIFT_AMOUNT)} |
| ), |
| {This3, ok} = write(This2, {ui32, Seqid}), |
| {This4, ok} = write(This3, {string, Name}), |
| {This4, ok}; |
| write(This, message_end) -> |
| {This, ok}; |
| write(This0, #protocol_field_begin{ |
| name = _Name, |
| type = Type, |
| id = Id |
| }) when |
| (Type =:= ?tType_BOOL) |
| -> |
| {This0#t_compact{write_id = Id}, ok}; |
| write(This0, #protocol_field_begin{ |
| name = _Name, |
| type = Type, |
| id = Id |
| }) -> |
| write_field_begin(This0, typeid_to_compact(Type), Id); |
| write(This, field_stop) -> |
| write(This, {byte, ?tType_STOP}); |
| write(This, field_end) -> |
| {This, ok}; |
| write(This0, #protocol_map_begin{ |
| ktype = _Ktype, |
| vtype = _Vtype, |
| size = Size |
| }) when |
| Size =:= 0 |
| -> |
| write(This0, {byte, 0}); |
| write(This0, #protocol_map_begin{ |
| ktype = Ktype, |
| vtype = Vtype, |
| size = Size |
| }) -> |
| {This1, ok} = write(This0, {ui32, Size}), |
| write(This1, {byte, (typeid_to_compact(Ktype) bsl 4) bor typeid_to_compact(Vtype)}); |
| write(This, map_end) -> |
| {This, ok}; |
| write(This0, #protocol_list_begin{ |
| etype = Etype, |
| size = Size |
| }) when |
| Size < 16#f |
| -> |
| write(This0, {byte, (Size bsl 4) bor typeid_to_compact(Etype)}); |
| write(This0, #protocol_list_begin{ |
| etype = Etype, |
| size = Size |
| }) -> |
| {This1, ok} = write(This0, {byte, 16#f0 bor typeid_to_compact(Etype)}), |
| write(This1, {ui32, Size}); |
| write(This, list_end) -> |
| {This, ok}; |
| write(This0, #protocol_set_begin{ |
| etype = Etype, |
| size = Size |
| }) -> |
| write(This0, #protocol_list_begin{etype = Etype, size = Size}); |
| write(This, set_end) -> |
| {This, ok}; |
| write(This = #t_compact{write_stack = Stack}, #protocol_struct_begin{}) -> |
| {This#t_compact{write_stack = [0 | Stack]}, ok}; |
| write(This = #t_compact{write_stack = [_ | T]}, struct_end) -> |
| {This#t_compact{write_stack = T}, ok}; |
| write(This = #t_compact{write_id = ?ID_NONE}, {bool, Value}) -> |
| write(This, {byte, bool_to_cbool(Value)}); |
| write(This0 = #t_compact{write_id = Id}, {bool, Value}) -> |
| {This1, ok} = write_field_begin(This0, bool_to_cbool(Value), Id), |
| {This1#t_compact{write_id = ?ID_NONE}, ok}; |
| write(This, {byte, Value}) when is_integer(Value) -> |
| write(This, <<Value:8/big-signed>>); |
| write(This, {i16, Value}) when is_integer(Value) -> write(This, to_varint(to_zigzag(Value), [])); |
| write(This, {ui32, Value}) when is_integer(Value) -> write(This, to_varint(Value, [])); |
| write(This, {i32, Value}) when is_integer(Value) -> |
| write(This, to_varint(to_zigzag(Value), [])); |
| write(This, {i64, Value}) when is_integer(Value) -> write(This, to_varint(to_zigzag(Value), [])); |
| write(This, {double, Double}) -> |
| write(This, <<Double:64/float-signed-little>>); |
| write(This0, {string, Str}) when is_list(Str) -> |
| % TODO: limit length |
| {This1, ok} = write(This0, {ui32, length(Str)}), |
| {This2, ok} = write(This1, list_to_binary(Str)), |
| {This2, ok}; |
| write(This0, {string, Bin}) when is_binary(Bin) -> |
| % TODO: limit length |
| {This1, ok} = write(This0, {ui32, size(Bin)}), |
| {This2, ok} = write(This1, Bin), |
| {This2, ok}; |
| %% Data :: iolist() |
| write(This = #t_compact{transport = Trans}, Data) -> |
| {NewTransport, Result} = thrift_transport:write(Trans, Data), |
| {This#t_compact{transport = NewTransport}, Result}. |
| |
| %% |
| %% |
| |
| read(This0, message_begin) -> |
| {This1, {ok, ?PROTOCOL_ID}} = read(This0, ubyte), |
| {This2, {ok, VerAndType}} = read(This1, ubyte), |
| ?VERSION_1 = VerAndType band ?VERSION_MASK, |
| {This3, {ok, SeqId}} = read(This2, ui32), |
| {This4, {ok, Name}} = read(This3, string), |
| {This4, #protocol_message_begin{ |
| name = binary_to_list(Name), |
| type = (VerAndType bsr ?TYPE_SHIFT_AMOUNT) band ?TYPE_BITS, |
| seqid = SeqId |
| }}; |
| read(This, message_end) -> |
| {This, ok}; |
| read(This = #t_compact{read_stack = Stack}, struct_begin) -> |
| {This#t_compact{read_stack = [0 | Stack]}, ok}; |
| read(This = #t_compact{read_stack = [_H | T]}, struct_end) -> |
| {This#t_compact{read_stack = T}, ok}; |
| read(This0 = #t_compact{read_stack = [LastId | T]}, field_begin) -> |
| {This1, {ok, Byte}} = read(This0, ubyte), |
| case Byte band 16#f of |
| CompactType = ?tType_STOP -> |
| {This1, #protocol_field_begin{type = CompactType}}; |
| CompactType -> |
| {This2, {ok, Id}} = |
| case Byte bsr 4 of |
| 0 -> read(This1, i16); |
| IdDiff -> {This1, {ok, LastId + IdDiff}} |
| end, |
| case compact_to_typeid(CompactType) of |
| ?tType_BOOL -> |
| { |
| This2#t_compact{ |
| read_stack = [Id | T], read_value = CompactType |
| }, |
| #protocol_field_begin{type = ?tType_BOOL, id = Id} |
| }; |
| Type -> |
| {This2#t_compact{read_stack = [Id | T]}, #protocol_field_begin{ |
| type = Type, id = Id |
| }} |
| end |
| end; |
| read(This, field_end) -> |
| {This, ok}; |
| read(This0, map_begin) -> |
| {This1, {ok, Size}} = read(This0, ui32), |
| {This2, {ok, KV}} = |
| case Size of |
| 0 -> {This1, {ok, 0}}; |
| _ -> read(This1, ubyte) |
| end, |
| {This2, #protocol_map_begin{ |
| ktype = compact_to_typeid(KV bsr 4), |
| vtype = compact_to_typeid(KV band 16#f), |
| size = Size |
| }}; |
| read(This, map_end) -> |
| {This, ok}; |
| read(This0, list_begin) -> |
| {This1, {ok, SizeAndType}} = read(This0, ubyte), |
| {This2, {ok, Size}} = |
| case (SizeAndType bsr 4) band 16#f of |
| 16#f -> read(This1, ui32); |
| Else -> {This1, {ok, Else}} |
| end, |
| {This2, #protocol_list_begin{ |
| etype = compact_to_typeid(SizeAndType band 16#f), |
| size = Size |
| }}; |
| read(This, list_end) -> |
| {This, ok}; |
| read(This0, set_begin) -> |
| {This1, {ok, SizeAndType}} = read(This0, ubyte), |
| {This2, {ok, Size}} = |
| case (SizeAndType bsr 4) band 16#f of |
| 16#f -> read(This1, ui32); |
| Else -> {This1, {ok, Else}} |
| end, |
| {This2, #protocol_set_begin{ |
| etype = compact_to_typeid(SizeAndType band 16#f), |
| size = Size |
| }}; |
| read(This, set_end) -> |
| {This, ok}; |
| read(This0, field_stop) -> |
| {This1, {ok, ?tType_STOP}} = read(This0, ubyte), |
| {This1, ok}; |
| %% |
| |
| read(This0 = #t_compact{read_value = ?CBOOL_NONE}, bool) -> |
| {This1, {ok, Byte}} = read(This0, ubyte), |
| {This1, {ok, cbool_to_bool(Byte)}}; |
| read(This0 = #t_compact{read_value = Bool}, bool) -> |
| {This0#t_compact{read_value = ?CBOOL_NONE}, {ok, cbool_to_bool(Bool)}}; |
| read(This0, ubyte) -> |
| {This1, {ok, <<Val:8/integer-unsigned-big, _/binary>>}} = read_data(This0, 1), |
| {This1, {ok, Val}}; |
| read(This0, byte) -> |
| {This1, Bytes} = read_data(This0, 1), |
| case Bytes of |
| {ok, <<Val:8/integer-signed-big, _/binary>>} -> {This1, {ok, Val}}; |
| Else -> {This1, Else} |
| end; |
| read(This0, i16) -> |
| {This1, {ok, Zigzag}} = read_varint(This0, 0, 0), |
| {This1, {ok, from_zigzag(Zigzag)}}; |
| read(This0, ui32) -> |
| read_varint(This0, 0, 0); |
| read(This0, i32) -> |
| {This1, {ok, Zigzag}} = read_varint(This0, 0, 0), |
| {This1, {ok, from_zigzag(Zigzag)}}; |
| read(This0, i64) -> |
| {This1, {ok, Zigzag}} = read_varint(This0, 0, 0), |
| {This1, {ok, from_zigzag(Zigzag)}}; |
| read(This0, double) -> |
| {This1, Bytes} = read_data(This0, 8), |
| case Bytes of |
| {ok, <<Val:64/float-signed-little, _/binary>>} -> {This1, {ok, Val}}; |
| Else -> {This1, Else} |
| end; |
| % returns a binary directly, call binary_to_list if necessary |
| read(This0, string) -> |
| {This1, {ok, Sz}} = read(This0, ui32), |
| read_data(This1, Sz). |
| |
| -spec read_data(#t_compact{}, non_neg_integer()) -> |
| {#t_compact{}, {ok, binary()} | {error, _Reason}}. |
| read_data(This, 0) -> |
| {This, {ok, <<>>}}; |
| read_data(This = #t_compact{transport = Trans}, Len) when is_integer(Len) andalso Len > 0 -> |
| {NewTransport, Result} = thrift_transport:read(Trans, Len), |
| {This#t_compact{transport = NewTransport}, Result}. |
| |
| %%%% FACTORY GENERATION %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| |
| %% returns a (fun() -> thrift_protocol()) |
| new_protocol_factory(TransportFactory, _Options) -> |
| F = fun() -> |
| case TransportFactory() of |
| {ok, Transport} -> |
| thrift_compact_protocol:new( |
| Transport, |
| [] |
| ); |
| {error, Error} -> |
| {error, Error} |
| end |
| end, |
| {ok, F}. |