1 /*
2  * Licensed to the Apache Software Foundation (ASF) under one
3  * or more contributor license agreements. See the NOTICE file
4  * distributed with this work for additional information
5  * regarding copyright ownership. The ASF licenses this file
6  * to you under the Apache License, Version 2.0 (the
7  * "License"); you may not use this file except in compliance
8  * with the License. You may obtain a copy of the License at
9  *
10  *   http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing,
13  * software distributed under the License is distributed on an
14  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15  * KIND, either express or implied. See the License for the
16  * specific language governing permissions and limitations
17  * under the License.
18  */
19 
20 /**
21  * Defines the basic interface for a Thrift protocol and associated exception
22  * types.
23  *
24  * Most parts of the protocol API are typically not used in client code, as
25  * the actual serialization code is generated by thrift.codegen.* – the only
26  * interesting thing usually is that there are protocols which can be created
27  * from transports and passed around.
28  */
29 module thrift.protocol.base;
30 
31 import thrift.base;
32 import thrift.transport.base;
33 
34 /**
35  * The field types Thrift protocols support.
36  */
37 enum TType : byte {
38   STOP   = 0, /// Used to mark the end of a sequence of fields.
39   VOID   = 1, ///
40   BOOL   = 2, ///
41   BYTE   = 3, ///
42   DOUBLE = 4, ///
43   I16    = 6, ///
44   I32    = 8, ///
45   I64    = 10, ///
46   STRING = 11, ///
47   STRUCT = 12, ///
48   MAP    = 13, ///
49   SET    = 14, ///
50   LIST   = 15 ///
51 }
52 
53 /**
54  * Types of Thrift RPC messages.
55  */
56 enum TMessageType : byte {
57   CALL = 1, /// Call of a normal, two-way RPC method.
58   REPLY = 2, /// Reply to a normal method call.
59   EXCEPTION = 3, /// Reply to a method call if target raised a TApplicationException.
60   ONEWAY = 4 /// Call of a one-way RPC method which is not followed by a reply.
61 }
62 
63 /**
64  * Descriptions of Thrift entities.
65  */
66 struct TField {
67   string name;
68   TType type;
69   short id;
70 }
71 
72 /// ditto
73 struct TList {
74   TType elemType;
75   size_t size;
76 }
77 
78 /// ditto
79 struct TMap {
80   TType keyType;
81   TType valueType;
82   size_t size;
83 }
84 
85 /// ditto
86 struct TMessage {
87   string name;
88   TMessageType type;
89   int seqid;
90 }
91 
92 /// ditto
93 struct TSet {
94   TType elemType;
95   size_t size;
96 }
97 
98 /// ditto
99 struct TStruct {
100   string name;
101 }
102 
103 /**
104  * Interface for a Thrift protocol implementation. Essentially, it defines
105  * a way of reading and writing all the base types, plus a mechanism for
106  * writing out structs with indexed fields.
107  *
108  * TProtocol objects should not be shared across multiple encoding contexts,
109  * as they may need to maintain internal state in some protocols (e.g. JSON).
110  * Note that is is acceptable for the TProtocol module to do its own internal
111  * buffered reads/writes to the underlying TTransport where appropriate (i.e.
112  * when parsing an input XML stream, reading could be batched rather than
113  * looking ahead character by character for a close tag).
114  */
115 interface TProtocol {
116   /// The underlying transport used by the protocol.
117   TTransport transport() @property;
118 
119   /*
120    * Writing methods.
121    */
122 
123   void writeBool(bool b); ///
124   void writeByte(byte b); ///
125   void writeI16(short i16); ///
126   void writeI32(int i32); ///
127   void writeI64(long i64); ///
128   void writeDouble(double dub); ///
129   void writeString(string str); ///
130   void writeBinary(ubyte[] buf); ///
131 
132   void writeMessageBegin(TMessage message); ///
133   void writeMessageEnd(); ///
134   void writeStructBegin(TStruct tstruct); ///
135   void writeStructEnd(); ///
136   void writeFieldBegin(TField field); ///
137   void writeFieldEnd(); ///
138   void writeFieldStop(); ///
139   void writeListBegin(TList list); ///
140   void writeListEnd(); ///
141   void writeMapBegin(TMap map); ///
142   void writeMapEnd(); ///
143   void writeSetBegin(TSet set); ///
144   void writeSetEnd(); ///
145 
146   /*
147    * Reading methods.
148    */
149 
150   bool readBool(); ///
151   byte readByte(); ///
152   short readI16(); ///
153   int readI32(); ///
154   long readI64(); ///
155   double readDouble(); ///
156   string readString(); ///
157   ubyte[] readBinary(); ///
158 
159   TMessage readMessageBegin(); ///
160   void readMessageEnd(); ///
161   TStruct readStructBegin(); ///
162   void readStructEnd(); ///
163   TField readFieldBegin(); ///
164   void readFieldEnd(); ///
165   TList readListBegin(); ///
166   void readListEnd(); ///
167   TMap readMapBegin(); ///
168   void readMapEnd(); ///
169   TSet readSetBegin(); ///
170   void readSetEnd(); ///
171 
172   /**
173    * Reset any internal state back to a blank slate, if the protocol is
174    * stateful.
175    */
176   void reset();
177 }
178 
179 /**
180  * true if T is a TProtocol.
181  */
182 template isTProtocol(T) {
183   enum isTProtocol = is(T : TProtocol);
184 }
185 
186 unittest {
187   static assert(isTProtocol!TProtocol);
188   static assert(!isTProtocol!void);
189 }
190 
191 /**
192  * Creates a protocol operating on a given transport.
193  */
194 interface TProtocolFactory {
195   ///
196   TProtocol getProtocol(TTransport trans);
197 }
198 
199 /**
200  * A protocol-level exception.
201  */
202 class TProtocolException : TException {
203   /// The possible exception types.
204   enum Type {
205     UNKNOWN, ///
206     INVALID_DATA, ///
207     NEGATIVE_SIZE, ///
208     SIZE_LIMIT, ///
209     BAD_VERSION, ///
210     NOT_IMPLEMENTED ///
211   }
212 
213   ///
214   this(Type type, string file = __FILE__, size_t line = __LINE__, Throwable next = null) {
215     static string msgForType(Type type) {
216       switch (type) {
217         case Type.UNKNOWN: return "Unknown protocol exception";
218         case Type.INVALID_DATA: return "Invalid data";
219         case Type.NEGATIVE_SIZE: return "Negative size";
220         case Type.SIZE_LIMIT: return "Exceeded size limit";
221         case Type.BAD_VERSION: return "Invalid version";
222         case Type.NOT_IMPLEMENTED: return "Not implemented";
223         default: return "(Invalid exception type)";
224       }
225     }
226     this(msgForType(type), type, file, line, next);
227   }
228 
229   ///
230   this(string msg, string file = __FILE__, size_t line = __LINE__,
231     Throwable next = null)
232   {
233     this(msg, Type.UNKNOWN, file, line, next);
234   }
235 
236   ///
237   this(string msg, Type type, string file = __FILE__, size_t line = __LINE__,
238     Throwable next = null)
239   {
240     super(msg, file, line, next);
241     type_ = type;
242   }
243 
244   ///
245   Type type() const @property {
246     return type_;
247   }
248 
249 protected:
250   Type type_;
251 }
252 
253 /**
254  * Skips a field of the given type on the protocol.
255  *
256  * The main purpose of skip() is to allow treating struct and container types,
257  * (where multiple primitive types have to be skipped) the same as scalar types
258  * in generated code.
259  */
260 void skip(Protocol)(Protocol prot, TType type) if (is(Protocol : TProtocol)) {
261   final switch (type) {
262     case TType.BOOL:
263       prot.readBool();
264       break;
265 
266     case TType.BYTE:
267       prot.readByte();
268       break;
269 
270     case TType.I16:
271       prot.readI16();
272       break;
273 
274     case TType.I32:
275       prot.readI32();
276       break;
277 
278     case TType.I64:
279       prot.readI64();
280       break;
281 
282     case TType.DOUBLE:
283       prot.readDouble();
284       break;
285 
286     case TType.STRING:
287       prot.readBinary();
288       break;
289 
290     case TType.STRUCT:
291       prot.readStructBegin();
292       while (true) {
293         auto f = prot.readFieldBegin();
294         if (f.type == TType.STOP) break;
295         skip(prot, f.type);
296         prot.readFieldEnd();
297       }
298       prot.readStructEnd();
299       break;
300 
301     case TType.LIST:
302       auto l = prot.readListBegin();
303       foreach (i; 0 .. l.size) {
304         skip(prot, l.elemType);
305       }
306       prot.readListEnd();
307       break;
308 
309     case TType.MAP:
310       auto m = prot.readMapBegin();
311       foreach (i; 0 .. m.size) {
312         skip(prot, m.keyType);
313         skip(prot, m.valueType);
314       }
315       prot.readMapEnd();
316       break;
317 
318     case TType.SET:
319       auto s = prot.readSetBegin();
320       foreach (i; 0 .. s.size) {
321         skip(prot, s.elemType);
322       }
323       prot.readSetEnd();
324       break;
325     case TType.STOP: goto case;
326     case TType.VOID:
327       assert(false, "Invalid field type passed.");
328   }
329 }
330 
331 /**
332  * Application-level exception.
333  *
334  * It is thrown if an RPC call went wrong on the application layer, e.g. if
335  * the receiver does not know the method name requested or a method invoked by
336  * the service processor throws an exception not part of the Thrift API.
337  */
338 class TApplicationException : TException {
339   /// The possible exception types.
340   enum Type {
341     UNKNOWN = 0, ///
342     UNKNOWN_METHOD = 1, ///
343     INVALID_MESSAGE_TYPE = 2, ///
344     WRONG_METHOD_NAME = 3, ///
345     BAD_SEQUENCE_ID = 4, ///
346     MISSING_RESULT = 5, ///
347     INTERNAL_ERROR = 6, ///
348     PROTOCOL_ERROR = 7, ///
349     INVALID_TRANSFORM = 8, ///
350     INVALID_PROTOCOL = 9, ///
351     UNSUPPORTED_CLIENT_TYPE = 10 ///
352   }
353 
354   ///
355   this(Type type, string file = __FILE__, size_t line = __LINE__, Throwable next = null) {
356     static string msgForType(Type type) {
357       switch (type) {
358         case Type.UNKNOWN: return "Unknown application exception";
359         case Type.UNKNOWN_METHOD: return "Unknown method";
360         case Type.INVALID_MESSAGE_TYPE: return "Invalid message type";
361         case Type.WRONG_METHOD_NAME: return "Wrong method name";
362         case Type.BAD_SEQUENCE_ID: return "Bad sequence identifier";
363         case Type.MISSING_RESULT: return "Missing result";
364         case Type.INTERNAL_ERROR: return "Internal error";
365         case Type.PROTOCOL_ERROR: return "Protocol error";
366         case Type.INVALID_TRANSFORM: return "Invalid transform";
367         case Type.INVALID_PROTOCOL: return "Invalid protocol";
368         case Type.UNSUPPORTED_CLIENT_TYPE: return "Unsupported client type";
369         default: return "(Invalid exception type)";
370       }
371     }
372     this(msgForType(type), type, file, line, next);
373   }
374 
375   ///
376   this(string msg, string file = __FILE__, size_t line = __LINE__,
377     Throwable next = null)
378   {
379     this(msg, Type.UNKNOWN, file, line, next);
380   }
381 
382   ///
383   this(string msg, Type type, string file = __FILE__, size_t line = __LINE__,
384     Throwable next = null)
385   {
386     super(msg, file, line, next);
387     type_ = type;
388   }
389 
390   ///
391   Type type() @property const {
392     return type_;
393   }
394 
395   // TODO: Replace hand-written read()/write() with thrift.codegen templates.
396 
397   ///
398   void read(TProtocol iprot) {
399     iprot.readStructBegin();
400     while (true) {
401       auto f = iprot.readFieldBegin();
402       if (f.type == TType.STOP) break;
403 
404       switch (f.id) {
405         case 1:
406           if (f.type == TType.STRING) {
407             msg = iprot.readString();
408           } else {
409             skip(iprot, f.type);
410           }
411           break;
412         case 2:
413           if (f.type == TType.I32) {
414             type_ = cast(Type)iprot.readI32();
415           } else {
416             skip(iprot, f.type);
417           }
418           break;
419         default:
420           skip(iprot, f.type);
421           break;
422       }
423     }
424     iprot.readStructEnd();
425   }
426 
427   ///
428   void write(TProtocol oprot) const {
429     oprot.writeStructBegin(TStruct("TApplicationException"));
430 
431     if (msg != null) {
432       oprot.writeFieldBegin(TField("message", TType.STRING, 1));
433       oprot.writeString(msg);
434       oprot.writeFieldEnd();
435     }
436 
437     oprot.writeFieldBegin(TField("type", TType.I32, 2));
438     oprot.writeI32(type_);
439     oprot.writeFieldEnd();
440 
441     oprot.writeFieldStop();
442     oprot.writeStructEnd();
443   }
444 
445 private:
446   Type type_;
447 }