blob: 5dd5a8c6a44b415bc715255b3a6c184db8ac34b2 [file] [log] [blame]
// Copyright (c) 2006- Facebook
// Distributed under the Thrift Software License
//
// See accompanying file LICENSE or visit the Thrift site at:
// http://developers.facebook.com/thrift/
#ifndef _THRIFT_PROTOCOL_TDENSEPROTOCOL_H_
#define _THRIFT_PROTOCOL_TDENSEPROTOCOL_H_ 1
#include "TBinaryProtocol.h"
namespace facebook { namespace thrift { namespace protocol {
/**
* !!!WARNING!!!
* This class is still highly experimental. Incompatible changes
* WILL be made to it without notice. DO NOT USE IT YET unless
* you are coordinating your testing with the author.
*
* The dense protocol is designed to use as little space as possible.
*
* There are two types of dense protocol instances. Standalone instances
* are not used for RPC and just encoded and decode structures of
* a predetermined type. Non-standalone instances are used for RPC.
* Currently, only standalone instances exist.
*
* To use a standalone dense protocol object, you must set the type_spec
* property (either in the constructor, or with setTypeSpec) to the local
* reflection TypeSpec of the structures you will write to (or read from) the
* protocol instance.
*
* BEST PRACTICES:
* - Never use optional for primitives or containers.
* - Only use optional for structures if they are very big and very rarely set.
* - All integers are variable-length, so you can use i64 without bloating.
* - NEVER EVER change the struct definitions IN ANY WAY without either
* changing your cache keys or talking to dreiss.
*
* TODO(dreiss): New class write with old meta.
*
* We override all of TBinaryProtocol's methods.
* We inherit so that we can can explicitly call TBPs's primitive-writing
* methods within our versions.
*
* @author David Reiss <dreiss@facebook.com>
*/
class TDenseProtocol : public TBinaryProtocol {
protected:
static const int32_t VERSION_MASK = 0xffff0000;
// VERSION_1 (0x80010000) is taken by TBinaryProtocol.
static const int32_t VERSION_2 = 0x80020000;
public:
typedef facebook::thrift::reflection::local::TypeSpec TypeSpec;
static const int FP_PREFIX_LEN;
/**
* @param tran The transport to use.
* @param type_spec The TypeSpec of the structures using this protocol.
*/
TDenseProtocol(boost::shared_ptr<TTransport> trans,
TypeSpec* type_spec = NULL) :
TBinaryProtocol(trans),
type_spec_(type_spec),
standalone_(true)
{}
void setTypeSpec(TypeSpec* type_spec) {
type_spec_ = type_spec;
}
TypeSpec* getTypeSpec() {
return type_spec_;
}
/*
* Writing functions.
*/
virtual uint32_t writeMessageBegin(const std::string& name,
const TMessageType messageType,
const int32_t seqid);
virtual uint32_t writeMessageEnd();
virtual uint32_t writeStructBegin(const char* name);
virtual uint32_t writeStructEnd();
virtual uint32_t writeFieldBegin(const char* name,
const TType fieldType,
const int16_t fieldId);
virtual uint32_t writeFieldEnd();
virtual uint32_t writeFieldStop();
virtual uint32_t writeMapBegin(const TType keyType,
const TType valType,
const uint32_t size);
virtual uint32_t writeMapEnd();
virtual uint32_t writeListBegin(const TType elemType,
const uint32_t size);
virtual uint32_t writeListEnd();
virtual uint32_t writeSetBegin(const TType elemType,
const uint32_t size);
virtual uint32_t writeSetEnd();
virtual uint32_t writeBool(const bool value);
virtual uint32_t writeByte(const int8_t byte);
virtual uint32_t writeI16(const int16_t i16);
virtual uint32_t writeI32(const int32_t i32);
virtual uint32_t writeI64(const int64_t i64);
virtual uint32_t writeDouble(const double dub);
virtual uint32_t writeString(const std::string& str);
virtual uint32_t writeBinary(const std::string& str);
/*
* Helper writing functions (don't do state transitions).
*/
inline uint32_t subWriteI32(const int32_t i32);
inline uint32_t subWriteString(const std::string& str);
uint32_t subWriteBool(const bool value) {
return TBinaryProtocol::writeBool(value);
}
/*
* Reading functions
*/
uint32_t readMessageBegin(std::string& name,
TMessageType& messageType,
int32_t& seqid);
uint32_t readMessageEnd();
uint32_t readStructBegin(std::string& name);
uint32_t readStructEnd();
uint32_t readFieldBegin(std::string& name,
TType& fieldType,
int16_t& fieldId);
uint32_t readFieldEnd();
uint32_t readMapBegin(TType& keyType,
TType& valType,
uint32_t& size);
uint32_t readMapEnd();
uint32_t readListBegin(TType& elemType,
uint32_t& size);
uint32_t readListEnd();
uint32_t readSetBegin(TType& elemType,
uint32_t& size);
uint32_t readSetEnd();
uint32_t readBool(bool& value);
uint32_t readByte(int8_t& byte);
uint32_t readI16(int16_t& i16);
uint32_t readI32(int32_t& i32);
uint32_t readI64(int64_t& i64);
uint32_t readDouble(double& dub);
uint32_t readString(std::string& str);
uint32_t readBinary(std::string& str);
/*
* Helper reading functions (don't do state transitions).
*/
inline uint32_t subReadI32(int32_t& i32);
inline uint32_t subReadString(std::string& str);
uint32_t subReadBool(bool& value) {
return TBinaryProtocol::readBool(value);
}
private:
// Implementation functions, documented in the .cpp.
inline void checkTType(const TType ttype);
inline void stateTransition();
// Read and write variable-length integers.
// Uses the same technique as the MIDI file format.
inline uint32_t vlqRead(uint64_t& vlq);
inline uint32_t vlqWrite(uint64_t vlq);
// Called before throwing an exception to make the object reusable.
void resetState() {
ts_stack_.clear();
idx_stack_.clear();
mkv_stack_.clear();
}
// TypeSpec of the top-level structure to write,
// for standalone protocol objects.
TypeSpec* type_spec_;
std::vector<TypeSpec*> ts_stack_; // TypeSpec stack.
std::vector<int> idx_stack_; // InDeX stack.
std::vector<bool> mkv_stack_; // Map Key/Vlue stack.
// True = key, False = value.
// True iff this is a standalone instance (no RPC).
bool standalone_;
};
}}} // facebook::thrift::protocol
#endif // #ifndef _THRIFT_PROTOCOL_TDENSEPROTOCOL_H_