lwtarantool 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (107) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +84 -0
  3. data/ext/lwtarantool/conn.c +314 -0
  4. data/ext/lwtarantool/depend +23 -0
  5. data/ext/lwtarantool/errors.c +24 -0
  6. data/ext/lwtarantool/extconf.rb +8 -0
  7. data/ext/lwtarantool/lwtarantool.c +12 -0
  8. data/ext/lwtarantool/lwtarantool.h +38 -0
  9. data/ext/lwtarantool/request.c +124 -0
  10. data/ext/lwtarantool/vendor/msgpuck/CMakeLists.txt +73 -0
  11. data/ext/lwtarantool/vendor/msgpuck/hints.c +674 -0
  12. data/ext/lwtarantool/vendor/msgpuck/msgpuck.c +375 -0
  13. data/ext/lwtarantool/vendor/msgpuck/msgpuck.h +2195 -0
  14. data/ext/lwtarantool/vendor/msgpuck/test/CMakeLists.txt +25 -0
  15. data/ext/lwtarantool/vendor/msgpuck/test/msgpuck.c +1126 -0
  16. data/ext/lwtarantool/vendor/msgpuck/test/test.c +105 -0
  17. data/ext/lwtarantool/vendor/msgpuck/test/test.h +123 -0
  18. data/ext/lwtarantool/vendor/tarantool-c/CMakeLists.txt +89 -0
  19. data/ext/lwtarantool/vendor/tarantool-c/cmake/FindMsgPuck.cmake +18 -0
  20. data/ext/lwtarantool/vendor/tarantool-c/cmake/FindSphinx.cmake +10 -0
  21. data/ext/lwtarantool/vendor/tarantool-c/doc/CMakeLists.txt +15 -0
  22. data/ext/lwtarantool/vendor/tarantool-c/doc/source/example.c +340 -0
  23. data/ext/lwtarantool/vendor/tarantool-c/include/CMakeLists.txt +6 -0
  24. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tarantool.h +67 -0
  25. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_auth.h +72 -0
  26. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_buf.h +88 -0
  27. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_call.h +83 -0
  28. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_delete.h +52 -0
  29. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_execute.h +24 -0
  30. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_insert.h +62 -0
  31. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_io.h +67 -0
  32. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iob.h +62 -0
  33. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iter.h +301 -0
  34. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_mem.h +102 -0
  35. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_net.h +242 -0
  36. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_object.h +248 -0
  37. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_opt.h +138 -0
  38. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_ping.h +49 -0
  39. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_proto.h +295 -0
  40. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_reply.h +204 -0
  41. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_request.h +377 -0
  42. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_schema.h +165 -0
  43. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_select.h +59 -0
  44. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_stream.h +79 -0
  45. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_update.h +226 -0
  46. data/ext/lwtarantool/vendor/tarantool-c/include/tp.h +1998 -0
  47. data/ext/lwtarantool/vendor/tarantool-c/test/CMakeLists.txt +36 -0
  48. data/ext/lwtarantool/vendor/tarantool-c/test/common.c +233 -0
  49. data/ext/lwtarantool/vendor/tarantool-c/test/common.h +28 -0
  50. data/ext/lwtarantool/vendor/tarantool-c/test/plain_test.c +152 -0
  51. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_call.c +57 -0
  52. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_disconnect.c +31 -0
  53. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_tcp.c +840 -0
  54. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_unix.c +384 -0
  55. data/ext/lwtarantool/vendor/tarantool-c/test/test.c +72 -0
  56. data/ext/lwtarantool/vendor/tarantool-c/test/test.h +94 -0
  57. data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.c +6 -0
  58. data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.h +67 -0
  59. data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.c +317 -0
  60. data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.h +68 -0
  61. data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.c +300 -0
  62. data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.h +91 -0
  63. data/ext/lwtarantool/vendor/tarantool-c/third_party/mhash.h +589 -0
  64. data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.c +219 -0
  65. data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.h +24 -0
  66. data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.c +6528 -0
  67. data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.h +81 -0
  68. data/ext/lwtarantool/vendor/tarantool-c/tnt/CMakeLists.txt +83 -0
  69. data/ext/lwtarantool/vendor/tarantool-c/tnt/pmatomic.h +494 -0
  70. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.c +9 -0
  71. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.h +172 -0
  72. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_auth.c +118 -0
  73. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_buf.c +171 -0
  74. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_call.c +79 -0
  75. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_delete.c +52 -0
  76. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_execute.c +60 -0
  77. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_insert.c +60 -0
  78. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_io.c +520 -0
  79. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iob.c +75 -0
  80. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iter.c +293 -0
  81. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_mem.c +82 -0
  82. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_net.c +336 -0
  83. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_object.c +476 -0
  84. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_opt.c +117 -0
  85. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_ping.c +38 -0
  86. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_proto_internal.h +43 -0
  87. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_reply.c +300 -0
  88. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_request.c +336 -0
  89. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_schema.c +334 -0
  90. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_select.c +58 -0
  91. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_stream.c +71 -0
  92. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_update.c +247 -0
  93. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/CMakeLists.txt +69 -0
  94. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.c +173 -0
  95. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.h +58 -0
  96. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.c +329 -0
  97. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.h +119 -0
  98. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.c +189 -0
  99. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.h +47 -0
  100. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.c +163 -0
  101. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.h +50 -0
  102. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.c +163 -0
  103. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.h +50 -0
  104. data/lib/lwtarantool/connection.rb +84 -0
  105. data/lib/lwtarantool/request.rb +58 -0
  106. data/lib/lwtarantool.rb +23 -0
  107. metadata +164 -0
@@ -0,0 +1,1998 @@
1
+ #ifndef TP_H_INCLUDED
2
+ #define TP_H_INCLUDED
3
+
4
+ #include <stdlib.h>
5
+ #include <string.h>
6
+ #include <assert.h>
7
+ #include <stdarg.h>
8
+
9
+ #include <msgpuck/msgpuck.h>
10
+ #include <sha1.h>
11
+ #include <base64.h>
12
+
13
+
14
+ #ifdef __cplusplus
15
+ extern "C" {
16
+ #endif
17
+
18
+ #define tpfunction_unused __attribute__((unused))
19
+
20
+ #if !defined __GNUC_MINOR__ || defined __INTEL_COMPILER || \
21
+ defined __SUNPRO_C || defined __SUNPRO_CC
22
+ #define TP_GCC_VERSION(major, minor) 0
23
+ #else
24
+ #define TP_GCC_VERSION(major, minor) (__GNUC__ > (major) || \
25
+ (__GNUC__ == (major) && __GNUC_MINOR__ >= (minor)))
26
+ #endif
27
+
28
+ #if !defined(__has_builtin)
29
+ #define __has_builtin(x) 0 /* clang */
30
+ #endif
31
+
32
+ #if TP_GCC_VERSION(2, 9) || __has_builtin(__builtin_expect)
33
+ #define tplikely(x) __builtin_expect(!!(x), 1)
34
+ #define tpunlikely(x) __builtin_expect(!!(x), 0)
35
+ #else
36
+ #define tplikely(x) (x)
37
+ #define tpunlikely(x) (x)
38
+ #endif
39
+
40
+ /* {{{ API declaration */
41
+ struct tp;
42
+
43
+ /* available types */
44
+ enum tp_type {
45
+ TP_NIL = MP_NIL,
46
+ TP_UINT = MP_UINT,
47
+ TP_INT = MP_INT,
48
+ TP_STR = MP_STR,
49
+ TP_BIN = MP_BIN,
50
+ TP_ARRAY = MP_ARRAY,
51
+ TP_MAP = MP_MAP,
52
+ TP_BOOL = MP_BOOL,
53
+ TP_FLOAT = MP_FLOAT,
54
+ TP_DOUBLE = MP_DOUBLE,
55
+ TP_EXT = MP_EXT
56
+ };
57
+
58
+ /* header */
59
+ enum tp_header_key_t {
60
+ TP_CODE = 0x00,
61
+ TP_SYNC = 0x01
62
+ };
63
+
64
+ /* request body */
65
+ enum tp_body_key_t {
66
+ TP_SPACE = 0x10,
67
+ TP_INDEX = 0x11,
68
+ TP_LIMIT = 0x12,
69
+ TP_OFFSET = 0x13,
70
+ TP_ITERATOR = 0x14,
71
+ TP_KEY = 0x20,
72
+ TP_TUPLE = 0x21,
73
+ TP_FUNCTION = 0x22,
74
+ TP_USERNAME = 0x23,
75
+ TP_EXPRESSION = 0x27,
76
+ TP_SERVER_ID = 0x02,
77
+ TP_LSN = 0x03,
78
+ TP_TIMESTAMP = 0x04,
79
+ TP_SERVER_UUID = 0x24,
80
+ TP_CLUSTER_UUID = 0x25,
81
+ TP_VCLOCK = 0x26
82
+ };
83
+
84
+ /* response body */
85
+ enum tp_response_key_t {
86
+ TP_DATA = 0x30,
87
+ TP_ERROR = 0x31
88
+ };
89
+
90
+ /* request types */
91
+ enum tp_request_type {
92
+ TP_SELECT = 1,
93
+ TP_INSERT = 2,
94
+ TP_REPLACE = 3,
95
+ TP_UPDATE = 4,
96
+ TP_DELETE = 5,
97
+ TP_CALL = 6,
98
+ TP_AUTH = 7,
99
+ TP_EVAL = 8,
100
+ TP_PING = 64,
101
+ TP_JOIN = 65,
102
+ TP_SUBSCRIBE = 66
103
+ };
104
+
105
+ enum tp_iterator_type {
106
+ TP_ITERATOR_EQ = 0,
107
+ TP_ITERATOR_REQ,
108
+ TP_ITERATOR_ALL,
109
+ TP_ITERATOR_LT,
110
+ TP_ITERATOR_LE,
111
+ TP_ITERATOR_GE,
112
+ TP_ITERATOR_GT,
113
+ TP_ITERATOR_BITS_ALL_SET,
114
+ TP_ITERATOR_BITS_ANY_SET,
115
+ TP_ITERATOR_BITS_ALL_NON_SET,
116
+ TP_ITERATOR_OVERLAPS,
117
+ TP_ITERATOR_NEIGHBOR
118
+ };
119
+
120
+ static const uint32_t SCRAMBLE_SIZE = 20;
121
+
122
+ typedef char *(*tp_reserve)(struct tp *p, size_t req, size_t *size);
123
+
124
+ /**
125
+ * tp greetings object - points to parts of a greetings buffer.
126
+ */
127
+ struct tpgreeting {
128
+ const char *description; /* Points to a text containg tarantool name and version */
129
+ const char *salt_base64; /* Points to connection salt in base64 encoding */
130
+ };
131
+
132
+ /*
133
+ * Main tp request object - points to a request buffer.
134
+ *
135
+ * All fields except tp->p should not be accessed directly.
136
+ * Appropriate accessors should be used instead.
137
+ */
138
+ struct tp {
139
+ char *s, *p, *e; /* start, pos, end */
140
+ char *size; /* pointer to length field of current request */
141
+ char *sync; /* pointer to sync field of current request */
142
+ tp_reserve reserve; /* realloc function pointer */
143
+ void *obj; /* realloc function pointer */
144
+ };
145
+
146
+ /**
147
+ * Struct for iterating msgpack array
148
+ *
149
+ * struct tp_array_itr itr;
150
+ * if (tp_array_itr_init(&itr, buf, buf_size))
151
+ * panic("It's not a valid array!.");
152
+ * while (tp_array_itr_next(&itr)) {
153
+ * mp_print(itr.elem, itr.elem_end);
154
+ * // or do smth else with itr.elem
155
+ * }
156
+ */
157
+ struct tp_array_itr {
158
+ const char *data; /* pointer to the beginning of array */
159
+ const char *first_elem; /* pointer to the first element of array */
160
+ const char *elem; /* pointer to current element of array */
161
+ const char *elem_end; /* pointer to current element end of array */
162
+ uint32_t elem_count; /* number of elements in array */
163
+ int cur_index; /* index of current element */
164
+ };
165
+
166
+ /**
167
+ * Struct for iterating msgpack map
168
+ *
169
+ * struct tp_map_itr itr;
170
+ * if (tp_map_itr_init(&itr, buf, buf_size))
171
+ * panic("It's not a valid map!.");
172
+ * while(tp_map_itr_next(&itr)) {
173
+ * mp_print(itr.key, itr.key_end);
174
+ * mp_print(itr.value, itr.value_end);
175
+ * // or do smth else with itr.key and itr.value
176
+ * }
177
+ */
178
+ struct tp_map_itr {
179
+ const char *data; /* pointer to the beginning of map */
180
+ const char *first_key; /* pointer to the first key of map */
181
+ const char *key; /* pointer to current key of map */
182
+ const char *key_end; /* pointer to current key end */
183
+ const char *value; /* pointer to current value of map */
184
+ const char *value_end; /* pointer to current value end */
185
+ uint32_t pair_count; /* number of key-values pairs in array */
186
+ int cur_index; /* index of current pair */
187
+ };
188
+
189
+ /**
190
+ * tp response object - points to parts of a response buffer
191
+ */
192
+ struct tpresponse {
193
+ uint64_t bitmap; /* bitmap of field IDs that was read from response */
194
+ const char *buf; /* points to beginning of buffer */
195
+ uint32_t code; /* response code (0 is success, or errno if not) */
196
+ uint64_t sync; /* synchronization id */
197
+ const char *error; /* error message (NULL if not present) */
198
+ const char *error_end; /* end of error message (NULL if not present) */
199
+ const char *data; /* tuple data (NULL if not present) */
200
+ const char *data_end; /* end if tuple data (NULL if not present) */
201
+ struct tp_array_itr tuple_itr; /* internal iterator over tuples */
202
+ struct tp_array_itr field_itr; /* internal iterator over tuple fields */
203
+ };
204
+
205
+ /**
206
+ * Receive greetings from the server.
207
+ * Note, the input buffer is not copied,
208
+ * so freeing or reusing the input buffer will invalidate tpgreeting struct
209
+ * For example salt is used for authorization (tp_auth).
210
+ * Returns 0 if buffer is too small or greetings size (128) on success
211
+ */
212
+ static inline ssize_t
213
+ tp_greeting(struct tpgreeting *g, char *buf, size_t size);
214
+
215
+ /**
216
+ * Main request initialization function.
217
+ *
218
+ * buf - current buffer, may be NULL
219
+ * size - current buffer size
220
+ * reserve - reallocation function, may be NULL
221
+ * obj - pointer to be passed to the reallocation function as
222
+ * context
223
+ *
224
+ * Either a buffer pointer or a reserve function must be
225
+ * provided.
226
+ * If reserve function is provided, data must be manually freed
227
+ * when the buffer is no longer needed.
228
+ * (e.g. free(p->s) or tp_free(p) );
229
+ */
230
+ static inline void
231
+ tp_init(struct tp *p, char *buf, size_t size,
232
+ tp_reserve reserve, void *obj);
233
+
234
+ /**
235
+ * A common reallocation function, can be used
236
+ * for 'reserve' param in tp_init().
237
+ * Resizes the buffer twice the previous size using realloc().
238
+ *
239
+ * struct tp req;
240
+ * tp_init(&req, NULL, tp_realloc, NULL);
241
+ * tp_ping(&req); // will call the reallocator
242
+ *
243
+ * data must be manually freed when the buffer is no longer
244
+ * needed.
245
+ * (e.g. free(p->s) or tp_free(p) );
246
+ * if realloc will return NULL, then you must destroy previous memory.
247
+ * (e.g.
248
+ * if (tp_realloc(p, ..) == NULL) {
249
+ * tp_free(p)
250
+ * return NULL;
251
+ * }
252
+ */
253
+ tpfunction_unused static inline char *
254
+ tp_realloc(struct tp *p, size_t required, size_t *size);
255
+
256
+ /**
257
+ * Free function for use in a pair with tp_realloc.
258
+ * Don't use it when tp inited with static buffer!
259
+ */
260
+ static inline void
261
+ tp_free(struct tp *p);
262
+
263
+ /**
264
+ * Get currently allocated buffer pointer
265
+ */
266
+ static inline char *
267
+ tp_buf(struct tp *p);
268
+
269
+ /**
270
+ * Get currently allocated buffer size
271
+ */
272
+ static inline size_t
273
+ tp_size(struct tp *p);
274
+
275
+ /**
276
+ * Get the size of data in the buffer
277
+ */
278
+ static inline size_t
279
+ tp_used(struct tp *p);
280
+
281
+ /**
282
+ * Get the size available for write
283
+ */
284
+ static inline size_t
285
+ tp_unused(struct tp *p);
286
+
287
+ /**
288
+ * Create a ping request.
289
+ *
290
+ * char buf[64];
291
+ * struct tp req;
292
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
293
+ * tp_ping(&req);
294
+ */
295
+ static inline char *
296
+ tp_ping(struct tp *p);
297
+
298
+ /**
299
+ * Create an auth request.
300
+ *
301
+ * salt_base64 must be gathered from tpgreeting struct,
302
+ * that is initialized during tp_greeting call.
303
+ *
304
+ * tp_auth(p, greet.salt_base64, "admin", 5, "pass", 4);
305
+ */
306
+ static inline char *
307
+ tp_auth(struct tp *p, const char *salt_base64, const char *user, int ulen, const char *pass, int plen);
308
+
309
+ /**
310
+ * Append a select request.
311
+ *
312
+ * char buf[64];
313
+ * struct tp req;
314
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
315
+ * tp_select(&req, 0, 0, 0, 100);
316
+ * tp_key(&req, 1);
317
+ * tp_sz(&req, "key");
318
+ */
319
+ static inline char *
320
+ tp_select(struct tp *p, uint32_t space, uint32_t index,
321
+ uint32_t offset, enum tp_iterator_type iterator, uint32_t limit);
322
+
323
+ /**
324
+ * Create an insert request.
325
+ *
326
+ * char buf[64];
327
+ * struct tp req;
328
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
329
+ * tp_insert(&req, 0);
330
+ * tp_tuple(&req, 2);
331
+ * tp_sz(&req, "key");
332
+ * tp_sz(&req, "value");
333
+ */
334
+ static inline char *
335
+ tp_insert(struct tp *p, uint32_t space);
336
+
337
+ /**
338
+ * Create a replace request.
339
+ *
340
+ * char buf[64];
341
+ * struct tp req;
342
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
343
+ * tp_insert(&req, 0);
344
+ * tp_tuple(&req, 2);
345
+ * tp_sz(&req, "key");
346
+ * tp_sz(&req, "value");
347
+ */
348
+ static inline char *
349
+ tp_replace(struct tp *p, uint32_t space);
350
+
351
+ /**
352
+ * Create a delete request.
353
+ *
354
+ * char buf[64];
355
+ * struct tp req;
356
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
357
+ * tp_delete(&req, 0);
358
+ * tp_key(&req, 1);
359
+ * tp_sz(&req, "key");
360
+ */
361
+ static inline char *
362
+ tp_delete(struct tp *p, uint32_t space);
363
+
364
+ /**
365
+ * Create an update request.
366
+ *
367
+ * char buf[64];
368
+ * struct tp req;
369
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
370
+ * tp_update(&req, 0); // update of space 0
371
+ * tp_key(&req, 1); // key with one part
372
+ * tp_sz(&req, "key"); // one and only part of the key
373
+ * tp_updatebegin(&req, 2); // update with two operations
374
+ * tp_op(&req, "+", 2); // add to field 2 ..
375
+ * tp_encode_uint(&req, 1); // .. a value 1
376
+ * tp_op(&req, "=", 3); // set a field 3 ..
377
+ * tp_sz(&req, "value"); // .. a value "value"
378
+ */
379
+ static inline char *
380
+ tp_update(struct tp *p, uint32_t space);
381
+
382
+ /**
383
+ * Begin update operations.
384
+ * See tp_update description for details.
385
+ */
386
+ static inline char *
387
+ tp_updatebegin(struct tp *p, uint32_t op_count);
388
+
389
+ /**
390
+ * Add an update operation.
391
+ * See tp_update description.
392
+ * Operation op could be:
393
+ * "=" - assign operation argument to field <field_no>;
394
+ * will extend the tuple if <field_no> == <max_field_no> + 1
395
+ * "#" - delete <argument> fields starting from <field_no>
396
+ * "!" - insert <argument> before <field_no>
397
+ * The following operation(s) are only defined for integer
398
+ * types:
399
+ * "+" - add argument to field <field_no>, argument
400
+ * are integer
401
+ * "-" - subtract argument from the field <field_no>
402
+ * "&" - bitwise AND of argument and field <field_no>
403
+ * "^" - bitwise XOR of argument and field <field_no>
404
+ * "|" - bitwise OR of argument and field <field_no>
405
+ */
406
+ static inline char *
407
+ tp_op(struct tp *p, char op, uint32_t field_no);
408
+
409
+ /**
410
+ * Create a call request.
411
+ *
412
+ * char buf[64];
413
+ * struct tp req;
414
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
415
+ *
416
+ * char proc[] = "hello_proc";
417
+ * tp_call(&req, proc, sizeof(proc) - 1);
418
+ * tp_encode_array(&req, 2);
419
+ * tp_sz(&req, "arg1");
420
+ * tp_sz(&req, "arg2");
421
+ */
422
+ static inline char *
423
+ tp_call(struct tp *p, const char *function, int len);
424
+
425
+ /**
426
+ *
427
+ */
428
+ static inline char *
429
+ tp_format(struct tp *p, const char *format, ...);
430
+
431
+ /**
432
+ * Write a tuple header
433
+ * Same as tp_encode_array, added for compatibility.
434
+ */
435
+ static inline char *
436
+ tp_tuple(struct tp *p, uint32_t field_count);
437
+
438
+ /**
439
+ * Write a key header
440
+ * Same as tp_encode_array, added for compatibility.
441
+ */
442
+ static inline char *
443
+ tp_key(struct tp *p, uint32_t part_count);
444
+
445
+ /**
446
+ * Add an uint value to the request
447
+ */
448
+ static inline char *
449
+ tp_encode_uint(struct tp *p, uint64_t num);
450
+
451
+ /**
452
+ * Add an int value to the request
453
+ * the value must be less than zero
454
+ */
455
+ static inline char *
456
+ tp_encode_int(struct tp *p, int64_t num);
457
+
458
+ /**
459
+ * Add a string value to the request, with length provided.
460
+ */
461
+ static inline char *
462
+ tp_encode_str(struct tp *p, const char *str, uint32_t len);
463
+
464
+ /**
465
+ * Add a zero-end string value to the request.
466
+ */
467
+ static inline char *
468
+ tp_encode_sz(struct tp *p, const char *str);
469
+
470
+ /**
471
+ * Add a zero-end string value to the request.
472
+ * (added for compatibility with tarantool 1.5 connector)
473
+ */
474
+ static inline char *
475
+ tp_sz(struct tp *p, const char *str);
476
+
477
+ /**
478
+ * Add a nil value to the request
479
+ */
480
+ static inline char *
481
+ tp_encode_nil(struct tp *p);
482
+
483
+ /**
484
+ * Add binary data to the request.
485
+ */
486
+ static inline char *
487
+ tp_encode_bin(struct tp *p, const char *str, uint32_t len);
488
+
489
+ /**
490
+ * Add an array to the request with a given size
491
+ *
492
+ * tp_encode_array(p, 3);
493
+ * tp_encode_uint(p, 1);
494
+ * tp_encode_uint(p, 2);
495
+ * tp_encode_uint(p, 3);
496
+ */
497
+ static inline char *
498
+ tp_encode_array(struct tp *p, uint32_t size);
499
+
500
+ /**
501
+ * Add a map to the request with a given size
502
+ *
503
+ * tp_encode_array(p, 2);
504
+ * tp_encode_sz(p, "name");
505
+ * tp_encode_sz(p, "Alan");
506
+ * tp_encode_sz(p, "birth");
507
+ * tp_encode_uint(p, 1912);
508
+ */
509
+ static inline char *
510
+ tp_encode_map(struct tp *p, uint32_t size);
511
+
512
+ /**
513
+ * Add a bool value to the request.
514
+ */
515
+ static inline char *
516
+ tp_encode_bool(struct tp *p, bool val);
517
+
518
+ /**
519
+ * Add a float value to the request.
520
+ */
521
+ static inline char *
522
+ tp_encode_float(struct tp *p, float num);
523
+
524
+ /**
525
+ * Add a double float value to the request.
526
+ */
527
+ static inline char *
528
+ tp_encode_double(struct tp *p, double num);
529
+
530
+ /**
531
+ * Set the current request id.
532
+ */
533
+ static inline void
534
+ tp_reqid(struct tp *p, uint32_t reqid);
535
+
536
+ /**
537
+ * Ensure that buffer has enough space to fill size bytes, resize
538
+ * buffer if needed. Returns -1 on error, and new allocated size
539
+ * of success.
540
+ */
541
+ static inline ssize_t
542
+ tp_ensure(struct tp *p, size_t size);
543
+
544
+ /**
545
+ * Initialize struct tpresponse with a data buffer.
546
+ * Returns -1 if an error occured
547
+ * Returns 0 if buffer contains only part of the response
548
+ * Return size in bytes of the response in buffer on success
549
+ */
550
+ static inline ssize_t
551
+ tp_reply(struct tpresponse *r, const char * const buf, size_t size);
552
+
553
+ /**
554
+ * Return the current request id
555
+ */
556
+ static inline uint32_t
557
+ tp_getreqid(struct tpresponse *r);
558
+
559
+ /**
560
+ * Check if the response has a tuple.
561
+ * Automatically checked during tp_next() iteration.
562
+ */
563
+ static inline int
564
+ tp_hasdata(struct tpresponse *r);
565
+
566
+ /**
567
+ * Get tuple count in response
568
+ */
569
+ static inline uint32_t
570
+ tp_tuplecount(const struct tpresponse *r);
571
+
572
+ /**
573
+ * Skip to the next tuple or to the first tuple after rewind
574
+ */
575
+ static inline int
576
+ tp_next(struct tpresponse *r);
577
+
578
+ /**
579
+ * Check if there is one more tuple.
580
+ */
581
+ static inline int
582
+ tp_hasnext(struct tpresponse *r);
583
+
584
+ /**
585
+ * Rewind iteration to the first tuple.
586
+ * Note that initialization of tpresponse via tp_reply
587
+ * rewinds tuple iteration automatically
588
+ */
589
+ static inline void
590
+ tp_rewind(struct tpresponse *r);
591
+
592
+ /**
593
+ * Get the current tuple data, all fields.
594
+ */
595
+ static inline const char *
596
+ tp_gettuple(struct tpresponse *r);
597
+
598
+ /**
599
+ * Get the current tuple size in bytes.
600
+ */
601
+ static inline uint32_t
602
+ tp_tuplesize(struct tpresponse *r);
603
+
604
+ /**
605
+ * Get a pointer to the end of the current tuple.
606
+ */
607
+ static inline const char *
608
+ tp_tupleend(struct tpresponse *r);
609
+
610
+ /**
611
+ * Skip to the next field.
612
+ */
613
+ static inline int
614
+ tp_nextfield(struct tpresponse *r);
615
+
616
+ /**
617
+ * Get the current field.
618
+ */
619
+ static inline const char *
620
+ tp_getfield(struct tpresponse *r);
621
+
622
+ /*
623
+ * Rewind iteration to the first tuple field of the current tuple.
624
+ * Note that iterating tuples of the response
625
+ * rewinds field iteration automatically
626
+ */
627
+ static inline void
628
+ tp_rewindfield(struct tpresponse *r);
629
+
630
+ /*
631
+ * Check if the current tuple has one more field.
632
+ */
633
+ static inline int
634
+ tp_hasnextfield(struct tpresponse *r);
635
+
636
+
637
+ /**
638
+ * Get the current field size in bytes.
639
+ */
640
+ static inline uint32_t
641
+ tp_getfieldsize(struct tpresponse *r);
642
+
643
+ /*
644
+ * Determine MsgPack type by first byte of encoded data.
645
+ */
646
+ static inline enum tp_type
647
+ tp_typeof(const char c);
648
+
649
+ /**
650
+ * Read unsigned integer value
651
+ */
652
+ static inline uint64_t
653
+ tp_get_uint(const char *field);
654
+
655
+ /**
656
+ * Read signed integer value
657
+ */
658
+ static inline int64_t
659
+ tp_get_int(const char *field);
660
+
661
+ /**
662
+ * Read float value
663
+ */
664
+ static inline float
665
+ tp_get_float(const char *field);
666
+
667
+ /**
668
+ * Read double value
669
+ */
670
+ static inline double
671
+ tp_get_double(const char *field);
672
+
673
+ /**
674
+ * Read bool value
675
+ */
676
+ static inline bool
677
+ tp_get_bool(const char *field);
678
+
679
+ /**
680
+ * Read string value
681
+ */
682
+ static inline const char *
683
+ tp_get_str(const char *field, uint32_t *size);
684
+
685
+ /**
686
+ * Read binary data value
687
+ */
688
+ static inline const char *
689
+ tp_get_bin(const char *field, uint32_t *size);
690
+
691
+
692
+ /**
693
+ * Init msgpack iterator by a pointer to msgpack array beginning.
694
+ * First element will be accessible after tp_array_itr_next call.
695
+ * Returns -1 on error
696
+ */
697
+ static inline int
698
+ tp_array_itr_init(struct tp_array_itr *itr, const char *data, size_t size);
699
+
700
+ /**
701
+ * Iterate to next position.
702
+ * return true if success, or false if there are no elements left
703
+ */
704
+ static inline bool
705
+ tp_array_itr_next(struct tp_array_itr *itr);
706
+
707
+ /**
708
+ * Reset iterator to the beginning. First element will be
709
+ * accessible after tp_array_itr_next call.
710
+ * return true if success, or false if there are no elements left
711
+ */
712
+ static inline void
713
+ tp_array_itr_reset(struct tp_array_itr *itr);
714
+
715
+ /**
716
+ * Init msgpack map iterator by a pointer to msgpack map beginning.
717
+ * First element will be accessible after tp_map_itr_next call.
718
+ * Returns -1 on error
719
+ */
720
+ static inline int
721
+ tp_map_itr_init(struct tp_map_itr *itr, const char *data, size_t size);
722
+
723
+ /**
724
+ * Iterate to next position.
725
+ * return true if success, or false if there are no pairs left
726
+ */
727
+ static inline bool
728
+ tp_map_itr_next(struct tp_map_itr *itr);
729
+
730
+ /**
731
+ * Reset iterator to the beginning. First pair will be
732
+ * accessible after tp_map_itr_next call.
733
+ * return true if success, or false if there are no pairs left
734
+ */
735
+ static inline void
736
+ tp_map_itr_reset(struct tp_map_itr *itr);
737
+
738
+ /* }}} */
739
+
740
+
741
+ /* {{{ Implementation */
742
+
743
+ /**
744
+ * Receive greetings from the server.
745
+ * Note, the input buffer is not copied,
746
+ * so freeing or reusing the input buffer will invalidate tpgreeting struct
747
+ * For example salt is used for authorization (tp_auth).
748
+ * Returns 0 if buffer is too small or greetings size (128) on success
749
+ */
750
+ static inline ssize_t
751
+ tp_greeting(struct tpgreeting *g, char *buf, size_t size)
752
+ {
753
+ if (size < 128)
754
+ return 0;
755
+ g->description = buf;
756
+ g->salt_base64 = buf + 64;
757
+ return 128;
758
+ }
759
+
760
+
761
+ /**
762
+ * Get currently allocated buffer pointer
763
+ */
764
+ static inline char *
765
+ tp_buf(struct tp *p)
766
+ {
767
+ return p->s;
768
+ }
769
+
770
+ /**
771
+ * Get currently allocated buffer size
772
+ */
773
+ static inline size_t
774
+ tp_size(struct tp *p)
775
+ {
776
+ return p->e - p->s;
777
+ }
778
+
779
+ /**
780
+ * Get the size of data in the buffer
781
+ */
782
+ static inline size_t
783
+ tp_used(struct tp *p)
784
+ {
785
+ return p->p - p->s;
786
+ }
787
+
788
+ /**
789
+ * Get the size available for write
790
+ */
791
+ static inline size_t
792
+ tp_unused(struct tp *p)
793
+ {
794
+ return p->e - p->p;
795
+ }
796
+
797
+ /**
798
+ * A common reallocation function, can be used
799
+ * for 'reserve' param in tp_init().
800
+ * Resizes the buffer twice the previous size using realloc().
801
+ *
802
+ * struct tp req;
803
+ * tp_init(&req, NULL, tp_realloc, NULL);
804
+ * tp_ping(&req); // will call the reallocator
805
+ *
806
+ * data must be manually freed when the buffer is no longer
807
+ * needed.
808
+ * (e.g. free(p->s) or tp_free(p) );
809
+ * if realloc will return NULL, then you must destroy previous memory.
810
+ * (e.g.
811
+ * if (tp_realloc(p, ..) == NULL) {
812
+ * tp_free(p)
813
+ * return NULL;
814
+ * }
815
+ */
816
+ tpfunction_unused static char *
817
+ tp_realloc(struct tp *p, size_t required, size_t *size)
818
+ {
819
+ size_t toalloc = tp_size(p) * 2;
820
+ if (tpunlikely(toalloc < required))
821
+ toalloc = tp_size(p) + required;
822
+ *size = toalloc;
823
+ return (char *) realloc(p->s, toalloc);
824
+ }
825
+
826
+ /**
827
+ * Free function for use in a pair with tp_realloc.
828
+ * Don't use it when tp inited with static buffer!
829
+ */
830
+ static inline void
831
+ tp_free(struct tp *p)
832
+ {
833
+ free(p->s);
834
+ }
835
+
836
+ /**
837
+ * Main initialization function.
838
+ *
839
+ * buf - current buffer, may be NULL
840
+ * size - current buffer size
841
+ * reserve - reallocation function, may be NULL
842
+ * obj - pointer to be passed to the reallocation function as
843
+ * context
844
+ *
845
+ * Either a buffer pointer or a reserve function must be
846
+ * provided.
847
+ * If reserve function is provided, data must be manually freed
848
+ * when the buffer is no longer needed.
849
+ * (e.g. free(p->s) or tp_free(p) );
850
+ */
851
+ static inline void
852
+ tp_init(struct tp *p, char *buf, size_t size,
853
+ tp_reserve reserve, void *obj)
854
+ {
855
+ p->s = buf;
856
+ p->p = p->s;
857
+ p->e = p->s + size;
858
+ p->size = NULL;
859
+ p->sync = NULL;
860
+ p->reserve = reserve;
861
+ p->obj = obj;
862
+ }
863
+
864
+ /**
865
+ * Ensure that buffer has enough space to fill size bytes, resize
866
+ * buffer if needed. Returns -1 on error, and new allocated size
867
+ * on success.
868
+ */
869
+ static inline ssize_t
870
+ tp_ensure(struct tp *p, size_t size)
871
+ {
872
+ if (tplikely(tp_unused(p) >= size))
873
+ return 0;
874
+ if (tpunlikely(p->reserve == NULL))
875
+ return -1;
876
+ size_t sz;
877
+ char *np = p->reserve(p, size, &sz);
878
+ if (tpunlikely(np == NULL))
879
+ return -1;
880
+ if (tplikely(p->size))
881
+ p->size = (np + (((char *)p->size) - p->s));
882
+ if (tplikely(p->sync))
883
+ p->sync = (np + (((char *)p->sync) - p->s));
884
+ p->p = np + (p->p - p->s);
885
+ p->s = np;
886
+ p->e = np + sz;
887
+ return sz;
888
+ }
889
+
890
+ /**
891
+ * Accept data of specified size.
892
+ * This is a function for internal use, and is not part of an API
893
+ */
894
+ static inline char *
895
+ tp_add(struct tp *p, size_t size)
896
+ {
897
+ void *ptr = p->p;
898
+ p->p += size;
899
+ assert(p->size);
900
+ *p->size = 0xce;
901
+ *(uint32_t*)(p->size + 1) = mp_bswap_u32(p->p - p->size - 5);
902
+ return (char *) ptr;
903
+ }
904
+
905
+ /**
906
+ * Internal
907
+ * Function for getting size of header
908
+ */
909
+ static inline uint32_t
910
+ tpi_sizeof_header(uint32_t code)
911
+ {
912
+ return 5 + mp_sizeof_map(2) +
913
+ mp_sizeof_uint(TP_CODE) +
914
+ mp_sizeof_uint(code) +
915
+ mp_sizeof_uint(TP_SYNC) +
916
+ 5;
917
+ }
918
+
919
+ /**
920
+ * Internal
921
+ * Function for encoding header
922
+ */
923
+ static inline char *
924
+ tpi_encode_header(struct tp* p, uint32_t code)
925
+ {
926
+ p->size = p->p;
927
+ char *h = mp_encode_map(p->p + 5, 2);
928
+ h = mp_encode_uint(h, TP_CODE);
929
+ h = mp_encode_uint(h, code);
930
+ h = mp_encode_uint(h, TP_SYNC);
931
+ p->sync = h;
932
+ *h = 0xce;
933
+ *(uint32_t*)(h + 1) = 0;
934
+ h += 5;
935
+ return h;
936
+ }
937
+
938
+ /**
939
+ * Create a ping request.
940
+ *
941
+ * char buf[64];
942
+ * struct tp req;
943
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
944
+ * tp_ping(&req);
945
+ */
946
+ static inline char *
947
+ tp_ping(struct tp *p)
948
+ {
949
+ int hsz = tpi_sizeof_header(TP_PING);
950
+ int sz = mp_sizeof_map(0);
951
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
952
+ return NULL;
953
+ char *h = tpi_encode_header(p, TP_PING);
954
+ h = mp_encode_map(h, 0);
955
+ return tp_add(p, sz + hsz);
956
+ }
957
+
958
+ /**
959
+ * Append a select request.
960
+ *
961
+ * char buf[64];
962
+ * struct tp req;
963
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
964
+ * tp_select(&req, 0, 0, 0, 100);
965
+ * tp_key(&req, 1);
966
+ * tp_sz(&req, "key");
967
+ */
968
+ static inline char *
969
+ tp_select(struct tp *p, uint32_t space, uint32_t index,
970
+ uint32_t offset, enum tp_iterator_type iterator, uint32_t limit)
971
+ {
972
+ int hsz = tpi_sizeof_header(TP_SELECT);
973
+ int sz = mp_sizeof_map(6) +
974
+ mp_sizeof_uint(TP_SPACE) +
975
+ mp_sizeof_uint(space) +
976
+ mp_sizeof_uint(TP_INDEX) +
977
+ mp_sizeof_uint(index) +
978
+ mp_sizeof_uint(TP_OFFSET) +
979
+ mp_sizeof_uint(offset) +
980
+ mp_sizeof_uint(TP_LIMIT) +
981
+ mp_sizeof_uint(limit) +
982
+ mp_sizeof_uint(TP_ITERATOR) +
983
+ mp_sizeof_uint(iterator) +
984
+ mp_sizeof_uint(TP_KEY);
985
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
986
+ return NULL;
987
+ char *h = tpi_encode_header(p, TP_SELECT);
988
+ h = mp_encode_map(h, 6);
989
+ h = mp_encode_uint(h, TP_SPACE);
990
+ h = mp_encode_uint(h, space);
991
+ h = mp_encode_uint(h, TP_INDEX);
992
+ h = mp_encode_uint(h, index);
993
+ h = mp_encode_uint(h, TP_OFFSET);
994
+ h = mp_encode_uint(h, offset);
995
+ h = mp_encode_uint(h, TP_LIMIT);
996
+ h = mp_encode_uint(h, limit);
997
+ h = mp_encode_uint(h, TP_ITERATOR);
998
+ h = mp_encode_uint(h, iterator);
999
+ h = mp_encode_uint(h, TP_KEY);
1000
+ return tp_add(p, sz + hsz);
1001
+ }
1002
+
1003
+ /**
1004
+ * Internal
1005
+ * Function for encoding insert or replace request
1006
+ */
1007
+ static inline char *
1008
+ tpi_encode_store(struct tp *p, enum tp_request_type type, uint32_t space)
1009
+ {
1010
+ int hsz = tpi_sizeof_header(type);
1011
+ int sz = mp_sizeof_map(2) +
1012
+ mp_sizeof_uint(TP_SPACE) +
1013
+ mp_sizeof_uint(space) +
1014
+ mp_sizeof_uint(TP_TUPLE);
1015
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1016
+ return NULL;
1017
+ char *h = tpi_encode_header(p, type);
1018
+ h = mp_encode_map(h, 2);
1019
+ h = mp_encode_uint(h, TP_SPACE);
1020
+ h = mp_encode_uint(h, space);
1021
+ h = mp_encode_uint(h, TP_TUPLE);
1022
+ return tp_add(p, sz + hsz);
1023
+ }
1024
+
1025
+ /**
1026
+ * Create an insert request.
1027
+ *
1028
+ * char buf[64];
1029
+ * struct tp req;
1030
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
1031
+ * tp_insert(&req, 0);
1032
+ * tp_tuple(&req, 2);
1033
+ * tp_sz(&req, "key");
1034
+ * tp_sz(&req, "value");
1035
+ */
1036
+ static inline char *
1037
+ tp_insert(struct tp *p, uint32_t space)
1038
+ {
1039
+ return tpi_encode_store(p, TP_INSERT, space);
1040
+ }
1041
+
1042
+ /**
1043
+ * Create a replace request.
1044
+ *
1045
+ * char buf[64];
1046
+ * struct tp req;
1047
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
1048
+ * tp_insert(&req, 0);
1049
+ * tp_tuple(&req, 2);
1050
+ * tp_sz(&req, "key");
1051
+ * tp_sz(&req, "value");
1052
+ */
1053
+ static inline char *
1054
+ tp_replace(struct tp *p, uint32_t space)
1055
+ {
1056
+ return tpi_encode_store(p, TP_REPLACE, space);
1057
+ }
1058
+
1059
+ /**
1060
+ * Create a delete request.
1061
+ *
1062
+ * char buf[64];
1063
+ * struct tp req;
1064
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
1065
+ * tp_delete(&req, 0);
1066
+ * tp_key(&req, 1);
1067
+ * tp_sz(&req, "key");
1068
+ */
1069
+ static inline char *
1070
+ tp_delete(struct tp *p, uint32_t space)
1071
+ {
1072
+ int hsz = tpi_sizeof_header(TP_DELETE);
1073
+ int sz = mp_sizeof_map(2) +
1074
+ mp_sizeof_uint(TP_SPACE) +
1075
+ mp_sizeof_uint(space) +
1076
+ mp_sizeof_uint(TP_KEY);
1077
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1078
+ return NULL;
1079
+ char *h = tpi_encode_header(p, TP_DELETE);
1080
+ h = mp_encode_uint(h, TP_SPACE);
1081
+ h = mp_encode_uint(h, space);
1082
+ h = mp_encode_uint(h, TP_KEY);
1083
+ return tp_add(p, sz + hsz);
1084
+ }
1085
+
1086
+ /**
1087
+ * Create a call request.
1088
+ *
1089
+ * char buf[64];
1090
+ * struct tp req;
1091
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
1092
+ *
1093
+ * char proc[] = "hello_proc";
1094
+ * tp_call(&req, proc, sizeof(proc) - 1);
1095
+ * tp_encode_array(&req, 2);
1096
+ * tp_sz(&req, "arg1");
1097
+ * tp_sz(&req, "arg2");
1098
+ */
1099
+ static inline char *
1100
+ tp_call(struct tp *p, const char *function, int len)
1101
+ {
1102
+ int hsz = tpi_sizeof_header(TP_CALL);
1103
+ int sz = mp_sizeof_map(2) +
1104
+ mp_sizeof_uint(TP_FUNCTION) +
1105
+ mp_sizeof_str(len) +
1106
+ mp_sizeof_uint(TP_TUPLE);
1107
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1108
+ return NULL;
1109
+ char *h = tpi_encode_header(p, TP_CALL);
1110
+ h = mp_encode_map(h, 2);
1111
+ h = mp_encode_uint(h, TP_FUNCTION);
1112
+ h = mp_encode_str(h, function, len);
1113
+ h = mp_encode_uint(h, TP_TUPLE);
1114
+ return tp_add(p, sz + hsz);
1115
+ }
1116
+
1117
+ /**
1118
+ * Create an eval request.
1119
+ *
1120
+ * char buf[64];
1121
+ * struct tp req;
1122
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
1123
+ *
1124
+ * char proc[] = "hello_proc";
1125
+ * tp_eval(&req, proc, sizeof(proc) - 1);
1126
+ * tp_encode_array(&req, 2);
1127
+ * tp_sz(&req, "arg1");
1128
+ * tp_sz(&req, "arg2");
1129
+ */
1130
+ static inline char *
1131
+ tp_eval(struct tp *p, const char *expr, int len)
1132
+ {
1133
+ int hsz = tpi_sizeof_header(TP_EVAL);
1134
+ int sz = mp_sizeof_map(2) +
1135
+ mp_sizeof_uint(TP_FUNCTION) +
1136
+ mp_sizeof_str(len) +
1137
+ mp_sizeof_uint(TP_TUPLE);
1138
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1139
+ return NULL;
1140
+ char *h = tpi_encode_header(p, TP_EVAL);
1141
+ h = mp_encode_map(h, 2);
1142
+ h = mp_encode_uint(h, TP_EXPRESSION);
1143
+ h = mp_encode_str(h, expr, len);
1144
+ h = mp_encode_uint(h, TP_TUPLE);
1145
+ return tp_add(p, sz + hsz);
1146
+ }
1147
+
1148
+ /**
1149
+ * Create an update request.
1150
+ *
1151
+ * char buf[64];
1152
+ * struct tp req;
1153
+ * tp_init(&req, buf, sizeof(buf), NULL, NULL);
1154
+ * tp_update(&req, 0); // update of space 0
1155
+ * tp_key(&req, 1); // key with one part
1156
+ * tp_sz(&req, "key"); // one and only part of the key
1157
+ * tp_updatebegin(&req, 2); // update with two operations
1158
+ * tp_op(&req, "+", 2); // add to field 2 ..
1159
+ * tp_encode_uint(&req, 1); // .. a value 1
1160
+ * tp_op(&req, "=", 3); // set a field 3 ..
1161
+ * tp_sz(&req, "value"); // .. a value "value"
1162
+ */
1163
+ static inline char *
1164
+ tp_update(struct tp *p, uint32_t space)
1165
+ {
1166
+ int hsz = tpi_sizeof_header(TP_UPDATE);
1167
+ int sz = mp_sizeof_map(3) +
1168
+ mp_sizeof_uint(TP_SPACE) +
1169
+ mp_sizeof_uint(space) +
1170
+ mp_sizeof_uint(TP_KEY);
1171
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1172
+ return NULL;
1173
+ char *h = tpi_encode_header(p, TP_UPDATE);
1174
+ h = mp_encode_map(h, 3);
1175
+ h = mp_encode_uint(h, TP_SPACE);
1176
+ h = mp_encode_uint(h, space);
1177
+ h = mp_encode_uint(h, TP_KEY);
1178
+ return tp_add(p, sz + hsz);
1179
+ }
1180
+
1181
+ /**
1182
+ * Begin update operations.
1183
+ * See tp_update description for details.
1184
+ */
1185
+ static inline char *
1186
+ tp_updatebegin(struct tp *p, uint32_t op_count)
1187
+ {
1188
+ int sz = mp_sizeof_uint(TP_TUPLE) + mp_sizeof_array(op_count);
1189
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1190
+ return NULL;
1191
+ char *h = mp_encode_uint(p->p, TP_TUPLE);
1192
+ mp_encode_array(h, op_count);
1193
+ return tp_add(p, sz);
1194
+ }
1195
+
1196
+ /**
1197
+ * Add an update operation.
1198
+ * See tp_update description.
1199
+ * Operation op could be:
1200
+ * "=" - assign operation argument to field <field_no>;
1201
+ * will extend the tuple if <field_no> == <max_field_no> + 1
1202
+ * "#" - delete <argument> fields starting from <field_no>
1203
+ * "!" - insert <argument> before <field_no>
1204
+ * The following operation(s) are only defined for integer
1205
+ * types:
1206
+ * "+" - add argument to field <field_no>, argument
1207
+ * are integer
1208
+ * "-" - subtract argument from the field <field_no>
1209
+ * "&" - bitwise AND of argument and field <field_no>
1210
+ * "^" - bitwise XOR of argument and field <field_no>
1211
+ * "|" - bitwise OR of argument and field <field_no>
1212
+ */
1213
+ static inline char *
1214
+ tp_op(struct tp *p, char op, uint32_t field_no)
1215
+ {
1216
+ int sz = mp_sizeof_array(3) +
1217
+ mp_sizeof_str(1) +
1218
+ mp_sizeof_uint(field_no);
1219
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1220
+ return NULL;
1221
+ char *h = mp_encode_array(p->p, 3);
1222
+ h = mp_encode_str(h, &op, 1);
1223
+ h = mp_encode_uint(h, field_no);
1224
+ return tp_add(p, sz);
1225
+ }
1226
+
1227
+ /**
1228
+ * Add an slice operation
1229
+ * See tp_update description.
1230
+ */
1231
+ static inline char *
1232
+ tp_op_splice(struct tp *p, uint32_t field_no,
1233
+ uint32_t offset, uint32_t cut_limit,
1234
+ const char *paste, uint32_t paste_len)
1235
+ {
1236
+ int sz = mp_sizeof_array(5) +
1237
+ mp_sizeof_str(1) +
1238
+ mp_sizeof_uint(field_no) +
1239
+ mp_sizeof_uint(field_no) +
1240
+ mp_sizeof_uint(field_no) +
1241
+ mp_sizeof_str(paste_len);
1242
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1243
+ return NULL;
1244
+ char *h = mp_encode_array(p->p, 5);
1245
+ h = mp_encode_str(h, ":", 1);
1246
+ h = mp_encode_uint(h, field_no);
1247
+ h = mp_encode_uint(h, offset);
1248
+ h = mp_encode_uint(h, cut_limit);
1249
+ h = mp_encode_str(h, paste, paste_len);
1250
+ return tp_add(p, sz);
1251
+ }
1252
+
1253
+
1254
+ /**
1255
+ * Internal
1256
+ * tpi_xor
1257
+ * The function is for internal use, not part of the API
1258
+ */
1259
+ static inline void
1260
+ tpi_xor(unsigned char *to, const unsigned char *left,
1261
+ const unsigned char *right, uint32_t len)
1262
+ {
1263
+ const uint8_t *end = to + len;
1264
+ while (to < end)
1265
+ *to++= *left++ ^ *right++;
1266
+ }
1267
+
1268
+ /**
1269
+ * Internal
1270
+ * tpi_scramble_prepare
1271
+ * The function is for internal use, not part of the API
1272
+ */
1273
+ static inline void
1274
+ tpi_scramble_prepare(void *out, const void *salt, const void *password,
1275
+ int password_len)
1276
+ {
1277
+ unsigned char hash1[SCRAMBLE_SIZE];
1278
+ unsigned char hash2[SCRAMBLE_SIZE];
1279
+ SHA1_CTX ctx;
1280
+
1281
+ SHA1Init(&ctx);
1282
+ SHA1Update(&ctx, (const unsigned char *) password, password_len);
1283
+ SHA1Final(hash1, &ctx);
1284
+
1285
+ SHA1Init(&ctx);
1286
+ SHA1Update(&ctx, hash1, SCRAMBLE_SIZE);
1287
+ SHA1Final(hash2, &ctx);
1288
+
1289
+ SHA1Init(&ctx);
1290
+ SHA1Update(&ctx, (const unsigned char *) salt, SCRAMBLE_SIZE);
1291
+ SHA1Update(&ctx, hash2, SCRAMBLE_SIZE);
1292
+ SHA1Final((unsigned char *) out, &ctx);
1293
+
1294
+ tpi_xor((unsigned char *) out, hash1, (const unsigned char *) out,
1295
+ SCRAMBLE_SIZE);
1296
+ }
1297
+
1298
+ /**
1299
+ * Create an auth request.
1300
+ *
1301
+ * salt_base64 must be gathered from tpgreeting struct,
1302
+ * that is initialized during tp_greeting call.
1303
+ *
1304
+ * tp_auth(p, greet.salt_base64, "admin", 5, "pass", 4);
1305
+ */
1306
+ static inline char *
1307
+ tp_auth(struct tp *p, const char *salt_base64, const char *user,
1308
+ int ulen, const char *pass, int plen)
1309
+ {
1310
+ int hsz = tpi_sizeof_header(TP_AUTH);
1311
+ int sz = mp_sizeof_array(2) +
1312
+ mp_sizeof_str(0) +
1313
+ mp_sizeof_str(SCRAMBLE_SIZE);
1314
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1315
+ return NULL;
1316
+ char *h = tpi_encode_header(p, TP_AUTH);
1317
+ h = mp_encode_map(h, 2);
1318
+ h = mp_encode_uint(h, TP_USERNAME);
1319
+ h = mp_encode_str(h, user, ulen);
1320
+ h = mp_encode_uint(h, TP_TUPLE);
1321
+ h = mp_encode_array(h, 2);
1322
+ h = mp_encode_str(h, "chap-sha1", 9);
1323
+
1324
+ char salt[64];
1325
+ base64_decode(salt_base64, 44, salt, 64);
1326
+ char scramble[SCRAMBLE_SIZE];
1327
+ tpi_scramble_prepare(scramble, salt, pass, plen);
1328
+ h = mp_encode_str(h, scramble, SCRAMBLE_SIZE);
1329
+
1330
+ return tp_add(p, sz + hsz);
1331
+ }
1332
+
1333
+ /**
1334
+ * Create an deauth (auth as a guest) request.
1335
+ *
1336
+ * tp_deauth(p);
1337
+ */
1338
+ static inline char *
1339
+ tp_deauth(struct tp *p)
1340
+ {
1341
+ int hsz = tpi_sizeof_header(TP_AUTH);
1342
+ int sz = mp_sizeof_array(0);
1343
+ if (tpunlikely(tp_ensure(p, sz + hsz) == -1))
1344
+ return NULL;
1345
+ p->size = p->p;
1346
+ char *h = tpi_encode_header(p, TP_AUTH);
1347
+ h = mp_encode_map(h, 2);
1348
+ h = mp_encode_uint(h, TP_USERNAME);
1349
+ h = mp_encode_str(h, "guest", 5);
1350
+ h = mp_encode_uint(h, TP_TUPLE);
1351
+ h = mp_encode_array(h, 0);
1352
+
1353
+ return tp_add(p, sz + hsz);
1354
+ }
1355
+
1356
+ /**
1357
+ * Set the current request id.
1358
+ */
1359
+ static inline void
1360
+ tp_reqid(struct tp *p, uint32_t reqid)
1361
+ {
1362
+ assert(p->sync != NULL);
1363
+ char *h = p->sync;
1364
+ *h = 0xce;
1365
+ *(uint32_t*)(h + 1) = mp_bswap_u32(reqid);
1366
+ }
1367
+
1368
+ /**
1369
+ * Add a nil value to the request
1370
+ */
1371
+ static inline char *
1372
+ tp_encode_nil(struct tp *p)
1373
+ {
1374
+ int sz = mp_sizeof_nil();
1375
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1376
+ return NULL;
1377
+ mp_encode_nil(p->p);
1378
+ return tp_add(p, sz);
1379
+ }
1380
+
1381
+ /**
1382
+ * Add an uint value to the request
1383
+ */
1384
+ static inline char *
1385
+ tp_encode_uint(struct tp *p, uint64_t num)
1386
+ {
1387
+ int sz = mp_sizeof_uint(num);
1388
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1389
+ return NULL;
1390
+ mp_encode_uint(p->p, num);
1391
+ return tp_add(p, sz);
1392
+ }
1393
+
1394
+ /**
1395
+ * Add an int value to the request
1396
+ * the value must be less than zero
1397
+ */
1398
+ static inline char *
1399
+ tp_encode_int(struct tp *p, int64_t num)
1400
+ {
1401
+ int sz = mp_sizeof_int(num);
1402
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1403
+ return NULL;
1404
+ mp_encode_int(p->p, num);
1405
+ return tp_add(p, sz);
1406
+ }
1407
+
1408
+ /**
1409
+ * Add a string value to the request, with length provided.
1410
+ */
1411
+ static inline char *
1412
+ tp_encode_str(struct tp *p, const char *str, uint32_t len)
1413
+ {
1414
+ int sz = mp_sizeof_str(len);
1415
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1416
+ return NULL;
1417
+ mp_encode_str(p->p, str, len);
1418
+ return tp_add(p, sz);
1419
+ }
1420
+
1421
+ /**
1422
+ * Add a zero-end string value to the request.
1423
+ */
1424
+ static inline char *
1425
+ tp_encode_sz(struct tp *p, const char *str)
1426
+ {
1427
+ uint32_t len = (uint32_t)strlen(str);
1428
+ int sz = mp_sizeof_str(len);
1429
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1430
+ return NULL;
1431
+ mp_encode_str(p->p, str, len);
1432
+ return tp_add(p, sz);
1433
+ }
1434
+
1435
+ /**
1436
+ * Add a zero-end string value to the request.
1437
+ * (added for compatibility with tarantool 1.5 connector)
1438
+ */
1439
+ static inline char *
1440
+ tp_sz(struct tp *p, const char *str)
1441
+ {
1442
+ return tp_encode_sz(p, str);
1443
+ }
1444
+
1445
+ /**
1446
+ * Add binary data to the request.
1447
+ */
1448
+ static inline char *
1449
+ tp_encode_bin(struct tp *p, const char *str, uint32_t len)
1450
+ {
1451
+ int sz = mp_sizeof_bin(len);
1452
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1453
+ return NULL;
1454
+ mp_encode_bin(p->p, str, len);
1455
+ return tp_add(p, sz);
1456
+ }
1457
+
1458
+ /**
1459
+ * Add an array to the request with a given size
1460
+ *
1461
+ * tp_encode_array(p, 3);
1462
+ * tp_encode_uint(p, 1);
1463
+ * tp_encode_uint(p, 2);
1464
+ * tp_encode_uint(p, 3);
1465
+ */
1466
+ static inline char *
1467
+ tp_encode_array(struct tp *p, uint32_t size)
1468
+ {
1469
+ int sz = mp_sizeof_array(size);
1470
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1471
+ return NULL;
1472
+ mp_encode_array(p->p, size);
1473
+ return tp_add(p, sz);
1474
+ }
1475
+
1476
+ /**
1477
+ * Add a map to the request with a given size
1478
+ *
1479
+ * tp_encode_array(p, 2);
1480
+ * tp_encode_sz(p, "name");
1481
+ * tp_encode_sz(p, "Alan");
1482
+ * tp_encode_sz(p, "birth");
1483
+ * tp_encode_uint(p, 1912);
1484
+ */
1485
+ static inline char *
1486
+ tp_encode_map(struct tp *p, uint32_t size)
1487
+ {
1488
+ int sz = mp_sizeof_map(size);
1489
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1490
+ return NULL;
1491
+ mp_encode_map(p->p, size);
1492
+ return tp_add(p, sz);
1493
+ }
1494
+
1495
+ /**
1496
+ * Add a bool value to the request.
1497
+ */
1498
+ static inline char *
1499
+ tp_encode_bool(struct tp *p, bool val)
1500
+ {
1501
+ int sz = mp_sizeof_bool(val);
1502
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1503
+ return NULL;
1504
+ mp_encode_bool(p->p, val);
1505
+ return tp_add(p, sz);
1506
+ }
1507
+
1508
+ /**
1509
+ * Add a float value to the request.
1510
+ */
1511
+ static inline char *
1512
+ tp_encode_float(struct tp *p, float num)
1513
+ {
1514
+ int sz = mp_sizeof_float(num);
1515
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1516
+ return NULL;
1517
+ mp_encode_float(p->p, num);
1518
+ return tp_add(p, sz);
1519
+ }
1520
+
1521
+ /**
1522
+ * Add a double float value to the request.
1523
+ */
1524
+ static inline char *
1525
+ tp_encode_double(struct tp *p, double num)
1526
+ {
1527
+ int sz = mp_sizeof_double(num);
1528
+ if (tpunlikely(tp_ensure(p, sz) == -1))
1529
+ return NULL;
1530
+ mp_encode_double(p->p, num);
1531
+ return tp_add(p, sz);
1532
+ }
1533
+
1534
+ /**
1535
+ *
1536
+ */
1537
+ static inline char *
1538
+ tp_format(struct tp *p, const char *format, ...)
1539
+ {
1540
+ va_list args;
1541
+ va_start(args, format);
1542
+ size_t unused = tp_unused(p);
1543
+ size_t sz = mp_vformat(p->p, unused, format, args);
1544
+ if (sz > unused) {
1545
+ if (tpunlikely(tp_ensure(p, sz) == -1)) {
1546
+ va_end(args);
1547
+ return NULL;
1548
+ }
1549
+ mp_vformat(p->p, unused, format, args);
1550
+ }
1551
+ va_end(args);
1552
+ return tp_add(p, sz);
1553
+ }
1554
+
1555
+
1556
+ /**
1557
+ * Write a tuple header
1558
+ * Same as tp_encode_array, added for compatibility.
1559
+ */
1560
+ static inline char *
1561
+ tp_tuple(struct tp *p, uint32_t field_count)
1562
+ {
1563
+ return tp_encode_array(p, field_count);
1564
+ }
1565
+
1566
+ /**
1567
+ * Write a key header
1568
+ * Same as tp_encode_array, added for compatibility.
1569
+ */
1570
+ static inline char *
1571
+ tp_key(struct tp *p, uint32_t part_count)
1572
+ {
1573
+ return tp_encode_array(p, part_count);
1574
+ }
1575
+
1576
+ /**
1577
+ * Init msgpack iterator by a pointer to msgpack array begin.
1578
+ * First element will be accessible after tp_array_itr_next call.
1579
+ * Returns -1 on error
1580
+ */
1581
+ static inline int
1582
+ tp_array_itr_init(struct tp_array_itr *itr, const char *data, size_t size)
1583
+ {
1584
+ memset(itr, 0, sizeof(*itr));
1585
+ if (size == 0 || mp_typeof(*data) != MP_ARRAY)
1586
+ return -1;
1587
+ const char *e = data;
1588
+ if (mp_check(&e, data + size))
1589
+ return -1;
1590
+ itr->data = data;
1591
+ itr->first_elem = data;
1592
+ itr->elem_count = mp_decode_array(&itr->first_elem);
1593
+ itr->cur_index = -1;
1594
+ return 0;
1595
+ }
1596
+
1597
+ /**
1598
+ * Iterate to next position.
1599
+ * return true if success, or false if there are no elements left
1600
+ */
1601
+ static inline bool
1602
+ tp_array_itr_next(struct tp_array_itr *itr)
1603
+ {
1604
+ itr->cur_index++;
1605
+ if ((uint32_t)itr->cur_index >= itr->elem_count)
1606
+ return false;
1607
+ if (itr->cur_index == 0)
1608
+ itr->elem = itr->first_elem;
1609
+ else
1610
+ itr->elem = itr->elem_end;
1611
+ itr->elem_end = itr->elem;
1612
+ mp_next(&itr->elem_end);
1613
+ return true;
1614
+ }
1615
+
1616
+ /**
1617
+ * Reset iterator to the beginning. First element will be
1618
+ * accessible after tp_array_itr_next call.
1619
+ * return true if success, or false if there are no elements left
1620
+ */
1621
+ static inline void
1622
+ tp_array_itr_reset(struct tp_array_itr *itr)
1623
+ {
1624
+ itr->cur_index = -1;
1625
+ itr->elem = 0;
1626
+ itr->elem_end = 0;
1627
+ }
1628
+
1629
+ /**
1630
+ * Init msgpack map iterator by a pointer to msgpack map begin.
1631
+ * First element will be accessible after tp_map_itr_next call.
1632
+ * Returns -1 on error
1633
+ */
1634
+ static inline int
1635
+ tp_map_itr_init(struct tp_map_itr *itr, const char *data, size_t size)
1636
+ {
1637
+ memset(itr, 0, sizeof(*itr));
1638
+ if (size == 0 || mp_typeof(*data) != MP_MAP)
1639
+ return -1;
1640
+ const char *e = data;
1641
+ if (mp_check(&e, data + size))
1642
+ return -1;
1643
+ itr->data = data;
1644
+ itr->first_key = data;
1645
+ itr->pair_count = mp_decode_map(&itr->first_key);
1646
+ itr->cur_index = -1;
1647
+ return 0;
1648
+ }
1649
+
1650
+ /**
1651
+ * Iterate to next position.
1652
+ * return true if success, or false if there are no pairs left
1653
+ */
1654
+ static inline bool
1655
+ tp_map_itr_next(struct tp_map_itr *itr)
1656
+ {
1657
+ itr->cur_index++;
1658
+ if ((uint32_t)itr->cur_index >= itr->pair_count)
1659
+ return false;
1660
+ if (itr->cur_index == 0)
1661
+ itr->key = itr->first_key;
1662
+ else
1663
+ itr->key = itr->value_end;
1664
+ itr->key_end = itr->key;
1665
+ mp_next(&itr->key_end);
1666
+ itr->value = itr->key_end;
1667
+ itr->value_end = itr->value;
1668
+ mp_next(&itr->value_end);
1669
+ return true;
1670
+ }
1671
+
1672
+ /**
1673
+ * Reset iterator to the beginning. First pair will be
1674
+ * accessible after tp_map_itr_next call.
1675
+ * return true if success, or false if there are no pairs left
1676
+ */
1677
+ static inline void
1678
+ tp_map_itr_reset(struct tp_map_itr *itr)
1679
+ {
1680
+ itr->cur_index = -1;
1681
+ itr->key = 0;
1682
+ itr->key_end = 0;
1683
+ itr->value = 0;
1684
+ itr->value_end = 0;
1685
+ }
1686
+
1687
+ /**
1688
+ * Initialize struct tpresponse with a data buffer.
1689
+ * Returns -1 if an error occured
1690
+ * Returns 0 if buffer contains only part of the response
1691
+ * Return size in bytes of the response in buffer on success
1692
+ */
1693
+ static inline ssize_t
1694
+ tp_reply(struct tpresponse *r, const char * const buf, size_t size)
1695
+ {
1696
+ memset(r, 0, sizeof(*r));
1697
+ if (size == 0)
1698
+ return 0;
1699
+ const char *p = buf;
1700
+ /* len */
1701
+ const char *test = p;
1702
+ if (mp_check(&test, buf + size))
1703
+ return -1;
1704
+ if (mp_typeof(*p) != MP_UINT)
1705
+ return -1;
1706
+ uint32_t len = mp_decode_uint(&p);
1707
+ if (size < len + (uint32_t)(p - buf))
1708
+ return 0;
1709
+ /* header */
1710
+ test = p;
1711
+ if (mp_check(&test, buf + size))
1712
+ return -1;
1713
+ if (mp_typeof(*p) != MP_MAP)
1714
+ return -1;
1715
+ uint32_t n = mp_decode_map(&p);
1716
+ while (n-- > 0) {
1717
+ if (mp_typeof(*p) != MP_UINT)
1718
+ return -1;
1719
+ uint32_t key = mp_decode_uint(&p);
1720
+ if (mp_typeof(*p) != MP_UINT)
1721
+ return -1;
1722
+ switch (key) {
1723
+ case TP_SYNC:
1724
+ if (mp_typeof(*p) != MP_UINT)
1725
+ return -1;
1726
+ r->sync = mp_decode_uint(&p);
1727
+ break;
1728
+ case TP_CODE:
1729
+ if (mp_typeof(*p) != MP_UINT)
1730
+ return -1;
1731
+ r->code = mp_decode_uint(&p);
1732
+ break;
1733
+ default:
1734
+ mp_next(&p);
1735
+ break;
1736
+ }
1737
+ r->bitmap |= (1ULL << key);
1738
+ }
1739
+
1740
+ /* body */
1741
+ if (p == buf + len + 5)
1742
+ return len + 5; /* no body */
1743
+ test = p;
1744
+ if (mp_check(&test, buf + size))
1745
+ return -1;
1746
+ if (mp_typeof(*p) != MP_MAP)
1747
+ return -1;
1748
+ n = mp_decode_map(&p);
1749
+ while (n-- > 0) {
1750
+ uint32_t key = mp_decode_uint(&p);
1751
+ switch (key) {
1752
+ case TP_ERROR: {
1753
+ if (mp_typeof(*p) != MP_STR)
1754
+ return -1;
1755
+ uint32_t elen = 0;
1756
+ r->error = mp_decode_str(&p, &elen);
1757
+ r->error_end = r->error + elen;
1758
+ break;
1759
+ }
1760
+ case TP_DATA: {
1761
+ if (mp_typeof(*p) != MP_ARRAY)
1762
+ return -1;
1763
+ r->data = p;
1764
+ mp_next(&p);
1765
+ r->data_end = p;
1766
+ break;
1767
+ }
1768
+ default:
1769
+ mp_next(&p);
1770
+ break;
1771
+ }
1772
+ r->bitmap |= (1ULL << key);
1773
+ }
1774
+ if (r->data) {
1775
+ if (tp_array_itr_init(&r->tuple_itr, r->data, r->data_end - r->data))
1776
+ return -1;
1777
+ }
1778
+ return p - buf;
1779
+ }
1780
+
1781
+ /**
1782
+ * Return the current response id
1783
+ */
1784
+ static inline uint32_t
1785
+ tp_getreqid(struct tpresponse *r)
1786
+ {
1787
+ return r->sync;
1788
+ }
1789
+
1790
+ /**
1791
+ * Check if the response has a tuple.
1792
+ * Automatically checked during tp_next() iteration.
1793
+ */
1794
+ static inline int
1795
+ tp_hasdata(struct tpresponse *r)
1796
+ {
1797
+ return r->tuple_itr.elem_count > 0;
1798
+ }
1799
+
1800
+ /**
1801
+ * Get tuple count in response
1802
+ */
1803
+ static inline uint32_t
1804
+ tp_tuplecount(const struct tpresponse *r)
1805
+ {
1806
+ return r->tuple_itr.elem_count;
1807
+ }
1808
+
1809
+ /**
1810
+ * Rewind iteration to the first tuple.
1811
+ * Note that initialization of tpresponse via tp_reply
1812
+ * rewinds tuple iteration automatically
1813
+ */
1814
+ static inline void
1815
+ tp_rewind(struct tpresponse *r)
1816
+ {
1817
+ tp_array_itr_reset(&r->tuple_itr);
1818
+ memset(&r->field_itr, 0, sizeof(r->field_itr));
1819
+ }
1820
+
1821
+ /**
1822
+ * Skip to the next tuple or to the first tuple after rewind
1823
+ */
1824
+ static inline int
1825
+ tp_next(struct tpresponse *r)
1826
+ {
1827
+ if (!tp_array_itr_next(&r->tuple_itr)) {
1828
+ memset(&r->field_itr, 0, sizeof(r->field_itr));
1829
+ return 0;
1830
+ }
1831
+ tp_array_itr_init(&r->field_itr, r->tuple_itr.elem, r->tuple_itr.elem_end - r->tuple_itr.elem);
1832
+ return 1;
1833
+
1834
+ }
1835
+
1836
+ /**
1837
+ * Check if there is one more tuple.
1838
+ */
1839
+ static inline int
1840
+ tp_hasnext(struct tpresponse *r)
1841
+ {
1842
+ return (uint32_t)(r->tuple_itr.cur_index + 1) < r->tuple_itr.elem_count;
1843
+ }
1844
+
1845
+ /**
1846
+ * Get the current tuple data, all fields.
1847
+ */
1848
+ static inline const char *
1849
+ tp_gettuple(struct tpresponse *r)
1850
+ {
1851
+ return r->tuple_itr.elem;
1852
+ }
1853
+
1854
+ /**
1855
+ * Get the current tuple size in bytes.
1856
+ */
1857
+ static inline uint32_t
1858
+ tp_tuplesize(struct tpresponse *r)
1859
+ {
1860
+ return (uint32_t)(r->tuple_itr.elem_end - r->tuple_itr.elem);
1861
+ }
1862
+
1863
+ /**
1864
+ * Get a pointer to the end of the current tuple.
1865
+ */
1866
+ static inline const char *
1867
+ tp_tupleend(struct tpresponse *r)
1868
+ {
1869
+ return r->tuple_itr.elem_end;
1870
+ }
1871
+
1872
+ /*
1873
+ * Rewind iteration to the first tuple field of the current tuple.
1874
+ * Note that iterating tuples of the response
1875
+ * rewinds field iteration automatically
1876
+ */
1877
+ static inline void
1878
+ tp_rewindfield(struct tpresponse *r)
1879
+ {
1880
+ tp_array_itr_reset(&r->field_itr);
1881
+ }
1882
+
1883
+ /**
1884
+ * Skip to the next field.
1885
+ */
1886
+ static inline int
1887
+ tp_nextfield(struct tpresponse *r)
1888
+ {
1889
+ return tp_array_itr_next(&r->field_itr);
1890
+ }
1891
+
1892
+ /*
1893
+ * Check if the current tuple has one more field.
1894
+ */
1895
+ static inline int
1896
+ tp_hasnextfield(struct tpresponse *r)
1897
+ {
1898
+ return (uint32_t)(r->field_itr.cur_index + 1) < r->field_itr.elem_count;
1899
+ }
1900
+
1901
+
1902
+ /**
1903
+ * Get the current field.
1904
+ */
1905
+ static inline const char *
1906
+ tp_getfield(struct tpresponse *r)
1907
+ {
1908
+ return r->field_itr.elem;
1909
+ }
1910
+
1911
+ /**
1912
+ * Get the current field size in bytes.
1913
+ */
1914
+ static inline uint32_t
1915
+ tp_getfieldsize(struct tpresponse *r)
1916
+ {
1917
+ return (uint32_t)(r->field_itr.elem_end - r->field_itr.elem);
1918
+ }
1919
+
1920
+ /*
1921
+ * Determine MsgPack type by first byte of encoded data.
1922
+ */
1923
+ static inline enum tp_type
1924
+ tp_typeof(const char c)
1925
+ {
1926
+ return (enum tp_type) mp_typeof(c);
1927
+ }
1928
+
1929
+ /**
1930
+ * Read unsigned integer value
1931
+ */
1932
+ static inline uint64_t
1933
+ tp_get_uint(const char *field)
1934
+ {
1935
+ return mp_decode_uint(&field);
1936
+ }
1937
+
1938
+ /**
1939
+ * Read signed integer value
1940
+ */
1941
+ static inline int64_t
1942
+ tp_get_int(const char *field)
1943
+ {
1944
+ return mp_decode_int(&field);
1945
+ }
1946
+
1947
+ /**
1948
+ * Read float value
1949
+ */
1950
+ static inline float
1951
+ tp_get_float(const char *field)
1952
+ {
1953
+ return mp_decode_float(&field);
1954
+ }
1955
+
1956
+ /**
1957
+ * Read double value
1958
+ */
1959
+ static inline double
1960
+ tp_get_double(const char *field)
1961
+ {
1962
+ return mp_decode_double(&field);
1963
+ }
1964
+
1965
+ /**
1966
+ * Read bool value
1967
+ */
1968
+ static inline bool
1969
+ tp_get_bool(const char *field)
1970
+ {
1971
+ return mp_decode_bool(&field);
1972
+ }
1973
+
1974
+ /**
1975
+ * Read string value
1976
+ */
1977
+ static inline const char *
1978
+ tp_get_str(const char *field, uint32_t *size)
1979
+ {
1980
+ return mp_decode_str(&field, size);
1981
+ }
1982
+
1983
+ /**
1984
+ * Read binary data value
1985
+ */
1986
+ static inline const char *
1987
+ tp_get_bin(const char *field, uint32_t *size)
1988
+ {
1989
+ return mp_decode_bin(&field, size);
1990
+ }
1991
+
1992
+ /* }}} */
1993
+
1994
+ #ifdef __cplusplus
1995
+ } /* extern "C" */
1996
+ #endif
1997
+
1998
+ #endif