/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * Licensed 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. */ #ifndef THRIFT_PROTOCOL_TPROTOCOL_H_ #define THRIFT_PROTOCOL_TPROTOCOL_H_ 1 #include #include #include #include #include #include #include #include #include #include namespace apache { namespace thrift { namespace reflection { class Schema; } } // namespace thrift } // namespace apache namespace apache { namespace thrift { namespace protocol { using apache::thrift::transport::TTransport; /** * Enumerated definition of the message types that the Thrift protocol * supports. */ enum TMessageType { T_CALL = 1, T_REPLY = 2, T_EXCEPTION = 3, T_ONEWAY = 4 }; /** * Helper template for implementing TProtocol::skip(). * * Templatized to avoid having to make virtual function calls. */ template uint32_t skip(Protocol_& prot, TType arg_type) { switch (arg_type) { case T_BOOL: { bool boolv; return prot.readBool(boolv); } // case T_I08: // same numeric value as T_BYTE case T_BYTE: { int8_t bytev = 0; return prot.readByte(bytev); } case T_I16: { int16_t i16; return prot.readI16(i16); } case T_I32: { int32_t i32; return prot.readI32(i32); } case T_U64: case T_I64: { int64_t i64; return prot.readI64(i64); } case T_DOUBLE: { double dub; return prot.readDouble(dub); } case T_FLOAT: { float flt; return prot.readFloat(flt); } // case T_UTF7: // same numeric value as T_STRING case T_UTF8: case T_UTF16: case T_STRING: { std::string str; return prot.readBinary(str); } case T_STRUCT: { uint32_t result = 0; std::string name; int16_t fid; TType ftype; result += prot.readStructBegin(name); while (true) { result += prot.readFieldBegin(name, ftype, fid); if (ftype == T_STOP) { break; } result += skip(prot, ftype); result += prot.readFieldEnd(); } result += prot.readStructEnd(); return result; } case T_MAP: { uint32_t result = 0; TType keyType; TType valType; uint32_t i, size; bool sizeUnknown; result += prot.readMapBegin(keyType, valType, size, sizeUnknown); if (!sizeUnknown) { for (i = 0; i < size; i++) { result += skip(prot, keyType); result += skip(prot, valType); } } else { while (prot.peekMap()) { result += skip(prot, keyType); result += skip(prot, valType); } } result += prot.readMapEnd(); return result; } case T_SET: { uint32_t result = 0; TType elemType; uint32_t i, size; bool sizeUnknown; result += prot.readSetBegin(elemType, size, sizeUnknown); if (!sizeUnknown) { for (i = 0; i < size; i++) { result += skip(prot, elemType); } } else { while (prot.peekSet()) { result += skip(prot, elemType); } } result += prot.readSetEnd(); return result; } case T_LIST: { uint32_t result = 0; TType elemType; uint32_t i, size; bool sizeUnknown; result += prot.readListBegin(elemType, size, sizeUnknown); if (!sizeUnknown) { for (i = 0; i < size; i++) { result += skip(prot, elemType); } } else { while (prot.peekList()) { result += skip(prot, elemType); } } result += prot.readListEnd(); return result; } case T_STOP: case T_VOID: case T_STREAM: // Unimplemented, fallback to default default: { TProtocolException::throwInvalidSkipType(arg_type); } } } /** * Abstract class for a thrift protocol driver. These are all the methods that * a protocol must implement. Essentially, there must be some way of reading * and writing all the base types, plus a mechanism for writing out structs * with indexed fields. * * TProtocol objects should not be shared across multiple encoding contexts, * as they may need to maintain internal state in some protocols (i.e. XML). * Note that it is acceptable for the TProtocol module to do its own internal * buffered reads/writes to the underlying TTransport where appropriate (i.e. * when parsing an input XML stream, reading should be batched rather than * looking ahead character by character for a close tag). * */ class TProtocol { public: virtual ~TProtocol() {} virtual void setVersion_virt(const int8_t version) = 0; void setVersion(const int8_t version) { return setVersion_virt(version); } virtual ::apache::thrift::reflection::Schema* getSchema_virt() = 0; ::apache::thrift::reflection::Schema* getSchema() { return getSchema_virt(); } /** * Writing functions. */ virtual uint32_t writeMessageBegin_virt( const std::string& name, const TMessageType messageType, const int32_t seqid) = 0; virtual uint32_t writeMessageEnd_virt() = 0; virtual uint32_t writeStructBegin_virt(const char* name) = 0; virtual uint32_t writeStructEnd_virt() = 0; virtual uint32_t writeFieldBegin_virt( const char* name, const TType fieldType, const int16_t fieldId) = 0; virtual uint32_t writeFieldEnd_virt() = 0; virtual uint32_t writeFieldStop_virt() = 0; virtual uint32_t writeMapBegin_virt( const TType keyType, const TType valType, const uint32_t size) = 0; virtual uint32_t writeMapEnd_virt() = 0; virtual uint32_t writeListBegin_virt( const TType elemType, const uint32_t size) = 0; virtual uint32_t writeListEnd_virt() = 0; virtual uint32_t writeSetBegin_virt( const TType elemType, const uint32_t size) = 0; virtual uint32_t writeSetEnd_virt() = 0; virtual uint32_t writeBool_virt(const bool value) = 0; virtual uint32_t writeByte_virt(const int8_t byte) = 0; virtual uint32_t writeI16_virt(const int16_t i16) = 0; virtual uint32_t writeI32_virt(const int32_t i32) = 0; virtual uint32_t writeI64_virt(const int64_t i64) = 0; virtual uint32_t writeDouble_virt(const double dub) = 0; virtual uint32_t writeFloat_virt(const float flt) = 0; virtual uint32_t writeString_virt(const std::string& str) = 0; virtual uint32_t writeBinary_virt(const std::string& str) = 0; uint32_t writeMessageBegin( const std::string& name, const TMessageType messageType, const int32_t seqid) { return writeMessageBegin_virt(name, messageType, seqid); } uint32_t writeMessageEnd() { return writeMessageEnd_virt(); } uint32_t writeStructBegin(const char* name) { return writeStructBegin_virt(name); } uint32_t writeStructEnd() { return writeStructEnd_virt(); } uint32_t writeFieldBegin( const char* name, const TType fieldType, const int16_t fieldId) { return writeFieldBegin_virt(name, fieldType, fieldId); } uint32_t writeFieldEnd() { return writeFieldEnd_virt(); } uint32_t writeFieldStop() { return writeFieldStop_virt(); } uint32_t writeMapBegin( const TType keyType, const TType valType, const uint32_t size) { return writeMapBegin_virt(keyType, valType, size); } uint32_t writeMapEnd() { return writeMapEnd_virt(); } uint32_t writeListBegin(const TType elemType, const uint32_t size) { return writeListBegin_virt(elemType, size); } uint32_t writeListEnd() { return writeListEnd_virt(); } uint32_t writeSetBegin(const TType elemType, const uint32_t size) { return writeSetBegin_virt(elemType, size); } uint32_t writeSetEnd() { return writeSetEnd_virt(); } uint32_t writeBool(const bool value) { return writeBool_virt(value); } uint32_t writeByte(const int8_t byte) { return writeByte_virt(byte); } uint32_t writeI16(const int16_t i16) { return writeI16_virt(i16); } uint32_t writeI32(const int32_t i32) { return writeI32_virt(i32); } uint32_t writeI64(const int64_t i64) { return writeI64_virt(i64); } uint32_t writeDouble(const double dub) { return writeDouble_virt(dub); } uint32_t writeFloat(const float flt) { return writeFloat_virt(flt); } uint32_t writeString(const std::string& str) { return writeString_virt(str); } uint32_t writeString(const folly::fbstring& str) { return writeString_virt(str.toStdString()); } uint32_t writeBinary(const std::string& str) { return writeBinary_virt(str); } uint32_t writeBinary(const folly::fbstring& str) { return writeBinary_virt(str.toStdString()); } /** * Reading functions */ virtual uint32_t readMessageBegin_virt( std::string& name, TMessageType& messageType, int32_t& seqid) = 0; virtual uint32_t readMessageEnd_virt() = 0; virtual void setNextStructType_virt(uint64_t reflection_id) = 0; virtual uint32_t readStructBegin_virt(std::string& name) = 0; virtual uint32_t readStructEnd_virt() = 0; virtual uint32_t readFieldBegin_virt( std::string& name, TType& fieldType, int16_t& fieldId) = 0; virtual uint32_t readFieldEnd_virt() = 0; virtual uint32_t readMapBegin_virt( TType& keyType, TType& valType, uint32_t& size, bool& sizeUnknown) = 0; virtual bool peekMap_virt() = 0; virtual uint32_t readMapEnd_virt() = 0; virtual uint32_t readListBegin_virt( TType& elemType, uint32_t& size, bool& sizeUnknown) = 0; virtual bool peekList_virt() = 0; virtual uint32_t readListEnd_virt() = 0; virtual uint32_t readSetBegin_virt( TType& elemType, uint32_t& size, bool& sizeUnknown) = 0; virtual bool peekSet_virt() = 0; virtual uint32_t readSetEnd_virt() = 0; virtual uint32_t readBool_virt(bool& value) = 0; virtual uint32_t readBool_virt(std::vector::reference value) = 0; virtual uint32_t readByte_virt(int8_t& byte) = 0; virtual uint32_t readI16_virt(int16_t& i16) = 0; virtual uint32_t readI32_virt(int32_t& i32) = 0; virtual uint32_t readI64_virt(int64_t& i64) = 0; virtual uint32_t readDouble_virt(double& dub) = 0; virtual uint32_t readFloat_virt(float& flt) = 0; virtual uint32_t readString_virt(std::string& str) = 0; virtual uint32_t readBinary_virt(std::string& str) = 0; uint32_t readMessageBegin( std::string& name, TMessageType& messageType, int32_t& seqid) { return readMessageBegin_virt(name, messageType, seqid); } uint32_t readMessageEnd() { return readMessageEnd_virt(); } void setNextStructType(uint64_t reflection_id) { return setNextStructType_virt(reflection_id); } uint32_t readStructBegin(std::string& name) { return readStructBegin_virt(name); } uint32_t readStructEnd() { return readStructEnd_virt(); } uint32_t readFieldBegin( std::string& name, TType& fieldType, int16_t& fieldId) { return readFieldBegin_virt(name, fieldType, fieldId); } uint32_t readFieldEnd() { return readFieldEnd_virt(); } uint32_t readMapBegin( TType& keyType, TType& valType, uint32_t& size, bool& sizeUnknown) { return readMapBegin_virt(keyType, valType, size, sizeUnknown); } bool peekMap() { return peekMap_virt(); } uint32_t readMapEnd() { return readMapEnd_virt(); } uint32_t readListBegin(TType& elemType, uint32_t& size, bool& sizeUnknown) { return readListBegin_virt(elemType, size, sizeUnknown); } bool peekList() { return peekList_virt(); } uint32_t readListEnd() { return readListEnd_virt(); } uint32_t readSetBegin(TType& elemType, uint32_t& size, bool& sizeUnknown) { return readSetBegin_virt(elemType, size, sizeUnknown); } bool peekSet() { return peekSet_virt(); } uint32_t readSetEnd() { return readSetEnd_virt(); } uint32_t readBool(bool& value) { return readBool_virt(value); } uint32_t readByte(int8_t& byte) { return readByte_virt(byte); } uint32_t readI16(int16_t& i16) { return readI16_virt(i16); } uint32_t readI32(int32_t& i32) { return readI32_virt(i32); } uint32_t readI64(int64_t& i64) { return readI64_virt(i64); } uint32_t readDouble(double& dub) { return readDouble_virt(dub); } uint32_t readFloat(float& flt) { return readFloat_virt(flt); } uint32_t readString(std::string& str) { return readString_virt(str); } uint32_t readString(folly::fbstring& str) { std::string data; uint32_t ret = readString_virt(data); str = data; return ret; } uint32_t readBinary(std::string& str) { return readBinary_virt(str); } uint32_t readBinary(folly::fbstring& str) { std::string data; uint32_t ret = readBinary_virt(data); str = data; return ret; } int32_t getStringSizeLimit() { return 0; // No limit } int32_t getContainerSizeLimit() { return 0; // No limit } /* * std::vector is specialized for bool, and its elements are individual bits * rather than bools. We need to define a different version of readBool() * to work with std::vector. */ uint32_t readBool(std::vector::reference value) { return readBool_virt(value); } /** * Method to arbitrarily skip over data. */ uint32_t skip(TType type) { return skip_virt(type); } virtual uint32_t skip_virt(TType type) { return ::apache::thrift::protocol::skip(*this, type); } inline std::shared_ptr getTransport() { return ptrans_; } // TODO: remove these two calls, they are for backwards // compatibility inline std::shared_ptr getInputTransport() { return ptrans_; } inline std::shared_ptr getOutputTransport() { return ptrans_; } protected: explicit TProtocol(std::shared_ptr ptrans) : ptrans_(ptrans) {} /** * Construct a TProtocol using a raw TTransport pointer. * * It is the callers responsibility to ensure that the TTransport remains * valid for the lifetime of the TProtocol object. */ explicit TProtocol(TTransport* ptrans) : ptrans_(ptrans, [](TTransport*) {}) {} std::shared_ptr ptrans_; private: TProtocol() {} }; /** * Constructs protocol objects given transports. */ class TProtocolFactory { public: TProtocolFactory() {} virtual ~TProtocolFactory() {} virtual std::shared_ptr getProtocol( std::shared_ptr trans) = 0; }; /** * Constructs both input and output protocol objects with a given pair of * input and output transports. * * TProtocolPair.first = Input Protocol * TProtocolPair.second = Output Protocol */ typedef std::pair, std::shared_ptr> TProtocolPair; class TDuplexProtocolFactory { public: TDuplexProtocolFactory() {} virtual ~TDuplexProtocolFactory() {} virtual TProtocolPair getProtocol(transport::TTransportPair transports) = 0; virtual std::shared_ptr getInputProtocolFactory() { return std::shared_ptr(); } virtual std::shared_ptr getOutputProtocolFactory() { return std::shared_ptr(); } }; /** * Adapts a TProtocolFactory to a TDuplexProtocolFactory that returns * a new protocol object for both input and output */ template class TSingleProtocolFactory : public TDuplexProtocolFactory { public: TSingleProtocolFactory() { factory_.reset(new Factory_()); } explicit TSingleProtocolFactory(std::shared_ptr factory) : factory_(factory) {} TProtocolPair getProtocol(transport::TTransportPair transports) override { return std::make_pair( factory_->getProtocol(transports.first), factory_->getProtocol(transports.second)); } std::shared_ptr getInputProtocolFactory() override { return factory_; } std::shared_ptr getOutputProtocolFactory() override { return factory_; } private: std::shared_ptr factory_; }; /** * Use TDualProtocolFactory to construct input and output protocols from * different factories. */ class TDualProtocolFactory : public TDuplexProtocolFactory { public: TDualProtocolFactory( std::shared_ptr inputFactory, std::shared_ptr outputFactory) : inputFactory_(inputFactory), outputFactory_(outputFactory) {} TProtocolPair getProtocol(transport::TTransportPair transports) override { return std::make_pair( inputFactory_->getProtocol(transports.first), outputFactory_->getProtocol(transports.second)); } std::shared_ptr getInputProtocolFactory() override { return inputFactory_; } std::shared_ptr getOutputProtocolFactory() override { return outputFactory_; } private: std::shared_ptr inputFactory_; std::shared_ptr outputFactory_; }; /** * Dummy protocol class. * * This class does nothing, and should never be instantiated. * It is used only by the generator code. */ class TDummyProtocol : public TProtocol {}; } // namespace protocol } // namespace thrift } // namespace apache #endif // #define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1