blob: 85cc48a9bce21504e78703f8ded05e2f066cf00d [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
David Reiss0c90f6f2008-02-06 22:18:40 +000020namespace facebook { namespace thrift { namespace protocol {
Marc Slemko6f038a72006-08-03 18:58:09 +000021
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 Slee1d2ead32007-06-09 01:23:04 +000028#ifndef __BYTE_ORDER
29# if defined(BYTE_ORDER) && defined(LITTLE_ENDIAN) && defined(BIG_ENDIAN)
30# define __BYTE_ORDER BYTE_ORDER
31# define __LITTLE_ENDIAN LITTLE_ENDIAN
32# define __BIG_ENDIAN BIG_ENDIAN
33# else
34# error "Cannot determine endianness"
35# endif
36#endif
37
Mark Sleedc8a2a22006-09-19 22:20:18 +000038#if __BYTE_ORDER == __BIG_ENDIAN
Mark Slee4f261c52007-04-13 00:33:24 +000039# define ntohll(n) (n)
40# define htonll(n) (n)
41#elif __BYTE_ORDER == __LITTLE_ENDIAN
42# if defined(__GNUC__) && defined(__GLIBC__)
43# include <byteswap.h>
44# define ntohll(n) bswap_64(n)
45# define htonll(n) bswap_64(n)
46# else /* GNUC & GLIBC */
47# define ntohll(n) ( (((unsigned long long)ntohl(n)) << 32) + ntohl(n >> 32) )
48# define htonll(n) ( (((unsigned long long)htonl(n)) << 32) + htonl(n >> 32) )
49# endif /* GNUC & GLIBC */
50#else /* __BYTE_ORDER */
51# error "Can't define htonll or ntohll!"
Mark Sleedc8a2a22006-09-19 22:20:18 +000052#endif
Mark Slee8d7e1f62006-06-07 06:48:56 +000053
Mark Sleee8540632006-05-30 09:24:40 +000054/**
Mark Slee8d7e1f62006-06-07 06:48:56 +000055 * Enumerated definition of the types that the Thrift protocol supports.
56 * Take special note of the T_END type which is used specifically to mark
57 * the end of a sequence of fields.
58 */
59enum TType {
Marc Slemkod42a2c22006-08-10 03:30:18 +000060 T_STOP = 0,
Marc Slemko5b126d62006-08-11 23:03:42 +000061 T_VOID = 1,
62 T_BOOL = 2,
63 T_BYTE = 3,
Mark Sleecfc01932006-09-01 22:18:16 +000064 T_I08 = 3,
Marc Slemko5b126d62006-08-11 23:03:42 +000065 T_I16 = 6,
Marc Slemko5b126d62006-08-11 23:03:42 +000066 T_I32 = 8,
67 T_U64 = 9,
68 T_I64 = 10,
Mark Sleec98d0502006-09-06 02:42:25 +000069 T_DOUBLE = 4,
Marc Slemko5b126d62006-08-11 23:03:42 +000070 T_STRING = 11,
Marc Slemkod97eb612006-08-24 23:37:36 +000071 T_UTF7 = 11,
72 T_STRUCT = 12,
73 T_MAP = 13,
74 T_SET = 14,
75 T_LIST = 15,
76 T_UTF8 = 16,
77 T_UTF16 = 17
Mark Slee8d7e1f62006-06-07 06:48:56 +000078};
79
80/**
Mark Sleef5f2be42006-09-05 21:05:31 +000081 * Enumerated definition of the message types that the Thrift protocol
82 * supports.
Marc Slemko16698852006-08-04 03:16:10 +000083 */
84enum TMessageType {
85 T_CALL = 1,
Mark Sleef9831082007-02-20 20:59:21 +000086 T_REPLY = 2,
87 T_EXCEPTION = 3
Marc Slemko16698852006-08-04 03:16:10 +000088};
89
90/**
Mark Sleee8540632006-05-30 09:24:40 +000091 * Abstract class for a thrift protocol driver. These are all the methods that
92 * a protocol must implement. Essentially, there must be some way of reading
93 * and writing all the base types, plus a mechanism for writing out structs
Mark Slee5d06fea2007-03-05 22:18:18 +000094 * with indexed fields.
95 *
96 * TProtocol objects should not be shared across multiple encoding contexts,
97 * as they may need to maintain internal state in some protocols (i.e. XML).
98 * Note that is is acceptable for the TProtocol module to do its own internal
99 * buffered reads/writes to the underlying TTransport where appropriate (i.e.
100 * when parsing an input XML stream, reading should be batched rather than
101 * looking ahead character by character for a close tag).
Mark Sleee8540632006-05-30 09:24:40 +0000102 *
103 * @author Mark Slee <mcslee@facebook.com>
104 */
105class TProtocol {
106 public:
107 virtual ~TProtocol() {}
108
109 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000110 * Writing functions.
Mark Sleee8540632006-05-30 09:24:40 +0000111 */
112
Mark Slee82a6c0f2007-04-04 21:08:21 +0000113 virtual uint32_t writeMessageBegin(const std::string& name,
114 const TMessageType messageType,
115 const int32_t seqid) = 0;
Marc Slemko16698852006-08-04 03:16:10 +0000116
Mark Slee4af6ed72006-10-25 19:02:49 +0000117 virtual uint32_t writeMessageEnd() = 0;
Marc Slemko16698852006-08-04 03:16:10 +0000118
119
Mark Slee4af6ed72006-10-25 19:02:49 +0000120 virtual uint32_t writeStructBegin(const std::string& name) = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000121
Mark Slee4af6ed72006-10-25 19:02:49 +0000122 virtual uint32_t writeStructEnd() = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000123
Mark Slee4af6ed72006-10-25 19:02:49 +0000124 virtual uint32_t writeFieldBegin(const std::string& name,
David Reiss96d23882007-07-26 21:10:32 +0000125 const TType fieldType,
126 const int16_t fieldId) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000127
Mark Slee4af6ed72006-10-25 19:02:49 +0000128 virtual uint32_t writeFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000129
Mark Slee4af6ed72006-10-25 19:02:49 +0000130 virtual uint32_t writeFieldStop() = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000131
Mark Slee4af6ed72006-10-25 19:02:49 +0000132 virtual uint32_t writeMapBegin(const TType keyType,
David Reiss96d23882007-07-26 21:10:32 +0000133 const TType valType,
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 writeMapEnd() = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000137
Mark Slee4af6ed72006-10-25 19:02:49 +0000138 virtual uint32_t writeListBegin(const TType elemType,
David Reiss96d23882007-07-26 21:10:32 +0000139 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000140
Mark Slee4af6ed72006-10-25 19:02:49 +0000141 virtual uint32_t writeListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000142
Mark Slee4af6ed72006-10-25 19:02:49 +0000143 virtual uint32_t writeSetBegin(const TType elemType,
David Reiss96d23882007-07-26 21:10:32 +0000144 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000145
Mark Slee4af6ed72006-10-25 19:02:49 +0000146 virtual uint32_t writeSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000147
Mark Slee4af6ed72006-10-25 19:02:49 +0000148 virtual uint32_t writeBool(const bool value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000149
Mark Slee4af6ed72006-10-25 19:02:49 +0000150 virtual uint32_t writeByte(const int8_t byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000151
Mark Slee4af6ed72006-10-25 19:02:49 +0000152 virtual uint32_t writeI16(const int16_t i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000153
Mark Slee4af6ed72006-10-25 19:02:49 +0000154 virtual uint32_t writeI32(const int32_t i32) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000155
Mark Slee4af6ed72006-10-25 19:02:49 +0000156 virtual uint32_t writeI64(const int64_t i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000157
Mark Slee4af6ed72006-10-25 19:02:49 +0000158 virtual uint32_t writeDouble(const double dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000159
Mark Slee4af6ed72006-10-25 19:02:49 +0000160 virtual uint32_t writeString(const std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000161
162 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000163 * Reading functions
Mark Sleee8540632006-05-30 09:24:40 +0000164 */
165
Mark Slee4af6ed72006-10-25 19:02:49 +0000166 virtual uint32_t readMessageBegin(std::string& name,
David Reiss96d23882007-07-26 21:10:32 +0000167 TMessageType& messageType,
168 int32_t& seqid) = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000169
Mark Slee4af6ed72006-10-25 19:02:49 +0000170 virtual uint32_t readMessageEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000171
Mark Slee4af6ed72006-10-25 19:02:49 +0000172 virtual uint32_t readStructBegin(std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000173
Mark Slee4af6ed72006-10-25 19:02:49 +0000174 virtual uint32_t readStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000175
Mark Slee4af6ed72006-10-25 19:02:49 +0000176 virtual uint32_t readFieldBegin(std::string& name,
David Reiss96d23882007-07-26 21:10:32 +0000177 TType& fieldType,
178 int16_t& fieldId) = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000179
Mark Slee4af6ed72006-10-25 19:02:49 +0000180 virtual uint32_t readFieldEnd() = 0;
David Reiss0c90f6f2008-02-06 22:18:40 +0000181
Mark Slee4af6ed72006-10-25 19:02:49 +0000182 virtual uint32_t readMapBegin(TType& keyType,
David Reiss96d23882007-07-26 21:10:32 +0000183 TType& valType,
184 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000185
Mark Slee4af6ed72006-10-25 19:02:49 +0000186 virtual uint32_t readMapEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000187
Mark Slee4af6ed72006-10-25 19:02:49 +0000188 virtual uint32_t readListBegin(TType& elemType,
David Reiss96d23882007-07-26 21:10:32 +0000189 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000190
Mark Slee4af6ed72006-10-25 19:02:49 +0000191 virtual uint32_t readListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000192
Mark Slee4af6ed72006-10-25 19:02:49 +0000193 virtual uint32_t readSetBegin(TType& elemType,
David Reiss96d23882007-07-26 21:10:32 +0000194 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000195
Mark Slee4af6ed72006-10-25 19:02:49 +0000196 virtual uint32_t readSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000197
Mark Slee4af6ed72006-10-25 19:02:49 +0000198 virtual uint32_t readBool(bool& value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000199
Mark Slee4af6ed72006-10-25 19:02:49 +0000200 virtual uint32_t readByte(int8_t& byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000201
Mark Slee4af6ed72006-10-25 19:02:49 +0000202 virtual uint32_t readI16(int16_t& i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000203
Mark Slee4af6ed72006-10-25 19:02:49 +0000204 virtual uint32_t readI32(int32_t& i32) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000205
Mark Slee4af6ed72006-10-25 19:02:49 +0000206 virtual uint32_t readI64(int64_t& i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000207
Mark Slee4af6ed72006-10-25 19:02:49 +0000208 virtual uint32_t readDouble(double& dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000209
Mark Slee4af6ed72006-10-25 19:02:49 +0000210 virtual uint32_t readString(std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000211
212 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000213 * Method to arbitrarily skip over data.
Mark Sleee8540632006-05-30 09:24:40 +0000214 */
Mark Slee4af6ed72006-10-25 19:02:49 +0000215 uint32_t skip(TType type) {
Mark Slee8d7e1f62006-06-07 06:48:56 +0000216 switch (type) {
Mark Slee78f58e22006-09-02 04:17:07 +0000217 case T_BOOL:
218 {
219 bool boolv;
Mark Slee4af6ed72006-10-25 19:02:49 +0000220 return readBool(boolv);
Mark Slee78f58e22006-09-02 04:17:07 +0000221 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000222 case T_BYTE:
223 {
Mark Slee78f58e22006-09-02 04:17:07 +0000224 int8_t bytev;
Mark Slee4af6ed72006-10-25 19:02:49 +0000225 return readByte(bytev);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000226 }
Mark Sleecfc01932006-09-01 22:18:16 +0000227 case T_I16:
Mark Slee8d7e1f62006-06-07 06:48:56 +0000228 {
Mark Sleecfc01932006-09-01 22:18:16 +0000229 int16_t i16;
Mark Slee4af6ed72006-10-25 19:02:49 +0000230 return readI16(i16);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000231 }
232 case T_I32:
233 {
234 int32_t i32;
Mark Slee4af6ed72006-10-25 19:02:49 +0000235 return readI32(i32);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000236 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000237 case T_I64:
238 {
239 int64_t i64;
Mark Slee4af6ed72006-10-25 19:02:49 +0000240 return readI64(i64);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000241 }
Mark Sleec98d0502006-09-06 02:42:25 +0000242 case T_DOUBLE:
243 {
244 double dub;
Mark Slee4af6ed72006-10-25 19:02:49 +0000245 return readDouble(dub);
Mark Sleec98d0502006-09-06 02:42:25 +0000246 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000247 case T_STRING:
248 {
249 std::string str;
Mark Slee4af6ed72006-10-25 19:02:49 +0000250 return readString(str);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000251 }
252 case T_STRUCT:
253 {
254 uint32_t result = 0;
255 std::string name;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000256 int16_t fid;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000257 TType ftype;
Mark Slee4af6ed72006-10-25 19:02:49 +0000258 result += readStructBegin(name);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000259 while (true) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000260 result += readFieldBegin(name, ftype, fid);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000261 if (ftype == T_STOP) {
262 break;
263 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000264 result += skip(ftype);
265 result += readFieldEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000266 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000267 result += readStructEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000268 return result;
269 }
270 case T_MAP:
271 {
272 uint32_t result = 0;
273 TType keyType;
274 TType valType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000275 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000276 result += readMapBegin(keyType, valType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000277 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000278 result += skip(keyType);
279 result += skip(valType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000280 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000281 result += readMapEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000282 return result;
283 }
284 case T_SET:
285 {
286 uint32_t result = 0;
287 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000288 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000289 result += readSetBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000290 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000291 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000292 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000293 result += readSetEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000294 return result;
295 }
296 case T_LIST:
297 {
298 uint32_t result = 0;
299 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000300 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000301 result += readListBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000302 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000303 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000304 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000305 result += readListEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000306 return result;
307 }
308 default:
309 return 0;
310 }
311 }
Mark Sleee8540632006-05-30 09:24:40 +0000312
Mark Slee5ea15f92007-03-05 22:55:59 +0000313 inline boost::shared_ptr<TTransport> getTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000314 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000315 }
316
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000317 // TODO: remove these two calls, they are for backwards
318 // compatibility
Mark Slee5ea15f92007-03-05 22:55:59 +0000319 inline boost::shared_ptr<TTransport> getInputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000320 return ptrans_;
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000321 }
Mark Slee5ea15f92007-03-05 22:55:59 +0000322 inline boost::shared_ptr<TTransport> getOutputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000323 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000324 }
325
Mark Sleee8540632006-05-30 09:24:40 +0000326 protected:
Mark Slee5ea15f92007-03-05 22:55:59 +0000327 TProtocol(boost::shared_ptr<TTransport> ptrans):
Mark Slee43b6c632007-02-07 00:54:17 +0000328 ptrans_(ptrans) {
329 trans_ = ptrans.get();
330 }
David Reiss0c90f6f2008-02-06 22:18:40 +0000331
Mark Slee5ea15f92007-03-05 22:55:59 +0000332 boost::shared_ptr<TTransport> ptrans_;
Mark Slee43b6c632007-02-07 00:54:17 +0000333 TTransport* trans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000334
335 private:
Mark Sleee8540632006-05-30 09:24:40 +0000336 TProtocol() {}
337};
338
Mark Slee4af6ed72006-10-25 19:02:49 +0000339/**
340 * Constructs input and output protocol objects given transports.
341 */
342class TProtocolFactory {
343 public:
344 TProtocolFactory() {}
345
346 virtual ~TProtocolFactory() {}
347
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000348 virtual boost::shared_ptr<TProtocol> getProtocol(boost::shared_ptr<TTransport> trans) = 0;
Mark Slee4af6ed72006-10-25 19:02:49 +0000349};
350
Marc Slemko6f038a72006-08-03 18:58:09 +0000351}}} // facebook::thrift::protocol
352
Mark Sleef5f2be42006-09-05 21:05:31 +0000353#endif // #define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1