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.
- checksums.yaml +7 -0
- data/README.md +84 -0
- data/ext/lwtarantool/conn.c +314 -0
- data/ext/lwtarantool/depend +23 -0
- data/ext/lwtarantool/errors.c +24 -0
- data/ext/lwtarantool/extconf.rb +8 -0
- data/ext/lwtarantool/lwtarantool.c +12 -0
- data/ext/lwtarantool/lwtarantool.h +38 -0
- data/ext/lwtarantool/request.c +124 -0
- data/ext/lwtarantool/vendor/msgpuck/CMakeLists.txt +73 -0
- data/ext/lwtarantool/vendor/msgpuck/hints.c +674 -0
- data/ext/lwtarantool/vendor/msgpuck/msgpuck.c +375 -0
- data/ext/lwtarantool/vendor/msgpuck/msgpuck.h +2195 -0
- data/ext/lwtarantool/vendor/msgpuck/test/CMakeLists.txt +25 -0
- data/ext/lwtarantool/vendor/msgpuck/test/msgpuck.c +1126 -0
- data/ext/lwtarantool/vendor/msgpuck/test/test.c +105 -0
- data/ext/lwtarantool/vendor/msgpuck/test/test.h +123 -0
- data/ext/lwtarantool/vendor/tarantool-c/CMakeLists.txt +89 -0
- data/ext/lwtarantool/vendor/tarantool-c/cmake/FindMsgPuck.cmake +18 -0
- data/ext/lwtarantool/vendor/tarantool-c/cmake/FindSphinx.cmake +10 -0
- data/ext/lwtarantool/vendor/tarantool-c/doc/CMakeLists.txt +15 -0
- data/ext/lwtarantool/vendor/tarantool-c/doc/source/example.c +340 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/CMakeLists.txt +6 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tarantool.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_auth.h +72 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_buf.h +88 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_call.h +83 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_delete.h +52 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_execute.h +24 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_insert.h +62 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_io.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iob.h +62 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iter.h +301 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_mem.h +102 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_net.h +242 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_object.h +248 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_opt.h +138 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_ping.h +49 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_proto.h +295 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_reply.h +204 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_request.h +377 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_schema.h +165 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_select.h +59 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_stream.h +79 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_update.h +226 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tp.h +1998 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/CMakeLists.txt +36 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/common.c +233 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/common.h +28 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/plain_test.c +152 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_call.c +57 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_disconnect.c +31 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_tcp.c +840 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_unix.c +384 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/test.c +72 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/test.h +94 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.c +6 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.c +317 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.h +68 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.c +300 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.h +91 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/mhash.h +589 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.c +219 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.h +24 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.c +6528 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.h +81 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/CMakeLists.txt +83 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/pmatomic.h +494 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.c +9 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.h +172 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_auth.c +118 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_buf.c +171 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_call.c +79 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_delete.c +52 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_execute.c +60 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_insert.c +60 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_io.c +520 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iob.c +75 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iter.c +293 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_mem.c +82 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_net.c +336 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_object.c +476 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_opt.c +117 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_ping.c +38 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_proto_internal.h +43 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_reply.c +300 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_request.c +336 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_schema.c +334 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_select.c +58 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_stream.c +71 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_update.c +247 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/CMakeLists.txt +69 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.c +173 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.h +58 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.c +329 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.h +119 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.c +189 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.h +47 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.c +163 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.h +50 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.c +163 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.h +50 -0
- data/lib/lwtarantool/connection.rb +84 -0
- data/lib/lwtarantool/request.rb +58 -0
- data/lib/lwtarantool.rb +23 -0
- metadata +164 -0
@@ -0,0 +1,300 @@
|
|
1
|
+
/*
|
2
|
+
* Redistribution and use in source and binary forms, with or
|
3
|
+
* without modification, are permitted provided that the following
|
4
|
+
* conditions are met:
|
5
|
+
*
|
6
|
+
* 1. Redistributions of source code must retain the above
|
7
|
+
* copyright notice, this list of conditions and the
|
8
|
+
* following disclaimer.
|
9
|
+
*
|
10
|
+
* 2. Redistributions in binary form must reproduce the above
|
11
|
+
* copyright notice, this list of conditions and the following
|
12
|
+
* disclaimer in the documentation and/or other materials
|
13
|
+
* provided with the distribution.
|
14
|
+
*
|
15
|
+
* THIS SOFTWARE IS PROVIDED BY <COPYRIGHT HOLDER> ``AS IS'' AND
|
16
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
17
|
+
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
18
|
+
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
19
|
+
* <COPYRIGHT HOLDER> OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
20
|
+
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
21
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
22
|
+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
23
|
+
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
24
|
+
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
26
|
+
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
27
|
+
* SUCH DAMAGE.
|
28
|
+
*/
|
29
|
+
|
30
|
+
#include <stddef.h>
|
31
|
+
#include <stdint.h>
|
32
|
+
#include <string.h>
|
33
|
+
|
34
|
+
#include <sys/types.h>
|
35
|
+
|
36
|
+
#include <msgpuck.h>
|
37
|
+
|
38
|
+
#include <tarantool/tnt_mem.h>
|
39
|
+
#include <tarantool/tnt_proto.h>
|
40
|
+
#include <tarantool/tnt_reply.h>
|
41
|
+
|
42
|
+
struct tnt_reply *tnt_reply_init(struct tnt_reply *r) {
|
43
|
+
int alloc = (r == NULL);
|
44
|
+
if (alloc) {
|
45
|
+
r = tnt_mem_alloc(sizeof(struct tnt_reply));
|
46
|
+
if (!r) return NULL;
|
47
|
+
}
|
48
|
+
memset(r, 0, sizeof(struct tnt_reply));
|
49
|
+
r->alloc = alloc;
|
50
|
+
return r;
|
51
|
+
}
|
52
|
+
|
53
|
+
void tnt_reply_free(struct tnt_reply *r) {
|
54
|
+
if (r->buf) {
|
55
|
+
tnt_mem_free((void *)r->buf);
|
56
|
+
r->buf = NULL;
|
57
|
+
}
|
58
|
+
if (r->alloc) tnt_mem_free(r);
|
59
|
+
}
|
60
|
+
|
61
|
+
int tnt_reply_from(struct tnt_reply *r, tnt_reply_t rcv, void *ptr) {
|
62
|
+
/* cleanup, before processing response */
|
63
|
+
int alloc = r->alloc;
|
64
|
+
memset(r, 0 , sizeof(struct tnt_reply));
|
65
|
+
r->alloc = alloc;
|
66
|
+
/* reading iproto header */
|
67
|
+
char length[TNT_REPLY_IPROTO_HDR_SIZE]; const char *data = (const char *)length;
|
68
|
+
if (rcv(ptr, length, sizeof(length)) == -1)
|
69
|
+
goto rollback;
|
70
|
+
if (mp_typeof(*length) != MP_UINT)
|
71
|
+
goto rollback;
|
72
|
+
size_t size = mp_decode_uint(&data);
|
73
|
+
r->buf = tnt_mem_alloc(size);
|
74
|
+
r->buf_size = size;
|
75
|
+
if (r->buf == NULL)
|
76
|
+
goto rollback;
|
77
|
+
if(rcv(ptr, (char *)r->buf, size) == -1)
|
78
|
+
goto rollback;
|
79
|
+
size_t hdr_length;
|
80
|
+
if (tnt_reply_hdr0(r, r->buf, r->buf_size, &hdr_length) != 0)
|
81
|
+
goto rollback;
|
82
|
+
if (size == (size_t)hdr_length)
|
83
|
+
return 0; /* no body */
|
84
|
+
if (tnt_reply_body0(r, r->buf + hdr_length, r->buf_size - hdr_length, NULL) != 0)
|
85
|
+
goto rollback;
|
86
|
+
|
87
|
+
return 0;
|
88
|
+
rollback:
|
89
|
+
if (r->buf) tnt_mem_free((void *)r->buf);
|
90
|
+
alloc = r->alloc;
|
91
|
+
memset(r, 0, sizeof(struct tnt_reply));
|
92
|
+
r->alloc = alloc;
|
93
|
+
return -1;
|
94
|
+
}
|
95
|
+
|
96
|
+
static ssize_t tnt_reply_cb(void *ptr[2], char *buf, ssize_t size) {
|
97
|
+
char *src = ptr[0];
|
98
|
+
ssize_t *off = ptr[1];
|
99
|
+
memcpy(buf, src + *off, size);
|
100
|
+
*off += size;
|
101
|
+
return size;
|
102
|
+
}
|
103
|
+
|
104
|
+
static int
|
105
|
+
tnt_reply_len(const char *buf, size_t size, size_t *len)
|
106
|
+
{
|
107
|
+
if (size < TNT_REPLY_IPROTO_HDR_SIZE) {
|
108
|
+
*len = TNT_REPLY_IPROTO_HDR_SIZE - size;
|
109
|
+
return 1;
|
110
|
+
}
|
111
|
+
const char *p = buf;
|
112
|
+
if (mp_typeof(*p) != MP_UINT)
|
113
|
+
return -1;
|
114
|
+
size_t length = mp_decode_uint(&p);
|
115
|
+
if (size < length + TNT_REPLY_IPROTO_HDR_SIZE) {
|
116
|
+
*len = (length + TNT_REPLY_IPROTO_HDR_SIZE) - size;
|
117
|
+
return 1;
|
118
|
+
}
|
119
|
+
*len = length + TNT_REPLY_IPROTO_HDR_SIZE;
|
120
|
+
return 0;
|
121
|
+
}
|
122
|
+
|
123
|
+
int
|
124
|
+
tnt_reply_hdr0(struct tnt_reply *r, const char *buf, size_t size, size_t *off) {
|
125
|
+
const char *test = buf;
|
126
|
+
const char *p = buf;
|
127
|
+
if (mp_check(&test, p + size))
|
128
|
+
return -1;
|
129
|
+
if (mp_typeof(*p) != MP_MAP)
|
130
|
+
return -1;
|
131
|
+
|
132
|
+
uint32_t n = mp_decode_map(&p);
|
133
|
+
uint64_t sync = 0, code = 0, schema_id = 0, bitmap = 0;
|
134
|
+
while (n-- > 0) {
|
135
|
+
if (mp_typeof(*p) != MP_UINT)
|
136
|
+
return -1;
|
137
|
+
uint32_t key = mp_decode_uint(&p);
|
138
|
+
if (mp_typeof(*p) != MP_UINT)
|
139
|
+
return -1;
|
140
|
+
switch (key) {
|
141
|
+
case TNT_SYNC:
|
142
|
+
sync = mp_decode_uint(&p);
|
143
|
+
break;
|
144
|
+
case TNT_CODE:
|
145
|
+
code = mp_decode_uint(&p);
|
146
|
+
break;
|
147
|
+
case TNT_SCHEMA_ID:
|
148
|
+
schema_id = mp_decode_uint(&p);
|
149
|
+
break;
|
150
|
+
default:
|
151
|
+
return -1;
|
152
|
+
}
|
153
|
+
bitmap |= (1ULL << key);
|
154
|
+
}
|
155
|
+
if (r) {
|
156
|
+
r->sync = sync;
|
157
|
+
r->code = code & ((1 << 15) - 1);
|
158
|
+
r->schema_id = schema_id;
|
159
|
+
r->bitmap = bitmap;
|
160
|
+
}
|
161
|
+
if (off)
|
162
|
+
*off = p - buf;
|
163
|
+
return 0;
|
164
|
+
}
|
165
|
+
|
166
|
+
int
|
167
|
+
tnt_reply_body0(struct tnt_reply *r, const char *buf, size_t size, size_t *off) {
|
168
|
+
const char *test = buf;
|
169
|
+
const char *p = buf;
|
170
|
+
if (mp_check(&test, p + size))
|
171
|
+
return -1;
|
172
|
+
if (mp_typeof(*p) != MP_MAP)
|
173
|
+
return -1;
|
174
|
+
const char *error = NULL, *error_end = NULL,
|
175
|
+
*data = NULL, *data_end = NULL,
|
176
|
+
*metadata = NULL, *metadata_end = NULL,
|
177
|
+
*sqlinfo = NULL, *sqlinfo_end = NULL;
|
178
|
+
uint64_t bitmap = 0;
|
179
|
+
uint32_t n = mp_decode_map(&p);
|
180
|
+
while (n-- > 0) {
|
181
|
+
uint32_t key = mp_decode_uint(&p);
|
182
|
+
switch (key) {
|
183
|
+
case TNT_ERROR: {
|
184
|
+
if (mp_typeof(*p) != MP_STR)
|
185
|
+
return -1;
|
186
|
+
uint32_t elen = 0;
|
187
|
+
error = mp_decode_str(&p, &elen);
|
188
|
+
error_end = error + elen;
|
189
|
+
break;
|
190
|
+
}
|
191
|
+
case TNT_DATA: {
|
192
|
+
if (mp_typeof(*p) != MP_ARRAY)
|
193
|
+
return -1;
|
194
|
+
data = p;
|
195
|
+
mp_next(&p);
|
196
|
+
data_end = p;
|
197
|
+
break;
|
198
|
+
}
|
199
|
+
case TNT_METADATA: {
|
200
|
+
if (mp_typeof(*p) != MP_ARRAY)
|
201
|
+
return -1;
|
202
|
+
metadata = p;
|
203
|
+
mp_next(&p);
|
204
|
+
metadata_end = p;
|
205
|
+
break;
|
206
|
+
}
|
207
|
+
case TNT_SQL_INFO: {
|
208
|
+
if (mp_typeof(*p) != MP_MAP)
|
209
|
+
return -1;
|
210
|
+
sqlinfo = p;
|
211
|
+
mp_next(&p);
|
212
|
+
sqlinfo_end = p;
|
213
|
+
break;
|
214
|
+
}
|
215
|
+
default: {
|
216
|
+
mp_next(&p);
|
217
|
+
break;
|
218
|
+
}
|
219
|
+
}
|
220
|
+
bitmap |= (1ULL << key);
|
221
|
+
}
|
222
|
+
if (r) {
|
223
|
+
r->error = error;
|
224
|
+
r->error_end = error_end;
|
225
|
+
r->data = data;
|
226
|
+
r->data_end = data_end;
|
227
|
+
r->metadata = metadata;
|
228
|
+
r->metadata_end = metadata_end;
|
229
|
+
r->sqlinfo = sqlinfo;
|
230
|
+
r->sqlinfo_end = sqlinfo_end;
|
231
|
+
r->bitmap |= bitmap;
|
232
|
+
}
|
233
|
+
if (off)
|
234
|
+
*off = p - buf;
|
235
|
+
return 0;
|
236
|
+
}
|
237
|
+
|
238
|
+
int
|
239
|
+
tnt_reply(struct tnt_reply *r, char *buf, size_t size, size_t *off) {
|
240
|
+
/* supplied buffer must contain full reply,
|
241
|
+
* if it doesn't then returning count of bytes
|
242
|
+
* needed to process */
|
243
|
+
size_t length;
|
244
|
+
switch (tnt_reply_len(buf, size, &length)) {
|
245
|
+
case 0:
|
246
|
+
break;
|
247
|
+
case 1:
|
248
|
+
if (off)
|
249
|
+
*off = length;
|
250
|
+
return 1;
|
251
|
+
default:
|
252
|
+
return -1;
|
253
|
+
}
|
254
|
+
if (r == NULL) {
|
255
|
+
if (off)
|
256
|
+
*off = length;
|
257
|
+
return 0;
|
258
|
+
}
|
259
|
+
size_t offv = 0;
|
260
|
+
void *ptr[2] = { buf, &offv };
|
261
|
+
int rc = tnt_reply_from(r, (tnt_reply_t)tnt_reply_cb, ptr);
|
262
|
+
if (off)
|
263
|
+
*off = offv;
|
264
|
+
return rc;
|
265
|
+
}
|
266
|
+
|
267
|
+
int
|
268
|
+
tnt_reply0(struct tnt_reply *r, const char *buf, size_t size, size_t *off) {
|
269
|
+
/* supplied buffer must contain full reply,
|
270
|
+
* if it doesn't then returning count of bytes
|
271
|
+
* needed to process */
|
272
|
+
size_t length;
|
273
|
+
switch (tnt_reply_len(buf, size, &length)) {
|
274
|
+
case 0:
|
275
|
+
break;
|
276
|
+
case 1:
|
277
|
+
if (off)
|
278
|
+
*off = length;
|
279
|
+
return 1;
|
280
|
+
default:
|
281
|
+
return -1;
|
282
|
+
}
|
283
|
+
if (r == NULL) {
|
284
|
+
if (off)
|
285
|
+
*off = length;
|
286
|
+
return 0;
|
287
|
+
}
|
288
|
+
const char *data = buf + TNT_REPLY_IPROTO_HDR_SIZE;
|
289
|
+
size_t data_length = length - TNT_REPLY_IPROTO_HDR_SIZE;
|
290
|
+
size_t hdr_length;
|
291
|
+
if (tnt_reply_hdr0(r, data, data_length, &hdr_length) != 0)
|
292
|
+
return -1;
|
293
|
+
if (data_length != hdr_length) {
|
294
|
+
if (tnt_reply_body0(r, data + hdr_length, data_length - hdr_length, NULL) != 0)
|
295
|
+
return -1;
|
296
|
+
}
|
297
|
+
if (off)
|
298
|
+
*off = length;
|
299
|
+
return 0;
|
300
|
+
}
|
@@ -0,0 +1,336 @@
|
|
1
|
+
#include <assert.h>
|
2
|
+
#include <stddef.h>
|
3
|
+
#include <stdint.h>
|
4
|
+
#include <limits.h>
|
5
|
+
#include <string.h>
|
6
|
+
#include <stdbool.h>
|
7
|
+
#include <sys/types.h>
|
8
|
+
|
9
|
+
#include <msgpuck.h>
|
10
|
+
|
11
|
+
#include <tarantool/tnt_mem.h>
|
12
|
+
#include <tarantool/tnt_reply.h>
|
13
|
+
#include <tarantool/tnt_stream.h>
|
14
|
+
#include <tarantool/tnt_net.h>
|
15
|
+
#include <tarantool/tnt_object.h>
|
16
|
+
#include <tarantool/tnt_buf.h>
|
17
|
+
#include <tarantool/tnt_proto.h>
|
18
|
+
#include <tarantool/tnt_schema.h>
|
19
|
+
|
20
|
+
#include <tarantool/tnt_request.h>
|
21
|
+
|
22
|
+
#include "tnt_proto_internal.h"
|
23
|
+
|
24
|
+
struct tnt_request *tnt_request_init(struct tnt_request *req) {
|
25
|
+
int alloc = (req == NULL);
|
26
|
+
if (req == NULL) {
|
27
|
+
req = tnt_mem_alloc(sizeof(struct tnt_request));
|
28
|
+
if (!req) return NULL;
|
29
|
+
}
|
30
|
+
memset(req, 0, sizeof(struct tnt_request));
|
31
|
+
req->limit = UINT32_MAX;
|
32
|
+
req->alloc = alloc;
|
33
|
+
return req;
|
34
|
+
};
|
35
|
+
|
36
|
+
void tnt_request_free(struct tnt_request *req) {
|
37
|
+
if (req->key_object)
|
38
|
+
tnt_stream_free(req->key_object);
|
39
|
+
req->key_object = NULL;
|
40
|
+
if (req->tuple_object)
|
41
|
+
tnt_stream_free(req->tuple_object);
|
42
|
+
req->tuple_object = NULL;
|
43
|
+
if (req->alloc) tnt_mem_free(req);
|
44
|
+
}
|
45
|
+
|
46
|
+
#define TNT_REQUEST_CUSTOM(NM, CNM) \
|
47
|
+
struct tnt_request *tnt_request_##NM(struct tnt_request *req) { \
|
48
|
+
req = tnt_request_init(req); \
|
49
|
+
if (req) { \
|
50
|
+
req->hdr.type = TNT_OP_##CNM; \
|
51
|
+
} \
|
52
|
+
return req; \
|
53
|
+
}
|
54
|
+
|
55
|
+
TNT_REQUEST_CUSTOM(select, SELECT);
|
56
|
+
TNT_REQUEST_CUSTOM(insert, INSERT);
|
57
|
+
TNT_REQUEST_CUSTOM(replace, REPLACE);
|
58
|
+
TNT_REQUEST_CUSTOM(update, UPDATE);
|
59
|
+
TNT_REQUEST_CUSTOM(delete, DELETE);
|
60
|
+
TNT_REQUEST_CUSTOM(call, CALL);
|
61
|
+
TNT_REQUEST_CUSTOM(call_16, CALL_16);
|
62
|
+
TNT_REQUEST_CUSTOM(auth, AUTH);
|
63
|
+
TNT_REQUEST_CUSTOM(eval, EVAL);
|
64
|
+
TNT_REQUEST_CUSTOM(upsert, UPSERT);
|
65
|
+
TNT_REQUEST_CUSTOM(ping, PING);
|
66
|
+
|
67
|
+
#undef TNT_REQUEST_CUSTOM
|
68
|
+
|
69
|
+
int tnt_request_set_space(struct tnt_request *req, uint32_t space)
|
70
|
+
{
|
71
|
+
req->space_id = space;
|
72
|
+
return 0;
|
73
|
+
}
|
74
|
+
|
75
|
+
int tnt_request_set_index(struct tnt_request *req, uint32_t index)
|
76
|
+
{
|
77
|
+
req->index_id = index;
|
78
|
+
return 0;
|
79
|
+
}
|
80
|
+
|
81
|
+
int tnt_request_set_offset(struct tnt_request *req, uint32_t offset)
|
82
|
+
{
|
83
|
+
req->offset = offset;
|
84
|
+
return 0;
|
85
|
+
}
|
86
|
+
|
87
|
+
int tnt_request_set_limit(struct tnt_request *req, uint32_t limit)
|
88
|
+
{
|
89
|
+
req->limit = limit;
|
90
|
+
return 0;
|
91
|
+
}
|
92
|
+
|
93
|
+
int
|
94
|
+
tnt_request_set_iterator(struct tnt_request *req, enum tnt_iterator_t iterator)
|
95
|
+
{
|
96
|
+
req->iterator = iterator;
|
97
|
+
return 0;
|
98
|
+
}
|
99
|
+
|
100
|
+
int tnt_request_set_index_base(struct tnt_request *req, uint32_t index_base)
|
101
|
+
{
|
102
|
+
req->index_base = index_base;
|
103
|
+
return 0;
|
104
|
+
}
|
105
|
+
|
106
|
+
int tnt_request_set_key(struct tnt_request *req, struct tnt_stream *s)
|
107
|
+
{
|
108
|
+
req->key = TNT_SBUF_DATA(s);
|
109
|
+
req->key_end = req->key + TNT_SBUF_SIZE(s);
|
110
|
+
return 0;
|
111
|
+
}
|
112
|
+
|
113
|
+
int tnt_request_set_key_format(struct tnt_request *req, const char *fmt, ...)
|
114
|
+
{
|
115
|
+
if (req->key_object)
|
116
|
+
tnt_object_reset(req->key_object);
|
117
|
+
else
|
118
|
+
req->key_object = tnt_object(NULL);
|
119
|
+
if (!req->key_object)
|
120
|
+
return -1;
|
121
|
+
va_list args;
|
122
|
+
va_start(args, fmt);
|
123
|
+
ssize_t res = tnt_object_vformat(req->key_object, fmt, args);
|
124
|
+
va_end(args);
|
125
|
+
if (res == -1)
|
126
|
+
return -1;
|
127
|
+
return tnt_request_set_key(req, req->key_object);
|
128
|
+
}
|
129
|
+
|
130
|
+
int
|
131
|
+
tnt_request_set_func(struct tnt_request *req, const char *func,
|
132
|
+
uint32_t flen)
|
133
|
+
{
|
134
|
+
if (!is_call(req->hdr.type))
|
135
|
+
return -1;
|
136
|
+
if (!func)
|
137
|
+
return -1;
|
138
|
+
req->key = func; req->key_end = req->key + flen;
|
139
|
+
return 0;
|
140
|
+
}
|
141
|
+
|
142
|
+
int
|
143
|
+
tnt_request_set_funcz(struct tnt_request *req, const char *func)
|
144
|
+
{
|
145
|
+
if (!is_call(req->hdr.type))
|
146
|
+
return -1;
|
147
|
+
if (!func)
|
148
|
+
return -1;
|
149
|
+
req->key = func; req->key_end = req->key + strlen(req->key);
|
150
|
+
return 0;
|
151
|
+
}
|
152
|
+
|
153
|
+
int
|
154
|
+
tnt_request_set_expr(struct tnt_request *req, const char *expr,
|
155
|
+
uint32_t elen)
|
156
|
+
{
|
157
|
+
if (req->hdr.type != TNT_OP_EVAL)
|
158
|
+
return -1;
|
159
|
+
if (!expr)
|
160
|
+
return -1;
|
161
|
+
req->key = expr; req->key_end = req->key + elen;
|
162
|
+
return 0;
|
163
|
+
}
|
164
|
+
|
165
|
+
int
|
166
|
+
tnt_request_set_exprz(struct tnt_request *req, const char *expr)
|
167
|
+
{
|
168
|
+
if (req->hdr.type != TNT_OP_EVAL)
|
169
|
+
return -1;
|
170
|
+
if (!expr)
|
171
|
+
return -1;
|
172
|
+
req->key = expr; req->key_end = req->key + strlen(req->key);
|
173
|
+
return 0;
|
174
|
+
}
|
175
|
+
|
176
|
+
int
|
177
|
+
tnt_request_set_ops(struct tnt_request *req, struct tnt_stream *s)
|
178
|
+
{
|
179
|
+
if (req->hdr.type == TNT_OP_UPDATE) {
|
180
|
+
req->tuple = TNT_SBUF_DATA(s);
|
181
|
+
req->tuple_end = req->tuple + TNT_SBUF_SIZE(s);
|
182
|
+
return 0;
|
183
|
+
} else if (req->hdr.type == TNT_OP_UPSERT) {
|
184
|
+
req->key = TNT_SBUF_DATA(s);
|
185
|
+
req->key_end = req->key + TNT_SBUF_SIZE(s);
|
186
|
+
return 0;
|
187
|
+
}
|
188
|
+
return -1;
|
189
|
+
}
|
190
|
+
|
191
|
+
int tnt_request_set_tuple(struct tnt_request *req, struct tnt_stream *s)
|
192
|
+
{
|
193
|
+
req->tuple = TNT_SBUF_DATA(s);
|
194
|
+
req->tuple_end = req->tuple + TNT_SBUF_SIZE(s);
|
195
|
+
return 0;
|
196
|
+
}
|
197
|
+
|
198
|
+
int tnt_request_set_tuple_format(struct tnt_request *req, const char *fmt, ...)
|
199
|
+
{
|
200
|
+
if (req->tuple_object)
|
201
|
+
tnt_object_reset(req->tuple_object);
|
202
|
+
else
|
203
|
+
req->tuple_object = tnt_object(NULL);
|
204
|
+
if (!req->tuple_object)
|
205
|
+
return -1;
|
206
|
+
va_list args;
|
207
|
+
va_start(args, fmt);
|
208
|
+
ssize_t res = tnt_object_vformat(req->tuple_object, fmt, args);
|
209
|
+
va_end(args);
|
210
|
+
if (res == -1)
|
211
|
+
return -1;
|
212
|
+
return tnt_request_set_tuple(req, req->tuple_object);
|
213
|
+
}
|
214
|
+
|
215
|
+
int
|
216
|
+
tnt_request_writeout(struct tnt_stream *s, struct tnt_request *req,
|
217
|
+
uint64_t *sync) {
|
218
|
+
enum tnt_request_t tp = req->hdr.type;
|
219
|
+
if (sync != NULL && *sync == INT64_MAX &&
|
220
|
+
(s->reqid & INT64_MAX) == INT64_MAX) {
|
221
|
+
s->reqid = 0;
|
222
|
+
}
|
223
|
+
req->hdr.sync = s->reqid++;
|
224
|
+
/* header */
|
225
|
+
/* int (9) + 1 + sync + 1 + op */
|
226
|
+
struct iovec v[10]; int v_sz = 0;
|
227
|
+
char header[128];
|
228
|
+
char *pos = header + 9;
|
229
|
+
char *begin = pos;
|
230
|
+
v[v_sz].iov_base = begin;
|
231
|
+
v[v_sz++].iov_len = 0;
|
232
|
+
pos = mp_encode_map(pos, 2); /* 1 */
|
233
|
+
pos = mp_encode_uint(pos, TNT_CODE); /* 1 */
|
234
|
+
pos = mp_encode_uint(pos, req->hdr.type); /* 1 */
|
235
|
+
pos = mp_encode_uint(pos, TNT_SYNC); /* 1 */
|
236
|
+
pos = mp_encode_uint(pos, req->hdr.sync); /* 9 */
|
237
|
+
char *map = pos++; /* 1 */
|
238
|
+
size_t nd = 0;
|
239
|
+
if (tp < TNT_OP_CALL_16 || tp == TNT_OP_UPSERT) {
|
240
|
+
pos = mp_encode_uint(pos, TNT_SPACE); /* 1 */
|
241
|
+
pos = mp_encode_uint(pos, req->space_id); /* 5 */
|
242
|
+
nd += 1;
|
243
|
+
}
|
244
|
+
if (req->index_id && (tp == TNT_OP_SELECT ||
|
245
|
+
tp == TNT_OP_UPDATE ||
|
246
|
+
tp == TNT_OP_DELETE)) {
|
247
|
+
pos = mp_encode_uint(pos, TNT_INDEX); /* 1 */
|
248
|
+
pos = mp_encode_uint(pos, req->index_id); /* 5 */
|
249
|
+
nd += 1;
|
250
|
+
}
|
251
|
+
if (tp == TNT_OP_SELECT) {
|
252
|
+
pos = mp_encode_uint(pos, TNT_LIMIT); /* 1 */
|
253
|
+
pos = mp_encode_uint(pos, req->limit); /* 5 */
|
254
|
+
nd += 1;
|
255
|
+
}
|
256
|
+
if (req->offset && tp == TNT_OP_SELECT) {
|
257
|
+
pos = mp_encode_uint(pos, TNT_OFFSET); /* 1 */
|
258
|
+
pos = mp_encode_uint(pos, req->offset); /* 5 */
|
259
|
+
nd += 1;
|
260
|
+
}
|
261
|
+
if (req->iterator && tp == TNT_OP_SELECT) {
|
262
|
+
pos = mp_encode_uint(pos, TNT_ITERATOR); /* 1 */
|
263
|
+
pos = mp_encode_uint(pos, req->iterator); /* 1 */
|
264
|
+
nd += 1;
|
265
|
+
}
|
266
|
+
if (req->key) {
|
267
|
+
switch (tp) {
|
268
|
+
case TNT_OP_EVAL:
|
269
|
+
pos = mp_encode_uint(pos, TNT_EXPRESSION); /* 1 */
|
270
|
+
pos = mp_encode_strl(pos, req->key_end - req->key); /* 5 */
|
271
|
+
break;
|
272
|
+
case TNT_OP_CALL_16:
|
273
|
+
case TNT_OP_CALL:
|
274
|
+
pos = mp_encode_uint(pos, TNT_FUNCTION); /* 1 */
|
275
|
+
pos = mp_encode_strl(pos, req->key_end - req->key); /* 5 */
|
276
|
+
break;
|
277
|
+
case TNT_OP_SELECT:
|
278
|
+
case TNT_OP_UPDATE:
|
279
|
+
case TNT_OP_DELETE:
|
280
|
+
pos = mp_encode_uint(pos, TNT_KEY); /* 1 */
|
281
|
+
break;
|
282
|
+
case TNT_OP_UPSERT:
|
283
|
+
pos = mp_encode_uint(pos, TNT_OPS); /* 1 */
|
284
|
+
break;
|
285
|
+
default:
|
286
|
+
return -1;
|
287
|
+
}
|
288
|
+
v[v_sz].iov_base = begin;
|
289
|
+
v[v_sz++].iov_len = pos - begin;
|
290
|
+
begin = pos;
|
291
|
+
v[v_sz].iov_base = (void *)req->key;
|
292
|
+
v[v_sz++].iov_len = req->key_end - req->key;
|
293
|
+
nd += 1;
|
294
|
+
}
|
295
|
+
if (req->tuple) {
|
296
|
+
pos = mp_encode_uint(pos, TNT_TUPLE); /* 1 */
|
297
|
+
v[v_sz].iov_base = begin;
|
298
|
+
v[v_sz++].iov_len = pos - begin;
|
299
|
+
begin = pos;
|
300
|
+
v[v_sz].iov_base = (void *)req->tuple;
|
301
|
+
v[v_sz++].iov_len = req->tuple_end - req->tuple;
|
302
|
+
nd += 1;
|
303
|
+
}
|
304
|
+
if (req->index_base && (tp == TNT_OP_UPDATE || tp == TNT_OP_UPSERT)) {
|
305
|
+
pos = mp_encode_uint(pos, TNT_INDEX_BASE); /* 1 */
|
306
|
+
pos = mp_encode_uint(pos, req->index_base); /* 1 */
|
307
|
+
nd += 1;
|
308
|
+
}
|
309
|
+
assert(mp_sizeof_map(nd) == 1);
|
310
|
+
if (pos != begin) {
|
311
|
+
v[v_sz].iov_base = begin;
|
312
|
+
v[v_sz++].iov_len = pos - begin;
|
313
|
+
}
|
314
|
+
mp_encode_map(map, nd);
|
315
|
+
|
316
|
+
size_t plen = 0;
|
317
|
+
for (int i = 1; i < v_sz; ++i) plen += v[i].iov_len;
|
318
|
+
size_t hlen = mp_sizeof_luint32(plen);
|
319
|
+
v[0].iov_base -= hlen;
|
320
|
+
v[0].iov_len += hlen;
|
321
|
+
mp_encode_luint32(v[0].iov_base, plen);
|
322
|
+
ssize_t rv = s->writev(s, v, v_sz);
|
323
|
+
if (rv == -1)
|
324
|
+
return -1;
|
325
|
+
if (sync != NULL)
|
326
|
+
*sync = req->hdr.sync;
|
327
|
+
return 0;
|
328
|
+
}
|
329
|
+
|
330
|
+
int64_t
|
331
|
+
tnt_request_compile(struct tnt_stream *s, struct tnt_request *req) {
|
332
|
+
uint64_t sync = INT64_MAX;
|
333
|
+
if (tnt_request_writeout(s, req, &sync) == -1)
|
334
|
+
return -1;
|
335
|
+
return sync;
|
336
|
+
}
|