blob: 8b2e6bd7a6130af1a20380533c9142f9f4658276 [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
Marc Slemko16698852006-08-04 03:16:10 +000022using namespace boost;
23
Marc Slemko6f038a72006-08-03 18:58:09 +000024using namespace facebook::thrift::transport;
25
Mark Sleedc8a2a22006-09-19 22:20:18 +000026#if __BYTE_ORDER == __BIG_ENDIAN
27#define ntohll(n) (n)
28#define htonll(n) (n)
29#else
30#define ntohll(n) ( (((unsigned long long)ntohl(n)) << 32) + ntohl(n >> 32) )
31#define htonll(n) ( (((unsigned long long)htonl(n)) << 32) + htonl(n >> 32) )
32#endif
Mark Slee8d7e1f62006-06-07 06:48:56 +000033
Mark Sleee8540632006-05-30 09:24:40 +000034/**
Mark Slee8d7e1f62006-06-07 06:48:56 +000035 * Enumerated definition of the types that the Thrift protocol supports.
36 * Take special note of the T_END type which is used specifically to mark
37 * the end of a sequence of fields.
38 */
39enum TType {
Marc Slemkod42a2c22006-08-10 03:30:18 +000040 T_STOP = 0,
Marc Slemko5b126d62006-08-11 23:03:42 +000041 T_VOID = 1,
42 T_BOOL = 2,
43 T_BYTE = 3,
Mark Sleecfc01932006-09-01 22:18:16 +000044 T_I08 = 3,
Marc Slemko5b126d62006-08-11 23:03:42 +000045 T_I16 = 6,
Marc Slemko5b126d62006-08-11 23:03:42 +000046 T_I32 = 8,
47 T_U64 = 9,
48 T_I64 = 10,
Mark Sleec98d0502006-09-06 02:42:25 +000049 T_DOUBLE = 4,
Marc Slemko5b126d62006-08-11 23:03:42 +000050 T_STRING = 11,
Marc Slemkod97eb612006-08-24 23:37:36 +000051 T_UTF7 = 11,
52 T_STRUCT = 12,
53 T_MAP = 13,
54 T_SET = 14,
55 T_LIST = 15,
56 T_UTF8 = 16,
57 T_UTF16 = 17
Mark Slee8d7e1f62006-06-07 06:48:56 +000058};
59
60/**
Mark Sleef5f2be42006-09-05 21:05:31 +000061 * Enumerated definition of the message types that the Thrift protocol
62 * supports.
Marc Slemko16698852006-08-04 03:16:10 +000063 */
64enum TMessageType {
65 T_CALL = 1,
Mark Sleef9831082007-02-20 20:59:21 +000066 T_REPLY = 2,
67 T_EXCEPTION = 3
Marc Slemko16698852006-08-04 03:16:10 +000068};
69
70/**
Mark Sleee8540632006-05-30 09:24:40 +000071 * Abstract class for a thrift protocol driver. These are all the methods that
72 * a protocol must implement. Essentially, there must be some way of reading
73 * and writing all the base types, plus a mechanism for writing out structs
Mark Slee5d06fea2007-03-05 22:18:18 +000074 * with indexed fields.
75 *
76 * TProtocol objects should not be shared across multiple encoding contexts,
77 * as they may need to maintain internal state in some protocols (i.e. XML).
78 * Note that is is acceptable for the TProtocol module to do its own internal
79 * buffered reads/writes to the underlying TTransport where appropriate (i.e.
80 * when parsing an input XML stream, reading should be batched rather than
81 * looking ahead character by character for a close tag).
Mark Sleee8540632006-05-30 09:24:40 +000082 *
83 * @author Mark Slee <mcslee@facebook.com>
84 */
85class TProtocol {
86 public:
87 virtual ~TProtocol() {}
88
89 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +000090 * Writing functions.
Mark Sleee8540632006-05-30 09:24:40 +000091 */
92
Mark Slee4af6ed72006-10-25 19:02:49 +000093 virtual uint32_t writeMessageBegin(const std::string name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +000094 const TMessageType messageType,
Mark Slee4af6ed72006-10-25 19:02:49 +000095 const int32_t seqid) = 0;
Marc Slemko16698852006-08-04 03:16:10 +000096
Mark Slee4af6ed72006-10-25 19:02:49 +000097 virtual uint32_t writeMessageEnd() = 0;
Marc Slemko16698852006-08-04 03:16:10 +000098
99
Mark Slee4af6ed72006-10-25 19:02:49 +0000100 virtual uint32_t writeStructBegin(const std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000101
Mark Slee4af6ed72006-10-25 19:02:49 +0000102 virtual uint32_t writeStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000103
Mark Slee4af6ed72006-10-25 19:02:49 +0000104 virtual uint32_t writeFieldBegin(const std::string& name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000105 const TType fieldType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000106 const int16_t fieldId) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000107
Mark Slee4af6ed72006-10-25 19:02:49 +0000108 virtual uint32_t writeFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000109
Mark Slee4af6ed72006-10-25 19:02:49 +0000110 virtual uint32_t writeFieldStop() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000111
Mark Slee4af6ed72006-10-25 19:02:49 +0000112 virtual uint32_t writeMapBegin(const TType keyType,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000113 const TType valType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000114 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000115
Mark Slee4af6ed72006-10-25 19:02:49 +0000116 virtual uint32_t writeMapEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000117
Mark Slee4af6ed72006-10-25 19:02:49 +0000118 virtual uint32_t writeListBegin(const TType elemType,
119 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000120
Mark Slee4af6ed72006-10-25 19:02:49 +0000121 virtual uint32_t writeListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000122
Mark Slee4af6ed72006-10-25 19:02:49 +0000123 virtual uint32_t writeSetBegin(const TType elemType,
124 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000125
Mark Slee4af6ed72006-10-25 19:02:49 +0000126 virtual uint32_t writeSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000127
Mark Slee4af6ed72006-10-25 19:02:49 +0000128 virtual uint32_t writeBool(const bool value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000129
Mark Slee4af6ed72006-10-25 19:02:49 +0000130 virtual uint32_t writeByte(const int8_t byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000131
Mark Slee4af6ed72006-10-25 19:02:49 +0000132 virtual uint32_t writeI16(const int16_t i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000133
Mark Slee4af6ed72006-10-25 19:02:49 +0000134 virtual uint32_t writeI32(const int32_t i32) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000135
Mark Slee4af6ed72006-10-25 19:02:49 +0000136 virtual uint32_t writeI64(const int64_t i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000137
Mark Slee4af6ed72006-10-25 19:02:49 +0000138 virtual uint32_t writeDouble(const double dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000139
Mark Slee4af6ed72006-10-25 19:02:49 +0000140 virtual uint32_t writeString(const std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000141
142 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000143 * Reading functions
Mark Sleee8540632006-05-30 09:24:40 +0000144 */
145
Mark Slee4af6ed72006-10-25 19:02:49 +0000146 virtual uint32_t readMessageBegin(std::string& name,
Marc Slemko91f67482006-08-11 23:58:57 +0000147 TMessageType& messageType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000148 int32_t& seqid) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000149
Mark Slee4af6ed72006-10-25 19:02:49 +0000150 virtual uint32_t readMessageEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000151
Mark Slee4af6ed72006-10-25 19:02:49 +0000152 virtual uint32_t readStructBegin(std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000153
Mark Slee4af6ed72006-10-25 19:02:49 +0000154 virtual uint32_t readStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000155
Mark Slee4af6ed72006-10-25 19:02:49 +0000156 virtual uint32_t readFieldBegin(std::string& name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000157 TType& fieldType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000158 int16_t& fieldId) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000159
Mark Slee4af6ed72006-10-25 19:02:49 +0000160 virtual uint32_t readFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000161
Mark Slee4af6ed72006-10-25 19:02:49 +0000162 virtual uint32_t readMapBegin(TType& keyType,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000163 TType& valType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000164 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000165
Mark Slee4af6ed72006-10-25 19:02:49 +0000166 virtual uint32_t readMapEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000167
Mark Slee4af6ed72006-10-25 19:02:49 +0000168 virtual uint32_t readListBegin(TType& elemType,
169 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000170
Mark Slee4af6ed72006-10-25 19:02:49 +0000171 virtual uint32_t readListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000172
Mark Slee4af6ed72006-10-25 19:02:49 +0000173 virtual uint32_t readSetBegin(TType& elemType,
174 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000175
Mark Slee4af6ed72006-10-25 19:02:49 +0000176 virtual uint32_t readSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000177
Mark Slee4af6ed72006-10-25 19:02:49 +0000178 virtual uint32_t readBool(bool& value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000179
Mark Slee4af6ed72006-10-25 19:02:49 +0000180 virtual uint32_t readByte(int8_t& byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000181
Mark Slee4af6ed72006-10-25 19:02:49 +0000182 virtual uint32_t readI16(int16_t& i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000183
Mark Slee4af6ed72006-10-25 19:02:49 +0000184 virtual uint32_t readI32(int32_t& i32) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000185
Mark Slee4af6ed72006-10-25 19:02:49 +0000186 virtual uint32_t readI64(int64_t& i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000187
Mark Slee4af6ed72006-10-25 19:02:49 +0000188 virtual uint32_t readDouble(double& dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000189
Mark Slee4af6ed72006-10-25 19:02:49 +0000190 virtual uint32_t readString(std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000191
192 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000193 * Method to arbitrarily skip over data.
Mark Sleee8540632006-05-30 09:24:40 +0000194 */
Mark Slee4af6ed72006-10-25 19:02:49 +0000195 uint32_t skip(TType type) {
Mark Slee8d7e1f62006-06-07 06:48:56 +0000196 switch (type) {
Mark Slee78f58e22006-09-02 04:17:07 +0000197 case T_BOOL:
198 {
199 bool boolv;
Mark Slee4af6ed72006-10-25 19:02:49 +0000200 return readBool(boolv);
Mark Slee78f58e22006-09-02 04:17:07 +0000201 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000202 case T_BYTE:
203 {
Mark Slee78f58e22006-09-02 04:17:07 +0000204 int8_t bytev;
Mark Slee4af6ed72006-10-25 19:02:49 +0000205 return readByte(bytev);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000206 }
Mark Sleecfc01932006-09-01 22:18:16 +0000207 case T_I16:
Mark Slee8d7e1f62006-06-07 06:48:56 +0000208 {
Mark Sleecfc01932006-09-01 22:18:16 +0000209 int16_t i16;
Mark Slee4af6ed72006-10-25 19:02:49 +0000210 return readI16(i16);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000211 }
212 case T_I32:
213 {
214 int32_t i32;
Mark Slee4af6ed72006-10-25 19:02:49 +0000215 return readI32(i32);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000216 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000217 case T_I64:
218 {
219 int64_t i64;
Mark Slee4af6ed72006-10-25 19:02:49 +0000220 return readI64(i64);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000221 }
Mark Sleec98d0502006-09-06 02:42:25 +0000222 case T_DOUBLE:
223 {
224 double dub;
Mark Slee4af6ed72006-10-25 19:02:49 +0000225 return readDouble(dub);
Mark Sleec98d0502006-09-06 02:42:25 +0000226 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000227 case T_STRING:
228 {
229 std::string str;
Mark Slee4af6ed72006-10-25 19:02:49 +0000230 return readString(str);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000231 }
232 case T_STRUCT:
233 {
234 uint32_t result = 0;
235 std::string name;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000236 int16_t fid;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000237 TType ftype;
Mark Slee4af6ed72006-10-25 19:02:49 +0000238 result += readStructBegin(name);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000239 while (true) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000240 result += readFieldBegin(name, ftype, fid);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000241 if (ftype == T_STOP) {
242 break;
243 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000244 result += skip(ftype);
245 result += readFieldEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000246 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000247 result += readStructEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000248 return result;
249 }
250 case T_MAP:
251 {
252 uint32_t result = 0;
253 TType keyType;
254 TType valType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000255 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000256 result += readMapBegin(keyType, valType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000257 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000258 result += skip(keyType);
259 result += skip(valType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000260 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000261 result += readMapEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000262 return result;
263 }
264 case T_SET:
265 {
266 uint32_t result = 0;
267 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000268 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000269 result += readSetBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000270 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000271 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000272 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000273 result += readSetEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000274 return result;
275 }
276 case T_LIST:
277 {
278 uint32_t result = 0;
279 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000280 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000281 result += readListBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000282 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000283 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000284 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000285 result += readListEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000286 return result;
287 }
288 default:
289 return 0;
290 }
291 }
Mark Sleee8540632006-05-30 09:24:40 +0000292
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000293 inline shared_ptr<TTransport> getTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000294 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000295 }
296
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000297 // TODO: remove these two calls, they are for backwards
298 // compatibility
299 inline shared_ptr<TTransport> getInputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000300 return ptrans_;
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000301 }
302 inline shared_ptr<TTransport> getOutputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000303 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000304 }
305
Mark Sleee8540632006-05-30 09:24:40 +0000306 protected:
Mark Slee43b6c632007-02-07 00:54:17 +0000307 TProtocol(shared_ptr<TTransport> ptrans):
308 ptrans_(ptrans) {
309 trans_ = ptrans.get();
310 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000311
Mark Slee43b6c632007-02-07 00:54:17 +0000312 shared_ptr<TTransport> ptrans_;
313 TTransport* trans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000314
315 private:
Mark Sleee8540632006-05-30 09:24:40 +0000316 TProtocol() {}
317};
318
Mark Slee4af6ed72006-10-25 19:02:49 +0000319/**
320 * Constructs input and output protocol objects given transports.
321 */
322class TProtocolFactory {
323 public:
324 TProtocolFactory() {}
325
326 virtual ~TProtocolFactory() {}
327
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000328 virtual boost::shared_ptr<TProtocol> getProtocol(boost::shared_ptr<TTransport> trans) = 0;
Mark Slee4af6ed72006-10-25 19:02:49 +0000329};
330
Marc Slemko6f038a72006-08-03 18:58:09 +0000331}}} // facebook::thrift::protocol
332
Mark Sleef5f2be42006-09-05 21:05:31 +0000333#endif // #define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1