blob: 47d74f817c6fc4f925ff18c80883be1da626d22c [file] [log] [blame]
Mark Slee9f0c6512007-02-28 23:58:26 +00001// Copyright (c) 2006- Facebook
2// Distributed under the Thrift Software License
3//
4// See accompanying file LICENSE or visit the Thrift site at:
5// http://developers.facebook.com/thrift/
6
Mark Sleef5f2be42006-09-05 21:05:31 +00007#ifndef _THRIFT_PROTOCOL_TPROTOCOL_H_
8#define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1
Mark Sleee8540632006-05-30 09:24:40 +00009
Marc Slemko16698852006-08-04 03:16:10 +000010#include <transport/TTransport.h>
Mark Sleef9831082007-02-20 20:59:21 +000011#include <protocol/TProtocolException.h>
Marc Slemko16698852006-08-04 03:16:10 +000012
13#include <boost/shared_ptr.hpp>
14
Mark Slee8d7e1f62006-06-07 06:48:56 +000015#include <netinet/in.h>
Mark Sleee8540632006-05-30 09:24:40 +000016#include <sys/types.h>
17#include <string>
18#include <map>
19
Marc Slemko6f038a72006-08-03 18:58:09 +000020namespace facebook { namespace thrift { namespace protocol {
21
Mark Slee5ea15f92007-03-05 22:55:59 +000022using facebook::thrift::transport::TTransport;
Marc Slemko6f038a72006-08-03 18:58:09 +000023
Mark Slee4f261c52007-04-13 00:33:24 +000024#ifdef HAVE_ENDIAN_H
25#include <endian.h>
26#endif
27
Mark Sleedc8a2a22006-09-19 22:20:18 +000028#if __BYTE_ORDER == __BIG_ENDIAN
Mark Slee4f261c52007-04-13 00:33:24 +000029# define ntohll(n) (n)
30# define htonll(n) (n)
31#elif __BYTE_ORDER == __LITTLE_ENDIAN
32# if defined(__GNUC__) && defined(__GLIBC__)
33# include <byteswap.h>
34# define ntohll(n) bswap_64(n)
35# define htonll(n) bswap_64(n)
36# else /* GNUC & GLIBC */
37# define ntohll(n) ( (((unsigned long long)ntohl(n)) << 32) + ntohl(n >> 32) )
38# define htonll(n) ( (((unsigned long long)htonl(n)) << 32) + htonl(n >> 32) )
39# endif /* GNUC & GLIBC */
40#else /* __BYTE_ORDER */
41# error "Can't define htonll or ntohll!"
Mark Sleedc8a2a22006-09-19 22:20:18 +000042#endif
Mark Slee8d7e1f62006-06-07 06:48:56 +000043
Mark Sleee8540632006-05-30 09:24:40 +000044/**
Mark Slee8d7e1f62006-06-07 06:48:56 +000045 * Enumerated definition of the types that the Thrift protocol supports.
46 * Take special note of the T_END type which is used specifically to mark
47 * the end of a sequence of fields.
48 */
49enum TType {
Marc Slemkod42a2c22006-08-10 03:30:18 +000050 T_STOP = 0,
Marc Slemko5b126d62006-08-11 23:03:42 +000051 T_VOID = 1,
52 T_BOOL = 2,
53 T_BYTE = 3,
Mark Sleecfc01932006-09-01 22:18:16 +000054 T_I08 = 3,
Marc Slemko5b126d62006-08-11 23:03:42 +000055 T_I16 = 6,
Marc Slemko5b126d62006-08-11 23:03:42 +000056 T_I32 = 8,
57 T_U64 = 9,
58 T_I64 = 10,
Mark Sleec98d0502006-09-06 02:42:25 +000059 T_DOUBLE = 4,
Marc Slemko5b126d62006-08-11 23:03:42 +000060 T_STRING = 11,
Marc Slemkod97eb612006-08-24 23:37:36 +000061 T_UTF7 = 11,
62 T_STRUCT = 12,
63 T_MAP = 13,
64 T_SET = 14,
65 T_LIST = 15,
66 T_UTF8 = 16,
67 T_UTF16 = 17
Mark Slee8d7e1f62006-06-07 06:48:56 +000068};
69
70/**
Mark Sleef5f2be42006-09-05 21:05:31 +000071 * Enumerated definition of the message types that the Thrift protocol
72 * supports.
Marc Slemko16698852006-08-04 03:16:10 +000073 */
74enum TMessageType {
75 T_CALL = 1,
Mark Sleef9831082007-02-20 20:59:21 +000076 T_REPLY = 2,
77 T_EXCEPTION = 3
Marc Slemko16698852006-08-04 03:16:10 +000078};
79
80/**
Mark Sleee8540632006-05-30 09:24:40 +000081 * Abstract class for a thrift protocol driver. These are all the methods that
82 * a protocol must implement. Essentially, there must be some way of reading
83 * and writing all the base types, plus a mechanism for writing out structs
Mark Slee5d06fea2007-03-05 22:18:18 +000084 * with indexed fields.
85 *
86 * TProtocol objects should not be shared across multiple encoding contexts,
87 * as they may need to maintain internal state in some protocols (i.e. XML).
88 * Note that is is acceptable for the TProtocol module to do its own internal
89 * buffered reads/writes to the underlying TTransport where appropriate (i.e.
90 * when parsing an input XML stream, reading should be batched rather than
91 * looking ahead character by character for a close tag).
Mark Sleee8540632006-05-30 09:24:40 +000092 *
93 * @author Mark Slee <mcslee@facebook.com>
94 */
95class TProtocol {
96 public:
97 virtual ~TProtocol() {}
98
99 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000100 * Writing functions.
Mark Sleee8540632006-05-30 09:24:40 +0000101 */
102
Mark Slee82a6c0f2007-04-04 21:08:21 +0000103 virtual uint32_t writeMessageBegin(const std::string& name,
104 const TMessageType messageType,
105 const int32_t seqid) = 0;
Marc Slemko16698852006-08-04 03:16:10 +0000106
Mark Slee4af6ed72006-10-25 19:02:49 +0000107 virtual uint32_t writeMessageEnd() = 0;
Marc Slemko16698852006-08-04 03:16:10 +0000108
109
Mark Slee4af6ed72006-10-25 19:02:49 +0000110 virtual uint32_t writeStructBegin(const std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000111
Mark Slee4af6ed72006-10-25 19:02:49 +0000112 virtual uint32_t writeStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000113
Mark Slee4af6ed72006-10-25 19:02:49 +0000114 virtual uint32_t writeFieldBegin(const std::string& name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000115 const TType fieldType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000116 const int16_t fieldId) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000117
Mark Slee4af6ed72006-10-25 19:02:49 +0000118 virtual uint32_t writeFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000119
Mark Slee4af6ed72006-10-25 19:02:49 +0000120 virtual uint32_t writeFieldStop() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000121
Mark Slee4af6ed72006-10-25 19:02:49 +0000122 virtual uint32_t writeMapBegin(const TType keyType,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000123 const TType valType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000124 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000125
Mark Slee4af6ed72006-10-25 19:02:49 +0000126 virtual uint32_t writeMapEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000127
Mark Slee4af6ed72006-10-25 19:02:49 +0000128 virtual uint32_t writeListBegin(const TType elemType,
129 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000130
Mark Slee4af6ed72006-10-25 19:02:49 +0000131 virtual uint32_t writeListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000132
Mark Slee4af6ed72006-10-25 19:02:49 +0000133 virtual uint32_t writeSetBegin(const TType elemType,
134 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000135
Mark Slee4af6ed72006-10-25 19:02:49 +0000136 virtual uint32_t writeSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000137
Mark Slee4af6ed72006-10-25 19:02:49 +0000138 virtual uint32_t writeBool(const bool value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000139
Mark Slee4af6ed72006-10-25 19:02:49 +0000140 virtual uint32_t writeByte(const int8_t byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000141
Mark Slee4af6ed72006-10-25 19:02:49 +0000142 virtual uint32_t writeI16(const int16_t i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000143
Mark Slee4af6ed72006-10-25 19:02:49 +0000144 virtual uint32_t writeI32(const int32_t i32) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000145
Mark Slee4af6ed72006-10-25 19:02:49 +0000146 virtual uint32_t writeI64(const int64_t i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000147
Mark Slee4af6ed72006-10-25 19:02:49 +0000148 virtual uint32_t writeDouble(const double dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000149
Mark Slee4af6ed72006-10-25 19:02:49 +0000150 virtual uint32_t writeString(const std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000151
152 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000153 * Reading functions
Mark Sleee8540632006-05-30 09:24:40 +0000154 */
155
Mark Slee4af6ed72006-10-25 19:02:49 +0000156 virtual uint32_t readMessageBegin(std::string& name,
Marc Slemko91f67482006-08-11 23:58:57 +0000157 TMessageType& messageType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000158 int32_t& seqid) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000159
Mark Slee4af6ed72006-10-25 19:02:49 +0000160 virtual uint32_t readMessageEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000161
Mark Slee4af6ed72006-10-25 19:02:49 +0000162 virtual uint32_t readStructBegin(std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000163
Mark Slee4af6ed72006-10-25 19:02:49 +0000164 virtual uint32_t readStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000165
Mark Slee4af6ed72006-10-25 19:02:49 +0000166 virtual uint32_t readFieldBegin(std::string& name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000167 TType& fieldType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000168 int16_t& fieldId) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000169
Mark Slee4af6ed72006-10-25 19:02:49 +0000170 virtual uint32_t readFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000171
Mark Slee4af6ed72006-10-25 19:02:49 +0000172 virtual uint32_t readMapBegin(TType& keyType,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000173 TType& valType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000174 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000175
Mark Slee4af6ed72006-10-25 19:02:49 +0000176 virtual uint32_t readMapEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000177
Mark Slee4af6ed72006-10-25 19:02:49 +0000178 virtual uint32_t readListBegin(TType& elemType,
179 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000180
Mark Slee4af6ed72006-10-25 19:02:49 +0000181 virtual uint32_t readListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000182
Mark Slee4af6ed72006-10-25 19:02:49 +0000183 virtual uint32_t readSetBegin(TType& elemType,
184 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000185
Mark Slee4af6ed72006-10-25 19:02:49 +0000186 virtual uint32_t readSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000187
Mark Slee4af6ed72006-10-25 19:02:49 +0000188 virtual uint32_t readBool(bool& value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000189
Mark Slee4af6ed72006-10-25 19:02:49 +0000190 virtual uint32_t readByte(int8_t& byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000191
Mark Slee4af6ed72006-10-25 19:02:49 +0000192 virtual uint32_t readI16(int16_t& i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000193
Mark Slee4af6ed72006-10-25 19:02:49 +0000194 virtual uint32_t readI32(int32_t& i32) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000195
Mark Slee4af6ed72006-10-25 19:02:49 +0000196 virtual uint32_t readI64(int64_t& i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000197
Mark Slee4af6ed72006-10-25 19:02:49 +0000198 virtual uint32_t readDouble(double& dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000199
Mark Slee4af6ed72006-10-25 19:02:49 +0000200 virtual uint32_t readString(std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000201
202 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000203 * Method to arbitrarily skip over data.
Mark Sleee8540632006-05-30 09:24:40 +0000204 */
Mark Slee4af6ed72006-10-25 19:02:49 +0000205 uint32_t skip(TType type) {
Mark Slee8d7e1f62006-06-07 06:48:56 +0000206 switch (type) {
Mark Slee78f58e22006-09-02 04:17:07 +0000207 case T_BOOL:
208 {
209 bool boolv;
Mark Slee4af6ed72006-10-25 19:02:49 +0000210 return readBool(boolv);
Mark Slee78f58e22006-09-02 04:17:07 +0000211 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000212 case T_BYTE:
213 {
Mark Slee78f58e22006-09-02 04:17:07 +0000214 int8_t bytev;
Mark Slee4af6ed72006-10-25 19:02:49 +0000215 return readByte(bytev);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000216 }
Mark Sleecfc01932006-09-01 22:18:16 +0000217 case T_I16:
Mark Slee8d7e1f62006-06-07 06:48:56 +0000218 {
Mark Sleecfc01932006-09-01 22:18:16 +0000219 int16_t i16;
Mark Slee4af6ed72006-10-25 19:02:49 +0000220 return readI16(i16);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000221 }
222 case T_I32:
223 {
224 int32_t i32;
Mark Slee4af6ed72006-10-25 19:02:49 +0000225 return readI32(i32);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000226 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000227 case T_I64:
228 {
229 int64_t i64;
Mark Slee4af6ed72006-10-25 19:02:49 +0000230 return readI64(i64);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000231 }
Mark Sleec98d0502006-09-06 02:42:25 +0000232 case T_DOUBLE:
233 {
234 double dub;
Mark Slee4af6ed72006-10-25 19:02:49 +0000235 return readDouble(dub);
Mark Sleec98d0502006-09-06 02:42:25 +0000236 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000237 case T_STRING:
238 {
239 std::string str;
Mark Slee4af6ed72006-10-25 19:02:49 +0000240 return readString(str);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000241 }
242 case T_STRUCT:
243 {
244 uint32_t result = 0;
245 std::string name;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000246 int16_t fid;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000247 TType ftype;
Mark Slee4af6ed72006-10-25 19:02:49 +0000248 result += readStructBegin(name);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000249 while (true) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000250 result += readFieldBegin(name, ftype, fid);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000251 if (ftype == T_STOP) {
252 break;
253 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000254 result += skip(ftype);
255 result += readFieldEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000256 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000257 result += readStructEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000258 return result;
259 }
260 case T_MAP:
261 {
262 uint32_t result = 0;
263 TType keyType;
264 TType valType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000265 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000266 result += readMapBegin(keyType, valType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000267 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000268 result += skip(keyType);
269 result += skip(valType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000270 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000271 result += readMapEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000272 return result;
273 }
274 case T_SET:
275 {
276 uint32_t result = 0;
277 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000278 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000279 result += readSetBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000280 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000281 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000282 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000283 result += readSetEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000284 return result;
285 }
286 case T_LIST:
287 {
288 uint32_t result = 0;
289 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000290 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000291 result += readListBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000292 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000293 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000294 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000295 result += readListEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000296 return result;
297 }
298 default:
299 return 0;
300 }
301 }
Mark Sleee8540632006-05-30 09:24:40 +0000302
Mark Slee5ea15f92007-03-05 22:55:59 +0000303 inline boost::shared_ptr<TTransport> getTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000304 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000305 }
306
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000307 // TODO: remove these two calls, they are for backwards
308 // compatibility
Mark Slee5ea15f92007-03-05 22:55:59 +0000309 inline boost::shared_ptr<TTransport> getInputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000310 return ptrans_;
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000311 }
Mark Slee5ea15f92007-03-05 22:55:59 +0000312 inline boost::shared_ptr<TTransport> getOutputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000313 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000314 }
315
Mark Sleee8540632006-05-30 09:24:40 +0000316 protected:
Mark Slee5ea15f92007-03-05 22:55:59 +0000317 TProtocol(boost::shared_ptr<TTransport> ptrans):
Mark Slee43b6c632007-02-07 00:54:17 +0000318 ptrans_(ptrans) {
319 trans_ = ptrans.get();
320 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000321
Mark Slee5ea15f92007-03-05 22:55:59 +0000322 boost::shared_ptr<TTransport> ptrans_;
Mark Slee43b6c632007-02-07 00:54:17 +0000323 TTransport* trans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000324
325 private:
Mark Sleee8540632006-05-30 09:24:40 +0000326 TProtocol() {}
327};
328
Mark Slee4af6ed72006-10-25 19:02:49 +0000329/**
330 * Constructs input and output protocol objects given transports.
331 */
332class TProtocolFactory {
333 public:
334 TProtocolFactory() {}
335
336 virtual ~TProtocolFactory() {}
337
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000338 virtual boost::shared_ptr<TProtocol> getProtocol(boost::shared_ptr<TTransport> trans) = 0;
Mark Slee4af6ed72006-10-25 19:02:49 +0000339};
340
Marc Slemko6f038a72006-08-03 18:58:09 +0000341}}} // facebook::thrift::protocol
342
Mark Sleef5f2be42006-09-05 21:05:31 +0000343#endif // #define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1