blob: e04a2840208ac189c2ec5814bda8770bc13f686d [file] [log] [blame]
Mark Sleef5f2be42006-09-05 21:05:31 +00001#ifndef _THRIFT_PROTOCOL_TPROTOCOL_H_
2#define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1
Mark Sleee8540632006-05-30 09:24:40 +00003
Marc Slemko16698852006-08-04 03:16:10 +00004#include <transport/TTransport.h>
Mark Sleef9831082007-02-20 20:59:21 +00005#include <protocol/TProtocolException.h>
Marc Slemko16698852006-08-04 03:16:10 +00006
7#include <boost/shared_ptr.hpp>
8
Mark Slee8d7e1f62006-06-07 06:48:56 +00009#include <netinet/in.h>
Mark Sleee8540632006-05-30 09:24:40 +000010#include <sys/types.h>
11#include <string>
12#include <map>
13
Marc Slemko6f038a72006-08-03 18:58:09 +000014namespace facebook { namespace thrift { namespace protocol {
15
Marc Slemko16698852006-08-04 03:16:10 +000016using namespace boost;
17
Marc Slemko6f038a72006-08-03 18:58:09 +000018using namespace facebook::thrift::transport;
19
Mark Sleedc8a2a22006-09-19 22:20:18 +000020#if __BYTE_ORDER == __BIG_ENDIAN
21#define ntohll(n) (n)
22#define htonll(n) (n)
23#else
24#define ntohll(n) ( (((unsigned long long)ntohl(n)) << 32) + ntohl(n >> 32) )
25#define htonll(n) ( (((unsigned long long)htonl(n)) << 32) + htonl(n >> 32) )
26#endif
Mark Slee8d7e1f62006-06-07 06:48:56 +000027
Mark Sleee8540632006-05-30 09:24:40 +000028/**
Mark Slee8d7e1f62006-06-07 06:48:56 +000029 * Enumerated definition of the types that the Thrift protocol supports.
30 * Take special note of the T_END type which is used specifically to mark
31 * the end of a sequence of fields.
32 */
33enum TType {
Marc Slemkod42a2c22006-08-10 03:30:18 +000034 T_STOP = 0,
Marc Slemko5b126d62006-08-11 23:03:42 +000035 T_VOID = 1,
36 T_BOOL = 2,
37 T_BYTE = 3,
Mark Sleecfc01932006-09-01 22:18:16 +000038 T_I08 = 3,
Marc Slemko5b126d62006-08-11 23:03:42 +000039 T_I16 = 6,
Marc Slemko5b126d62006-08-11 23:03:42 +000040 T_I32 = 8,
41 T_U64 = 9,
42 T_I64 = 10,
Mark Sleec98d0502006-09-06 02:42:25 +000043 T_DOUBLE = 4,
Marc Slemko5b126d62006-08-11 23:03:42 +000044 T_STRING = 11,
Marc Slemkod97eb612006-08-24 23:37:36 +000045 T_UTF7 = 11,
46 T_STRUCT = 12,
47 T_MAP = 13,
48 T_SET = 14,
49 T_LIST = 15,
50 T_UTF8 = 16,
51 T_UTF16 = 17
Mark Slee8d7e1f62006-06-07 06:48:56 +000052};
53
54/**
Mark Sleef5f2be42006-09-05 21:05:31 +000055 * Enumerated definition of the message types that the Thrift protocol
56 * supports.
Marc Slemko16698852006-08-04 03:16:10 +000057 */
58enum TMessageType {
59 T_CALL = 1,
Mark Sleef9831082007-02-20 20:59:21 +000060 T_REPLY = 2,
61 T_EXCEPTION = 3
Marc Slemko16698852006-08-04 03:16:10 +000062};
63
64/**
Mark Sleee8540632006-05-30 09:24:40 +000065 * Abstract class for a thrift protocol driver. These are all the methods that
66 * a protocol must implement. Essentially, there must be some way of reading
67 * and writing all the base types, plus a mechanism for writing out structs
68 * with indexed fields. Also notice that all methods are strictly const. This
69 * is by design. Protcol impelementations may NOT keep state, because the
70 * same TProtocol object may be used simultaneously by multiple threads. This
71 * theoretically introduces some limititations into the possible protocol
72 * formats, but with the benefit of performance, clarity, and simplicity.
73 *
74 * @author Mark Slee <mcslee@facebook.com>
75 */
76class TProtocol {
77 public:
78 virtual ~TProtocol() {}
79
80 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +000081 * Writing functions.
Mark Sleee8540632006-05-30 09:24:40 +000082 */
83
Mark Slee4af6ed72006-10-25 19:02:49 +000084 virtual uint32_t writeMessageBegin(const std::string name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +000085 const TMessageType messageType,
Mark Slee4af6ed72006-10-25 19:02:49 +000086 const int32_t seqid) = 0;
Marc Slemko16698852006-08-04 03:16:10 +000087
Mark Slee4af6ed72006-10-25 19:02:49 +000088 virtual uint32_t writeMessageEnd() = 0;
Marc Slemko16698852006-08-04 03:16:10 +000089
90
Mark Slee4af6ed72006-10-25 19:02:49 +000091 virtual uint32_t writeStructBegin(const std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +000092
Mark Slee4af6ed72006-10-25 19:02:49 +000093 virtual uint32_t writeStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +000094
Mark Slee4af6ed72006-10-25 19:02:49 +000095 virtual uint32_t writeFieldBegin(const std::string& name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +000096 const TType fieldType,
Mark Slee4af6ed72006-10-25 19:02:49 +000097 const int16_t fieldId) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +000098
Mark Slee4af6ed72006-10-25 19:02:49 +000099 virtual uint32_t writeFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000100
Mark Slee4af6ed72006-10-25 19:02:49 +0000101 virtual uint32_t writeFieldStop() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000102
Mark Slee4af6ed72006-10-25 19:02:49 +0000103 virtual uint32_t writeMapBegin(const TType keyType,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000104 const TType valType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000105 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000106
Mark Slee4af6ed72006-10-25 19:02:49 +0000107 virtual uint32_t writeMapEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000108
Mark Slee4af6ed72006-10-25 19:02:49 +0000109 virtual uint32_t writeListBegin(const TType elemType,
110 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000111
Mark Slee4af6ed72006-10-25 19:02:49 +0000112 virtual uint32_t writeListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000113
Mark Slee4af6ed72006-10-25 19:02:49 +0000114 virtual uint32_t writeSetBegin(const TType elemType,
115 const uint32_t size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000116
Mark Slee4af6ed72006-10-25 19:02:49 +0000117 virtual uint32_t writeSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000118
Mark Slee4af6ed72006-10-25 19:02:49 +0000119 virtual uint32_t writeBool(const bool value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000120
Mark Slee4af6ed72006-10-25 19:02:49 +0000121 virtual uint32_t writeByte(const int8_t byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000122
Mark Slee4af6ed72006-10-25 19:02:49 +0000123 virtual uint32_t writeI16(const int16_t i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000124
Mark Slee4af6ed72006-10-25 19:02:49 +0000125 virtual uint32_t writeI32(const int32_t i32) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000126
Mark Slee4af6ed72006-10-25 19:02:49 +0000127 virtual uint32_t writeI64(const int64_t i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000128
Mark Slee4af6ed72006-10-25 19:02:49 +0000129 virtual uint32_t writeDouble(const double dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000130
Mark Slee4af6ed72006-10-25 19:02:49 +0000131 virtual uint32_t writeString(const std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000132
133 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000134 * Reading functions
Mark Sleee8540632006-05-30 09:24:40 +0000135 */
136
Mark Slee4af6ed72006-10-25 19:02:49 +0000137 virtual uint32_t readMessageBegin(std::string& name,
Marc Slemko91f67482006-08-11 23:58:57 +0000138 TMessageType& messageType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000139 int32_t& seqid) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000140
Mark Slee4af6ed72006-10-25 19:02:49 +0000141 virtual uint32_t readMessageEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000142
Mark Slee4af6ed72006-10-25 19:02:49 +0000143 virtual uint32_t readStructBegin(std::string& name) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000144
Mark Slee4af6ed72006-10-25 19:02:49 +0000145 virtual uint32_t readStructEnd() = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000146
Mark Slee4af6ed72006-10-25 19:02:49 +0000147 virtual uint32_t readFieldBegin(std::string& name,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000148 TType& fieldType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000149 int16_t& fieldId) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000150
Mark Slee4af6ed72006-10-25 19:02:49 +0000151 virtual uint32_t readFieldEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000152
Mark Slee4af6ed72006-10-25 19:02:49 +0000153 virtual uint32_t readMapBegin(TType& keyType,
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000154 TType& valType,
Mark Slee4af6ed72006-10-25 19:02:49 +0000155 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000156
Mark Slee4af6ed72006-10-25 19:02:49 +0000157 virtual uint32_t readMapEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000158
Mark Slee4af6ed72006-10-25 19:02:49 +0000159 virtual uint32_t readListBegin(TType& elemType,
160 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000161
Mark Slee4af6ed72006-10-25 19:02:49 +0000162 virtual uint32_t readListEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000163
Mark Slee4af6ed72006-10-25 19:02:49 +0000164 virtual uint32_t readSetBegin(TType& elemType,
165 uint32_t& size) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000166
Mark Slee4af6ed72006-10-25 19:02:49 +0000167 virtual uint32_t readSetEnd() = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000168
Mark Slee4af6ed72006-10-25 19:02:49 +0000169 virtual uint32_t readBool(bool& value) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000170
Mark Slee4af6ed72006-10-25 19:02:49 +0000171 virtual uint32_t readByte(int8_t& byte) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000172
Mark Slee4af6ed72006-10-25 19:02:49 +0000173 virtual uint32_t readI16(int16_t& i16) = 0;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000174
Mark Slee4af6ed72006-10-25 19:02:49 +0000175 virtual uint32_t readI32(int32_t& i32) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000176
Mark Slee4af6ed72006-10-25 19:02:49 +0000177 virtual uint32_t readI64(int64_t& i64) = 0;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000178
Mark Slee4af6ed72006-10-25 19:02:49 +0000179 virtual uint32_t readDouble(double& dub) = 0;
Mark Sleec98d0502006-09-06 02:42:25 +0000180
Mark Slee4af6ed72006-10-25 19:02:49 +0000181 virtual uint32_t readString(std::string& str) = 0;
Mark Sleee8540632006-05-30 09:24:40 +0000182
183 /**
Mark Slee8d7e1f62006-06-07 06:48:56 +0000184 * Method to arbitrarily skip over data.
Mark Sleee8540632006-05-30 09:24:40 +0000185 */
Mark Slee4af6ed72006-10-25 19:02:49 +0000186 uint32_t skip(TType type) {
Mark Slee8d7e1f62006-06-07 06:48:56 +0000187 switch (type) {
Mark Slee78f58e22006-09-02 04:17:07 +0000188 case T_BOOL:
189 {
190 bool boolv;
Mark Slee4af6ed72006-10-25 19:02:49 +0000191 return readBool(boolv);
Mark Slee78f58e22006-09-02 04:17:07 +0000192 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000193 case T_BYTE:
194 {
Mark Slee78f58e22006-09-02 04:17:07 +0000195 int8_t bytev;
Mark Slee4af6ed72006-10-25 19:02:49 +0000196 return readByte(bytev);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000197 }
Mark Sleecfc01932006-09-01 22:18:16 +0000198 case T_I16:
Mark Slee8d7e1f62006-06-07 06:48:56 +0000199 {
Mark Sleecfc01932006-09-01 22:18:16 +0000200 int16_t i16;
Mark Slee4af6ed72006-10-25 19:02:49 +0000201 return readI16(i16);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000202 }
203 case T_I32:
204 {
205 int32_t i32;
Mark Slee4af6ed72006-10-25 19:02:49 +0000206 return readI32(i32);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000207 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000208 case T_I64:
209 {
210 int64_t i64;
Mark Slee4af6ed72006-10-25 19:02:49 +0000211 return readI64(i64);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000212 }
Mark Sleec98d0502006-09-06 02:42:25 +0000213 case T_DOUBLE:
214 {
215 double dub;
Mark Slee4af6ed72006-10-25 19:02:49 +0000216 return readDouble(dub);
Mark Sleec98d0502006-09-06 02:42:25 +0000217 }
Mark Slee8d7e1f62006-06-07 06:48:56 +0000218 case T_STRING:
219 {
220 std::string str;
Mark Slee4af6ed72006-10-25 19:02:49 +0000221 return readString(str);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000222 }
223 case T_STRUCT:
224 {
225 uint32_t result = 0;
226 std::string name;
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000227 int16_t fid;
Mark Slee8d7e1f62006-06-07 06:48:56 +0000228 TType ftype;
Mark Slee4af6ed72006-10-25 19:02:49 +0000229 result += readStructBegin(name);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000230 while (true) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000231 result += readFieldBegin(name, ftype, fid);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000232 if (ftype == T_STOP) {
233 break;
234 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000235 result += skip(ftype);
236 result += readFieldEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000237 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000238 result += readStructEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000239 return result;
240 }
241 case T_MAP:
242 {
243 uint32_t result = 0;
244 TType keyType;
245 TType valType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000246 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000247 result += readMapBegin(keyType, valType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000248 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000249 result += skip(keyType);
250 result += skip(valType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000251 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000252 result += readMapEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000253 return result;
254 }
255 case T_SET:
256 {
257 uint32_t result = 0;
258 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000259 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000260 result += readSetBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000261 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000262 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000263 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000264 result += readSetEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000265 return result;
266 }
267 case T_LIST:
268 {
269 uint32_t result = 0;
270 TType elemType;
Marc Slemkob09f5882006-08-23 22:03:34 +0000271 uint32_t i, size;
Mark Slee4af6ed72006-10-25 19:02:49 +0000272 result += readListBegin(elemType, size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000273 for (i = 0; i < size; i++) {
Mark Slee4af6ed72006-10-25 19:02:49 +0000274 result += skip(elemType);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000275 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000276 result += readListEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000277 return result;
278 }
279 default:
280 return 0;
281 }
282 }
Mark Sleee8540632006-05-30 09:24:40 +0000283
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000284 inline shared_ptr<TTransport> getTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000285 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000286 }
287
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000288 // TODO: remove these two calls, they are for backwards
289 // compatibility
290 inline shared_ptr<TTransport> getInputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000291 return ptrans_;
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000292 }
293 inline shared_ptr<TTransport> getOutputTransport() {
Mark Slee43b6c632007-02-07 00:54:17 +0000294 return ptrans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000295 }
296
Mark Sleee8540632006-05-30 09:24:40 +0000297 protected:
Mark Slee43b6c632007-02-07 00:54:17 +0000298 TProtocol(shared_ptr<TTransport> ptrans):
299 ptrans_(ptrans) {
300 trans_ = ptrans.get();
301 }
Mark Slee4af6ed72006-10-25 19:02:49 +0000302
Mark Slee43b6c632007-02-07 00:54:17 +0000303 shared_ptr<TTransport> ptrans_;
304 TTransport* trans_;
Mark Slee4af6ed72006-10-25 19:02:49 +0000305
306 private:
Mark Sleee8540632006-05-30 09:24:40 +0000307 TProtocol() {}
308};
309
Mark Slee4af6ed72006-10-25 19:02:49 +0000310/**
311 * Constructs input and output protocol objects given transports.
312 */
313class TProtocolFactory {
314 public:
315 TProtocolFactory() {}
316
317 virtual ~TProtocolFactory() {}
318
Aditya Agarwal9abb0d62007-01-24 22:53:54 +0000319 virtual boost::shared_ptr<TProtocol> getProtocol(boost::shared_ptr<TTransport> trans) = 0;
Mark Slee4af6ed72006-10-25 19:02:49 +0000320};
321
Marc Slemko6f038a72006-08-03 18:58:09 +0000322}}} // facebook::thrift::protocol
323
Mark Sleef5f2be42006-09-05 21:05:31 +0000324#endif // #define _THRIFT_PROTOCOL_TPROTOCOL_H_ 1