Line data Source code
1 : /*
2 : * SPDX-License-Identifier: MPL-2.0
3 : *
4 : * This Source Code Form is subject to the terms of the Mozilla Public
5 : * License, v. 2.0. If a copy of the MPL was not distributed with this
6 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
7 : *
8 : * Copyright 2024 MonetDB Foundation;
9 : * Copyright August 2008 - 2023 MonetDB B.V.;
10 : * Copyright 1997 - July 2008 CWI.
11 : */
12 :
13 : #include "monetdb_config.h"
14 :
15 : #include "monetdbe.h"
16 : #include "gdk.h"
17 : #include "mal.h"
18 : #include "mal_client.h"
19 : #include "mal_embedded.h"
20 : #include "mal_backend.h"
21 : #include "mal_builder.h"
22 : #include "mal_linker.h"
23 : #include "sql_mvc.h"
24 : #include "sql_catalog.h"
25 : #include "sql_gencode.h"
26 : #include "sql_semantic.h"
27 : #include "sql_scenario.h"
28 : #include "sql_optimizer.h"
29 : #include "rel_exp.h"
30 : #include "rel_rel.h"
31 : #include "rel_updates.h"
32 : #include "monet_options.h"
33 : #include "mapi.h"
34 : #include "monetdbe_mapi.h"
35 : #include "remote.h"
36 : #include "sql.h"
37 : #include "sql_result.h"
38 : #include "mutils.h"
39 :
40 : #define UNUSED(x) (void)(x)
41 :
42 : static int
43 0 : monetdbe_2_gdk_type(monetdbe_types t) {
44 0 : switch(t) {
45 : case monetdbe_bool: return TYPE_bit;
46 : case monetdbe_int8_t: return TYPE_bte;
47 : case monetdbe_int16_t: return TYPE_sht;
48 : case monetdbe_int32_t: return TYPE_int;
49 : case monetdbe_int64_t: return TYPE_lng;
50 : #ifdef HAVE_HGE
51 : case monetdbe_int128_t: return TYPE_hge;
52 : #endif
53 : case monetdbe_size_t: return TYPE_oid;
54 : case monetdbe_float: return TYPE_flt;
55 : case monetdbe_double: return TYPE_dbl;
56 : case monetdbe_str: return TYPE_str;
57 : case monetdbe_blob: return TYPE_blob;
58 : case monetdbe_date: return TYPE_date;
59 : case monetdbe_time: return TYPE_daytime;
60 : case monetdbe_timestamp: return TYPE_timestamp;
61 : default:
62 : return -1;
63 : }
64 : }
65 :
66 : static monetdbe_types
67 0 : embedded_type(int t) {
68 0 : switch(t) {
69 : case TYPE_bit: return monetdbe_bool;
70 0 : case TYPE_bte: return monetdbe_int8_t;
71 0 : case TYPE_sht: return monetdbe_int16_t;
72 0 : case TYPE_int: return monetdbe_int32_t;
73 0 : case TYPE_lng: return monetdbe_int64_t;
74 : #ifdef HAVE_HGE
75 0 : case TYPE_hge: return monetdbe_int128_t;
76 : #endif
77 0 : case TYPE_oid: return monetdbe_size_t;
78 0 : case TYPE_flt: return monetdbe_float;
79 0 : case TYPE_dbl: return monetdbe_double;
80 0 : case TYPE_str: return monetdbe_str;
81 0 : case TYPE_date: return monetdbe_date;
82 0 : case TYPE_daytime: return monetdbe_time;
83 0 : case TYPE_timestamp: return monetdbe_timestamp;
84 0 : default:
85 0 : if (t==TYPE_blob)
86 0 : return monetdbe_blob;
87 : return monetdbe_type_unknown;
88 : }
89 : }
90 :
91 : typedef struct {
92 : Client c;
93 : char *msg;
94 : int registered_thread; /* 1 = registered in monetdbe_open, 2 = done by GDK (also deregister done there) */
95 : monetdbe_data_blob blob_null;
96 : monetdbe_data_date date_null;
97 : monetdbe_data_time time_null;
98 : monetdbe_data_timestamp timestamp_null;
99 : str mid;
100 : } monetdbe_database_internal;
101 :
102 : typedef struct {
103 : monetdbe_result res;
104 : int type;
105 : res_table *monetdbe_resultset;
106 : monetdbe_column **converted_columns;
107 : monetdbe_database_internal *mdbe;
108 : } monetdbe_result_internal;
109 :
110 : typedef struct {
111 : monetdbe_statement res;
112 : ValRecord *data;
113 : ValPtr *args; /* only used during calls */
114 : int retc;
115 : monetdbe_database_internal *mdbe;
116 : cq *q;
117 : } monetdbe_stmt_internal;
118 :
119 : static MT_Lock embedded_lock = MT_LOCK_INITIALIZER(embedded_lock);
120 : static bool monetdbe_embedded_initialized = false;
121 : static char *monetdbe_embedded_url = NULL;
122 : static int open_dbs = 0;
123 :
124 : static void data_from_date(date d, monetdbe_data_date *ptr);
125 : static void data_from_time(daytime d, monetdbe_data_time *ptr);
126 : static void data_from_timestamp(timestamp d, monetdbe_data_timestamp *ptr);
127 : static timestamp timestamp_from_data(monetdbe_data_timestamp *ptr);
128 : static date date_from_data(monetdbe_data_date *ptr);
129 : static daytime time_from_data(monetdbe_data_time *ptr);
130 :
131 : static char* monetdbe_cleanup_result_internal(monetdbe_database_internal *mdbe, monetdbe_result_internal* res);
132 :
133 : static int
134 0 : date_is_null(monetdbe_data_date *value)
135 : {
136 0 : monetdbe_data_date null_value;
137 0 : data_from_date(date_nil, &null_value);
138 0 : return value->year == null_value.year && value->month == null_value.month &&
139 : value->day == null_value.day;
140 : }
141 :
142 : static int
143 0 : time_is_null(monetdbe_data_time *value)
144 : {
145 0 : monetdbe_data_time null_value;
146 0 : data_from_time(daytime_nil, &null_value);
147 0 : return value->hours == null_value.hours &&
148 0 : value->minutes == null_value.minutes &&
149 0 : value->seconds == null_value.seconds && value->ms == null_value.ms;
150 : }
151 :
152 : static int
153 0 : timestamp_is_null(monetdbe_data_timestamp *value)
154 : {
155 0 : return is_timestamp_nil(timestamp_from_data(value));
156 : }
157 :
158 : static int
159 2 : str_is_null(char **value)
160 : {
161 2 : return !value || *value == NULL;
162 : }
163 :
164 : static int
165 0 : blob_is_null(monetdbe_data_blob *value)
166 : {
167 0 : return !value || value->data == NULL;
168 : }
169 :
170 : const char *
171 0 : monetdbe_version(void)
172 : {
173 0 : return MONETDBE_VERSION;
174 : }
175 :
176 : static void
177 6 : clear_error( monetdbe_database_internal *mdbe)
178 : {
179 6 : if (mdbe->msg)
180 0 : freeException(mdbe->msg);
181 6 : mdbe->msg = NULL;
182 0 : }
183 :
184 : static char*
185 0 : set_error( monetdbe_database_internal *mdbe, char *err)
186 : {
187 0 : if (!err)
188 : return err;
189 0 : if (mdbe->msg) /* keep first error */
190 0 : freeException(err);
191 : else
192 0 : mdbe->msg = err;
193 0 : return mdbe->msg;
194 : }
195 :
196 : static char*
197 3 : commit_action(mvc* m, monetdbe_database_internal *mdbe, monetdbe_result **result, monetdbe_result_internal *res_internal)
198 : {
199 3 : char *commit_msg = MAL_SUCCEED;
200 :
201 : /* if an error already exists from MonetDBe set the session status to dirty */
202 3 : if (mdbe->msg != MAL_SUCCEED && m->session->tr->active && !m->session->status)
203 0 : m->session->status = -1;
204 3 : commit_msg = SQLautocommit(m); /* handle autocommit */
205 :
206 3 : if (mdbe->msg != MAL_SUCCEED || commit_msg != MAL_SUCCEED) {
207 0 : if (res_internal) {
208 0 : char* other = monetdbe_cleanup_result_internal(mdbe, res_internal);
209 0 : if (other)
210 0 : freeException(other);
211 : }
212 0 : if (result)
213 0 : *result = NULL;
214 0 : (void)set_error(mdbe, commit_msg);
215 : }
216 3 : return mdbe->msg;
217 : }
218 :
219 : static int
220 1 : validate_database_handle_noerror(monetdbe_database_internal *mdbe)
221 : {
222 1 : if (!monetdbe_embedded_initialized || !MCvalid(mdbe->c))
223 0 : return 0;
224 1 : assert(mdbe->c);
225 1 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
226 1 : clear_error(mdbe);
227 1 : return 1;
228 : }
229 :
230 : // Call this function always inside the embedded_lock
231 : static char*
232 5 : validate_database_handle(monetdbe_database_internal *mdbe, const char* call)
233 : {
234 5 : if (!monetdbe_embedded_initialized)
235 0 : return createException(MAL, call, "MonetDBe has not yet started");
236 5 : if (!MCvalid(mdbe->c))
237 0 : return createException(MAL, call, "Invalid database handle");
238 5 : clear_error(mdbe);
239 5 : return MAL_SUCCEED;
240 : }
241 :
242 : static void
243 2 : monetdbe_destroy_column(monetdbe_column* column)
244 : {
245 2 : size_t j;
246 :
247 2 : if (!column)
248 : return;
249 :
250 2 : if (column->type == monetdbe_str) {
251 : // FIXME: clean up individual strings
252 1 : char** data = (char**)column->data;
253 3 : for(j = 0; j < column->count; j++) {
254 2 : if (data[j])
255 2 : GDKfree(data[j]);
256 : }
257 1 : } else if (column->type == monetdbe_blob) {
258 0 : monetdbe_data_blob* data = (monetdbe_data_blob*)column->data;
259 0 : for(j = 0; j < column->count; j++) {
260 0 : if (data[j].data)
261 0 : GDKfree(data[j].data);
262 : }
263 : }
264 2 : GDKfree(column->sql_type.name);
265 2 : GDKfree(column->data);
266 2 : GDKfree(column);
267 : }
268 :
269 : static char*
270 1 : monetdbe_cleanup_result_internal(monetdbe_database_internal *mdbe, monetdbe_result_internal* result)
271 : {
272 1 : mvc *m = NULL;
273 :
274 1 : assert(!result || !result->mdbe || result->mdbe == mdbe);
275 1 : if ((mdbe->msg = validate_database_handle(mdbe, "monetdbe.monetdbe_cleanup_result_internal")) != MAL_SUCCEED)
276 : return mdbe->msg;
277 1 : if ((mdbe->msg = getSQLContext(mdbe->c, NULL, &m, NULL)) != MAL_SUCCEED)
278 0 : goto cleanup;
279 :
280 1 : if (result->monetdbe_resultset)
281 1 : res_tables_destroy(result->monetdbe_resultset);
282 :
283 1 : if (result->converted_columns) {
284 3 : for (size_t i = 0; i < result->res.ncols; i++)
285 2 : monetdbe_destroy_column(result->converted_columns[i]);
286 1 : GDKfree(result->converted_columns);
287 : }
288 1 : GDKfree(result);
289 1 : cleanup:
290 1 : return commit_action(m, mdbe, NULL, NULL);
291 : }
292 :
293 : static char*
294 1 : monetdbe_get_results(monetdbe_result** result, monetdbe_database_internal *mdbe)
295 : {
296 1 : backend *be = NULL;
297 :
298 1 : *result = NULL;
299 1 : if ((mdbe->msg = getBackendContext(mdbe->c, &be)) != NULL)
300 : return mdbe->msg;
301 :
302 1 : mvc *m = be->mvc;
303 1 : monetdbe_result_internal* res_internal;
304 :
305 1 : if (!(res_internal = GDKzalloc(sizeof(monetdbe_result_internal)))) {
306 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_results", MAL_MALLOC_FAIL));
307 0 : return mdbe->msg;
308 : }
309 : // TODO: set type of result outside.
310 1 : res_internal->res.last_id = be->last_id;
311 1 : res_internal->mdbe = mdbe;
312 1 : *result = (monetdbe_result*) res_internal;
313 1 : m->reply_size = -2; /* do not clean up result tables */
314 :
315 1 : if (be->results) {
316 1 : res_internal->res.ncols = (size_t) be->results->nr_cols;
317 1 : res_internal->monetdbe_resultset = be->results;
318 1 : if (be->results->nr_cols > 0)
319 1 : res_internal->res.nrows = be->results->nr_rows;
320 1 : be->results = NULL;
321 1 : res_internal->converted_columns = GDKzalloc(sizeof(monetdbe_column*) * res_internal->res.ncols);
322 1 : if (!res_internal->converted_columns) {
323 0 : GDKfree(res_internal);
324 0 : *result = NULL;
325 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_results", MAL_MALLOC_FAIL));
326 0 : return mdbe->msg;
327 : }
328 : }
329 :
330 : return MAL_SUCCEED;
331 : }
332 :
333 : static char*
334 0 : monetdbe_query_internal(monetdbe_database_internal *mdbe, char* query, monetdbe_result** result, monetdbe_cnt* affected_rows, int *prepare_id, char language)
335 : {
336 0 : char *nq = NULL;
337 0 : Client c = mdbe->c;
338 0 : mvc* m = NULL;
339 0 : backend *b;
340 0 : size_t query_len, input_query_len, prep_len = 0;
341 0 : buffer query_buf;
342 0 : stream *query_stream = NULL;
343 0 : bstream *old_bstream = c->fdin;
344 0 : stream *fdout = c->fdout;
345 0 : bool fdin_changed = false;
346 :
347 0 : if (result)
348 0 : *result = NULL;
349 :
350 0 : if ((mdbe->msg = validate_database_handle(mdbe, "monetdbe.monetdbe_query_internal")) != MAL_SUCCEED)
351 : return mdbe->msg;
352 :
353 0 : if ((mdbe->msg = getSQLContext(c, NULL, &m, NULL)) != MAL_SUCCEED)
354 0 : goto cleanup;
355 0 : b = (backend *) c->sqlcontext;
356 :
357 0 : if (!query) {
358 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_internal", "Query missing"));
359 0 : goto cleanup;
360 : }
361 0 : if (!(query_stream = buffer_rastream(&query_buf, "sqlstatement"))) {
362 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_internal", "Could not setup query stream"));
363 0 : goto cleanup;
364 : }
365 0 : input_query_len = strlen(query);
366 0 : query_len = input_query_len + 3;
367 0 : if (prepare_id) {
368 0 : prep_len = sizeof("PREPARE ")-1;
369 0 : query_len += prep_len;
370 : }
371 0 : if (!(nq = GDKmalloc(query_len))) {
372 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_internal", MAL_MALLOC_FAIL));
373 0 : goto cleanup;
374 : }
375 0 : if (prepare_id)
376 0 : strcpy(nq, "PREPARE ");
377 0 : strcpy(nq + prep_len, query);
378 0 : strcpy(nq + prep_len + input_query_len, "\n;");
379 :
380 0 : query_buf.pos = 0;
381 0 : query_buf.len = query_len;
382 0 : query_buf.buf = nq;
383 :
384 0 : fdin_changed = true;
385 0 : if (!(c->fdin = bstream_create(query_stream, query_len))) {
386 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_internal", "Could not setup query stream"));
387 0 : goto cleanup;
388 : }
389 0 : c->qryctx.bs = c->fdin;
390 0 : query_stream = NULL;
391 0 : if (bstream_next(c->fdin) < 0) {
392 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_internal", "Internal error while starting the query"));
393 0 : goto cleanup;
394 : }
395 :
396 0 : assert(language);
397 0 : b->language = language;
398 0 : b->output_format = OFMT_NONE;
399 0 : b->no_mitosis = 0;
400 0 : m->user_id = m->role_id = USER_MONETDB;
401 0 : m->errstr[0] = '\0';
402 0 : m->params = NULL;
403 0 : m->sym = NULL;
404 0 : m->runs = NULL;
405 0 : m->label = 0;
406 0 : if (m->sa)
407 0 : m->sa = sa_reset(m->sa);
408 0 : m->scanner.mode = LINE_N;
409 0 : m->scanner.rs = c->fdin;
410 0 : mvc_query_processed(m);
411 :
412 0 : if ((mdbe->msg = MSinitClientPrg(c, userRef, mainRef)) != MAL_SUCCEED)
413 0 : goto cleanup;
414 0 : if (prepare_id)
415 0 : m->emode = m_prepare;
416 0 : c->fdout = NULL;
417 0 : if ((mdbe->msg = SQLengine_(c)) != MAL_SUCCEED)
418 0 : goto cleanup;
419 0 : if (m->emode == m_prepare && prepare_id)
420 0 : *prepare_id = b->result_id;
421 0 : if (!b->results && b->rowcnt >= 0 && affected_rows)
422 0 : *affected_rows = b->rowcnt;
423 :
424 0 : if (result) {
425 0 : if ((mdbe->msg = monetdbe_get_results(result, mdbe)) != MAL_SUCCEED) {
426 0 : goto cleanup;
427 : }
428 :
429 0 : if (m->emode & m_prepare)
430 0 : (*(monetdbe_result_internal**) result)->type = Q_PREPARE;
431 : else
432 0 : (*(monetdbe_result_internal**) result)->type = (b->results) ? b->results->query_type : m->type;
433 : }
434 :
435 0 : cleanup:
436 0 : c->fdout = fdout;
437 0 : if (nq)
438 0 : GDKfree(nq);
439 0 : MSresetInstructions(c->curprg->def, 1);
440 0 : if (fdin_changed) { //c->fdin was set
441 0 : bstream_destroy(c->fdin);
442 0 : c->fdin = old_bstream;
443 0 : c->qryctx.bs = old_bstream;
444 : }
445 0 : if (query_stream)
446 0 : close_stream(query_stream);
447 :
448 0 : return commit_action(m, mdbe, result, result?*(monetdbe_result_internal**) result:NULL);
449 : }
450 :
451 : static int
452 1 : monetdbe_close_remote(monetdbe_database_internal *mdbe)
453 : {
454 1 : assert(mdbe && mdbe->mid);
455 :
456 1 : int err = 0;
457 :
458 1 : if (mdbe->msg) {
459 0 : err = 1;
460 0 : clear_error(mdbe);
461 : }
462 :
463 1 : if ( (mdbe->msg = RMTdisconnect(NULL, &(const char *){mdbe->mid})) != MAL_SUCCEED) {
464 0 : err = 1;
465 0 : clear_error(mdbe);
466 : }
467 :
468 1 : GDKfree(mdbe->mid);
469 1 : mdbe->mid = NULL;
470 :
471 1 : return err;
472 : }
473 :
474 : static int
475 1 : monetdbe_close_internal(monetdbe_database_internal *mdbe)
476 : {
477 1 : assert(mdbe);
478 :
479 1 : if (validate_database_handle_noerror(mdbe)) {
480 1 : open_dbs--;
481 1 : char *msg = SQLexitClient(mdbe->c);
482 1 : if (msg)
483 0 : freeException(msg);
484 1 : MCcloseClient(mdbe->c);
485 : }
486 1 : GDKfree(mdbe);
487 1 : return 0;
488 : }
489 :
490 : static int
491 2 : monetdbe_workers_internal(monetdbe_database_internal *mdbe, monetdbe_options *opts)
492 : {
493 2 : int workers = 0;
494 2 : if (opts && opts->nr_threads) {
495 0 : if (opts->nr_threads < 0)
496 0 : set_error(mdbe,createException(MAL, "monetdbe.monetdbe_startup", "Nr_threads should be positive"));
497 : else
498 : workers = opts->nr_threads;
499 : }
500 2 : return workers;
501 : }
502 :
503 : static int
504 2 : monetdbe_memory_internal(monetdbe_database_internal *mdbe, monetdbe_options *opts)
505 : {
506 2 : int memory = 0;
507 2 : if (opts && opts->memorylimit) {
508 0 : if (opts->memorylimit < 0)
509 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "Memorylimit should be positive"));
510 : else /* Memory limit is session specific */
511 : memory = opts->memorylimit;
512 : }
513 2 : return memory;
514 : }
515 :
516 : static int
517 1 : monetdbe_querytimeout_internal(monetdbe_database_internal *mdbe, monetdbe_options *opts)
518 : {
519 1 : int querytimeout = 0;
520 1 : if (opts && opts->querytimeout) {
521 0 : if (opts->querytimeout < 0)
522 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "Query timeout should be positive (in sec)"));
523 : else
524 : querytimeout = opts->querytimeout;
525 : }
526 1 : return querytimeout;
527 : }
528 :
529 : static int
530 1 : monetdbe_sessiontimeout_internal(monetdbe_database_internal *mdbe, monetdbe_options *opts)
531 : {
532 1 : int sessiontimeout = 0;
533 1 : if (opts && opts->sessiontimeout) {
534 0 : if (opts->sessiontimeout < 0)
535 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "Session timeout should be positive (in sec)"));
536 : else
537 : sessiontimeout = opts->sessiontimeout;
538 : }
539 1 : return sessiontimeout;
540 : }
541 :
542 : static int
543 1 : monetdbe_open_internal(monetdbe_database_internal *mdbe, monetdbe_options *opts )
544 : {
545 1 : mvc *m;
546 :
547 1 : if (!mdbe)
548 : return -1;
549 1 : if (!monetdbe_embedded_initialized) {
550 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_internal", "Embedded MonetDB is not started"));
551 0 : goto cleanup;
552 : }
553 1 : if (!mdbe->registered_thread) {
554 0 : if (!MT_thread_register()) {
555 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_internal", "Embedded MonetDB is not started"));
556 0 : goto cleanup;
557 : }
558 0 : mdbe->registered_thread = 1;
559 : }
560 1 : mdbe->c = MCinitClient((oid) 0, 0, 0);
561 1 : if (!MCvalid(mdbe->c)) {
562 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_internal", "Failed to initialize client"));
563 0 : goto cleanup;
564 : }
565 1 : mdbe->c->curmodule = mdbe->c->usermodule = userModule();
566 1 : mdbe->c->workerlimit = monetdbe_workers_internal(mdbe, opts);
567 1 : mdbe->c->memorylimit = monetdbe_memory_internal(mdbe, opts);
568 1 : mdbe->c->querytimeout = monetdbe_querytimeout_internal(mdbe, opts);
569 1 : mdbe->c->sessiontimeout = monetdbe_sessiontimeout_internal(mdbe, opts);
570 1 : if (mdbe->msg)
571 0 : goto cleanup;
572 1 : if (mdbe->c->usermodule == NULL) {
573 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_internal", "Failed to initialize client MAL module"));
574 0 : goto cleanup;
575 : }
576 1 : if ((mdbe->msg = SQLinitClient(mdbe->c, NULL, NULL, NULL)) != MAL_SUCCEED ||
577 1 : (mdbe->msg = getSQLContext(mdbe->c, NULL, &m, NULL)) != MAL_SUCCEED)
578 0 : goto cleanup;
579 1 : m->session->auto_commit = 1;
580 1 : if (!m->pa)
581 0 : m->pa = sa_create(NULL);
582 1 : if (!m->sa)
583 1 : m->sa = sa_create(m->pa);
584 1 : if (!m->ta)
585 0 : m->ta = sa_create(m->pa);
586 1 : if (!m->pa || !m->sa || !m->ta) {
587 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_open_internal", MAL_MALLOC_FAIL));
588 0 : goto cleanup;
589 : }
590 1 : m->no_int128 = opts?opts->no_int128:false;
591 1 : cleanup:
592 1 : if (mdbe->msg)
593 : return -2;
594 1 : mdbe->blob_null.data = NULL;
595 1 : data_from_date(date_nil, &mdbe->date_null);
596 1 : data_from_time(daytime_nil, &mdbe->time_null);
597 1 : data_from_timestamp(timestamp_nil, &mdbe->timestamp_null);
598 1 : open_dbs++;
599 1 : return 0;
600 : }
601 :
602 : static void
603 1 : monetdbe_shutdown_internal(void) // Call this function always inside the embedded_lock
604 : {
605 1 : if (monetdbe_embedded_initialized && (open_dbs == 0)) {
606 1 : malEmbeddedReset();
607 1 : monetdbe_embedded_initialized = false;
608 1 : if (monetdbe_embedded_url)
609 0 : GDKfree(monetdbe_embedded_url);
610 1 : monetdbe_embedded_url = NULL;
611 : }
612 1 : }
613 :
614 : static void
615 1 : monetdbe_startup(monetdbe_database_internal *mdbe, const char* dbdir, monetdbe_options *opts)
616 : {
617 : // Only call monetdbe_startup when there is no monetdb internal yet initialized.
618 1 : assert(!monetdbe_embedded_initialized);
619 :
620 1 : opt *set = NULL;
621 1 : int setlen;
622 1 : bool with_mapi_server;
623 1 : int workers, memory;
624 1 : gdk_return gdk_res;
625 :
626 1 : GDKfataljumpenable = 1;
627 :
628 1 : if(setjmp(GDKfataljump) != 0) {
629 0 : assert(0);
630 : mdbe->msg = GDKfatalmsg;
631 : // we will get here if GDKfatal was called.
632 : if (mdbe->msg == NULL)
633 : mdbe->msg = createException(MAL, "monetdbe.monetdbe_startup", "GDKfatal() with unspecified error");
634 : goto cleanup;
635 : }
636 :
637 1 : with_mapi_server = false;
638 :
639 1 : if (monetdbe_embedded_initialized) {
640 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "MonetDBe is already initialized"));
641 0 : GDKfataljumpenable = 0;
642 0 : return;
643 : }
644 :
645 1 : if ((setlen = mo_builtin_settings(&set)) == 0) {
646 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
647 0 : goto cleanup;
648 : }
649 1 : if (dbdir && (setlen = mo_add_option(&set, setlen, opt_cmdline, "gdk_dbpath", dbdir)) == 0) {
650 0 : mo_free_options(set, setlen);
651 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
652 0 : goto cleanup;
653 : }
654 1 : if (opts && opts->nr_threads == 1)
655 0 : setlen = mo_add_option(&set, setlen, opt_cmdline, "sql_optimizer", "sequential_pipe");
656 : else
657 1 : setlen = mo_add_option(&set, setlen, opt_cmdline, "sql_optimizer", "default_pipe");
658 :
659 1 : if (setlen == 0) {
660 0 : mo_free_options(set, setlen);
661 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
662 0 : goto cleanup;
663 : }
664 :
665 1 : if (opts && opts->mapi_server) {
666 : /*This monetdbe instance wants to listen to external mapi client connections.*/
667 0 : if (opts->mapi_server->host) {
668 0 : with_mapi_server = true;
669 0 : int psetlen = setlen;
670 0 : setlen = mo_add_option(&set, setlen, opt_cmdline, "mapi_listenaddr", opts->mapi_server->host);
671 0 : if (setlen == psetlen) {
672 0 : mo_free_options(set, setlen);
673 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
674 0 : goto cleanup;
675 : }
676 : }
677 0 : if (opts->mapi_server->port) {
678 0 : with_mapi_server = true;
679 0 : int psetlen = setlen;
680 0 : setlen = mo_add_option(&set, setlen, opt_cmdline, "mapi_port", opts->mapi_server->port);
681 0 : if (setlen == psetlen) {
682 0 : mo_free_options(set, setlen);
683 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
684 0 : goto cleanup;
685 : }
686 : }
687 0 : if (opts->mapi_server->usock) {
688 0 : with_mapi_server = true;
689 0 : int psetlen = setlen;
690 0 : setlen = mo_add_option(&set, setlen, opt_cmdline, "mapi_usock", opts->mapi_server->usock);
691 0 : if (setlen == psetlen) {
692 0 : mo_free_options(set, setlen);
693 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
694 0 : goto cleanup;
695 : }
696 : }
697 : }
698 :
699 : /* set the output of GDKtracer logs */
700 1 : if (opts && opts->trace_file) {
701 : /* if file specified, use it */
702 0 : if (GDKtracer_set_tracefile(opts->trace_file) != GDK_SUCCEED) {
703 0 : mo_free_options(set, setlen);
704 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", GDK_EXCEPTION));
705 0 : goto cleanup;
706 : }
707 0 : GDKtracer_set_adapter("BASIC");
708 : } else {
709 : /* otherwise no trace output */
710 1 : GDKtracer_set_adapter("MBEDDED");
711 : }
712 :
713 1 : if ((workers = monetdbe_workers_internal(mdbe, opts))) {
714 0 : int psetlen = setlen;
715 0 : char workstr[16];
716 :
717 0 : snprintf(workstr, sizeof(workstr), "%d", workers);
718 0 : if ((setlen = mo_add_option(&set, setlen, opt_cmdline, "gdk_nr_threads", workstr)) == psetlen) {
719 0 : mo_free_options(set, setlen);
720 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
721 0 : goto cleanup;
722 : }
723 : }
724 1 : if ((memory = monetdbe_memory_internal(mdbe, opts))) {
725 0 : int psetlen = setlen;
726 0 : char memstr[32];
727 :
728 0 : snprintf(memstr, sizeof(memstr), "%zu", (size_t) memory << 20); /* convert from MiB to bytes */
729 0 : if ((setlen = mo_add_option(&set, setlen, opt_cmdline, "gdk_vm_maxsize", memstr)) == psetlen) {
730 0 : mo_free_options(set, setlen);
731 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
732 0 : goto cleanup;
733 : }
734 : }
735 1 : if (mdbe->msg) {
736 0 : mo_free_options(set, setlen);
737 0 : goto cleanup;
738 : }
739 :
740 1 : if (!dbdir) { /* in-memory */
741 1 : if (BBPaddfarm(NULL, (1U << PERSISTENT) | (1U << TRANSIENT), false) != GDK_SUCCEED) {
742 0 : mo_free_options(set, setlen);
743 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "Cannot add in-memory farm"));
744 0 : goto cleanup;
745 : }
746 : } else {
747 0 : if (BBPaddfarm(dbdir, 1U << PERSISTENT, false) != GDK_SUCCEED ||
748 0 : BBPaddfarm(/*dbextra ? dbextra : */dbdir, 1U << TRANSIENT, false) != GDK_SUCCEED) {
749 0 : mo_free_options(set, setlen);
750 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "Cannot add farm %s", dbdir));
751 0 : goto cleanup;
752 : }
753 0 : if (GDKcreatedir(dbdir) != GDK_SUCCEED) {
754 0 : mo_free_options(set, setlen);
755 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "Cannot create directory %s", dbdir));
756 0 : goto cleanup;
757 : }
758 : }
759 1 : gdk_res = GDKinit(set, setlen, true, mercurial_revision());
760 1 : mo_free_options(set, setlen);
761 1 : if (gdk_res != GDK_SUCCEED) {
762 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", "GDKinit() failed"));
763 0 : goto cleanup;
764 : }
765 :
766 1 : if ((mdbe->msg = malEmbeddedBoot(workers, memory, 0, 0, with_mapi_server)) != MAL_SUCCEED)
767 0 : goto cleanup;
768 :
769 1 : monetdbe_embedded_initialized = true;
770 1 : monetdbe_embedded_url = dbdir?GDKstrdup(dbdir):NULL;
771 1 : if (dbdir && !monetdbe_embedded_url)
772 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_startup", MAL_MALLOC_FAIL));
773 1 : cleanup:
774 1 : GDKfataljumpenable = 0;
775 1 : if (mdbe->msg)
776 0 : monetdbe_shutdown_internal();
777 : }
778 :
779 0 : static bool urls_matches(const char* l, const char* r) {
780 0 : return (l && r && (strcmp(l, r) == 0)) || (l == NULL && r == NULL);
781 : }
782 :
783 : static inline str
784 1 : monetdbe_create_uri(const char* host, const int port, const char* database) {
785 1 : const char* protocol = "mapi:monetdb://";
786 :
787 1 : const size_t sl_protocol = strlen(protocol);
788 1 : const size_t sl_host = strlen(host);
789 1 : const size_t sl_max_port = 6; // 2^16-1 < 100 000 = 10^5, i.e. always less then 6 digits.
790 1 : const size_t sl_database = strlen(database);
791 1 : const size_t sl_total = sl_protocol + sl_host + 1 /* : */ + sl_max_port + 1 + /* / */ + sl_database;
792 :
793 1 : str uri_buffer = GDKmalloc(sl_total + 1 /* terminator */);
794 1 : if (!uri_buffer)
795 : return NULL;
796 :
797 1 : snprintf(uri_buffer, sl_total, "%s%s:%d/%s", protocol, host, port, database);
798 :
799 1 : return uri_buffer;
800 : }
801 :
802 : static int
803 1 : monetdbe_open_remote(monetdbe_database_internal *mdbe, monetdbe_options *opts) {
804 1 : assert(opts);
805 :
806 1 : monetdbe_remote* remote = opts->remote;
807 1 : if (!remote) {
808 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", "Missing remote proxy settings"));
809 0 : return -1;
810 : }
811 :
812 1 : Client c = mdbe->c;
813 :
814 1 : assert(!c->curprg);
815 :
816 1 : const char mod[] = "user";
817 1 : char nme[16];
818 1 : const char *name = number2name(nme, sizeof(nme), ++((backend*) c->sqlcontext)->remote);
819 1 : c->curprg = newFunction(putName(mod), putName(name), FUNCTIONsymbol);
820 :
821 1 : if (c->curprg == NULL) {
822 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", MAL_MALLOC_FAIL));
823 0 : return -2;
824 : }
825 :
826 1 : char* url;
827 1 : if ((url = monetdbe_create_uri(remote->host, remote->port, remote->database)) == NULL) {
828 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", MAL_MALLOC_FAIL));
829 0 : return -2;
830 : }
831 :
832 1 : MalBlkPtr mb = c->curprg->def;
833 :
834 1 : InstrPtr q = getInstrPtr(mb, 0);
835 1 : q->argc = q->retc = 0;
836 1 : q = pushReturn(mb, q, newTmpVariable(mb, TYPE_str));
837 :
838 1 : InstrPtr p = newFcnCall(mb, remoteRef, connectRef);
839 1 : p = pushStr(mb, p, url);
840 1 : p = pushStr(mb, p, remote->username);
841 1 : p = pushStr(mb, p, remote->password);
842 1 : p = pushStr(mb, p, "msql");
843 1 : p = pushBit(mb, p, 1);
844 :
845 1 : GDKfree(url);
846 1 : url = NULL;
847 :
848 1 : if (p == NULL) {
849 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", MAL_MALLOC_FAIL));
850 0 : freeSymbol(c->curprg);
851 0 : c->curprg= NULL;
852 0 : return -2;
853 : }
854 1 : pushInstruction(mb, p);
855 :
856 1 : q = newInstruction(mb, NULL, NULL);
857 1 : if (q == NULL) {
858 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", MAL_MALLOC_FAIL));
859 0 : freeSymbol(c->curprg);
860 0 : c->curprg= NULL;
861 0 : return -2;
862 : }
863 1 : q->barrier= RETURNsymbol;
864 1 : q = pushReturn(mb, q, getArg(p, 0));
865 :
866 1 : pushInstruction(mb, q);
867 :
868 1 : if ( (mdbe->msg = chkProgram(c->usermodule, mb)) != MAL_SUCCEED ) {
869 0 : freeSymbol(c->curprg);
870 0 : c->curprg= NULL;
871 0 : return -2;
872 : }
873 1 : MalStkPtr stk = prepareMALstack(mb, mb->vsize);
874 1 : if (!stk) {
875 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", MAL_MALLOC_FAIL));
876 0 : freeSymbol(c->curprg);
877 0 : c->curprg= NULL;
878 0 : return -2;
879 : }
880 1 : stk->keepAlive = TRUE;
881 1 : c->qryctx.starttime = GDKusec();
882 1 : c->qryctx.endtime = c->querytimeout ? c->qryctx.starttime + c->querytimeout : 0;
883 1 : if ( (mdbe->msg = runMALsequence(c, mb, 1, 0, stk, 0, 0)) != MAL_SUCCEED ) {
884 0 : freeStack(stk);
885 0 : freeSymbol(c->curprg);
886 0 : c->curprg= NULL;
887 0 : return -2;
888 : }
889 :
890 1 : if ((mdbe->mid = GDKstrdup(*getArgReference_str(stk, p, 0))) == NULL) {
891 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open_remote", MAL_MALLOC_FAIL));
892 0 : freeStack(stk);
893 0 : freeSymbol(c->curprg);
894 0 : c->curprg= NULL;
895 0 : return -2;
896 : }
897 :
898 1 : garbageCollector(c, mb, stk, TRUE);
899 1 : freeStack(stk);
900 :
901 1 : freeSymbol(c->curprg);
902 1 : c->curprg= NULL;
903 :
904 1 : return 0;
905 : }
906 :
907 : int
908 1 : monetdbe_open(monetdbe_database *dbhdl, char *url, monetdbe_options *opts)
909 : {
910 1 : int res = 0;
911 :
912 1 : if (!dbhdl)
913 : return -1;
914 1 : if (url &&
915 0 : (strcmp(url, "in-memory") == 0 ||
916 0 : /* backward compatibility: */ strcmp(url, ":memory:") == 0))
917 1 : url = NULL;
918 1 : MT_lock_set(&embedded_lock);
919 1 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)GDKzalloc(sizeof(monetdbe_database_internal));
920 1 : if (!mdbe) {
921 0 : MT_lock_unset(&embedded_lock);
922 0 : return -1;
923 : }
924 1 : *dbhdl = (monetdbe_database)mdbe;
925 1 : mdbe->msg = NULL;
926 1 : mdbe->c = NULL;
927 :
928 1 : bool is_remote = (opts && (opts->remote != NULL));
929 1 : if (!monetdbe_embedded_initialized) {
930 : /* When used as a remote mapi proxy,
931 : * it is still necessary to have an initialized monetdbe. E.g. for BAT life cycle management.
932 : * Use an ephemeral/anonymous dbfarm when there is no initialized monetdbe yet.
933 : */
934 1 : assert(!is_remote||url==NULL);
935 1 : monetdbe_startup(mdbe, url, opts);
936 1 : mdbe->registered_thread = 2;
937 0 : } else if (!is_remote && !urls_matches(monetdbe_embedded_url, url)) {
938 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_open", "monetdbe_open currently only one active database is supported"));
939 : }
940 1 : if (!mdbe->msg)
941 1 : res = monetdbe_open_internal(mdbe, opts);
942 :
943 1 : if (res == 0 && is_remote && !mdbe->msg)
944 1 : res = monetdbe_open_remote(mdbe, opts);
945 :
946 1 : MT_lock_unset(&embedded_lock);
947 1 : if (mdbe->msg)
948 0 : return -2;
949 : return res;
950 : }
951 :
952 : int
953 1 : monetdbe_close(monetdbe_database dbhdl)
954 : {
955 1 : if (!dbhdl)
956 : return 0;
957 :
958 1 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
959 :
960 1 : int err = 0;
961 1 : int registered_thread = mdbe->registered_thread;
962 :
963 1 : if (mdbe->c)
964 1 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
965 1 : MT_lock_set(&embedded_lock);
966 1 : if (mdbe->mid)
967 1 : err = monetdbe_close_remote(mdbe);
968 :
969 1 : err = (monetdbe_close_internal(mdbe) || err);
970 :
971 1 : if (registered_thread == 1) {
972 0 : MT_thread_deregister();
973 : }
974 1 : if (!open_dbs)
975 1 : monetdbe_shutdown_internal();
976 1 : MT_lock_unset(&embedded_lock);
977 :
978 1 : if (err) {
979 : return -2;
980 : }
981 :
982 : return 0;
983 : }
984 :
985 : char *
986 0 : monetdbe_error(monetdbe_database dbhdl)
987 : {
988 0 : if (!dbhdl)
989 : return NULL;
990 :
991 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
992 0 : return mdbe->msg;
993 : }
994 :
995 : char*
996 0 : monetdbe_load_extension(monetdbe_database dbhdl, const char *file)
997 : {
998 0 : if (!dbhdl)
999 : return 0;
1000 :
1001 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1002 :
1003 0 : if ((mdbe->msg = validate_database_handle(mdbe, "embedded.monetdbe_dump_database")) != MAL_SUCCEED) {
1004 : return mdbe->msg;
1005 : }
1006 0 : char *modules[2];
1007 0 : modules[0] = (char*)file;
1008 0 : modules[1] = NULL;
1009 0 : char *msg = loadLibrary(file, -1);
1010 0 : if (msg)
1011 : return msg;
1012 0 : return malIncludeModules(mdbe->c, modules, 0, true, NULL);
1013 : }
1014 :
1015 : char*
1016 0 : monetdbe_dump_database(monetdbe_database dbhdl, const char *filename)
1017 : {
1018 0 : if (!dbhdl)
1019 : return NULL;
1020 :
1021 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1022 :
1023 0 : if (mdbe->mid) {
1024 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_dump_database", PROGRAM_NYI));
1025 0 : return mdbe->msg;
1026 : }
1027 :
1028 0 : if ((mdbe->msg = validate_database_handle(mdbe, "embedded.monetdbe_dump_database")) != MAL_SUCCEED) {
1029 : return mdbe->msg;
1030 : }
1031 :
1032 0 : mdbe->msg = monetdbe_mapi_dump_database(dbhdl, filename);
1033 :
1034 0 : return mdbe->msg;
1035 : }
1036 :
1037 : char*
1038 0 : monetdbe_dump_table(monetdbe_database dbhdl, const char *sname, const char *tname, const char *filename)
1039 : {
1040 0 : if (!dbhdl)
1041 : return NULL;
1042 :
1043 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1044 :
1045 0 : if (mdbe->mid) {
1046 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_dump_database", PROGRAM_NYI));
1047 0 : return mdbe->msg;
1048 : }
1049 :
1050 0 : if ((mdbe->msg = validate_database_handle(mdbe, "embedded.monetdbe_dump_table")) != MAL_SUCCEED) {
1051 :
1052 : return mdbe->msg;
1053 : }
1054 :
1055 0 : mdbe->msg = monetdbe_mapi_dump_table(dbhdl, sname, tname, filename);
1056 :
1057 0 : return mdbe->msg;
1058 : }
1059 :
1060 : char*
1061 0 : monetdbe_get_autocommit(monetdbe_database dbhdl, int* result)
1062 : {
1063 0 : if (!dbhdl)
1064 : return NULL;
1065 :
1066 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1067 :
1068 0 : if ((mdbe->msg = validate_database_handle(mdbe, "monetdbe.monetdbe_get_autocommit")) != MAL_SUCCEED) {
1069 :
1070 : return mdbe->msg;
1071 : }
1072 :
1073 0 : if (!result) {
1074 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_autocommit", "Parameter result is NULL"));
1075 : } else {
1076 0 : mvc *m = ((backend *) mdbe->c->sqlcontext)->mvc;
1077 0 : *result = m->session->auto_commit;
1078 : }
1079 :
1080 0 : return mdbe->msg;
1081 : }
1082 :
1083 : char*
1084 0 : monetdbe_set_autocommit(monetdbe_database dbhdl, int value)
1085 : {
1086 0 : if (!dbhdl)
1087 : return NULL;
1088 :
1089 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1090 :
1091 0 : if (!validate_database_handle_noerror(mdbe)) {
1092 :
1093 : return NULL;
1094 : }
1095 :
1096 0 : mvc *m = ((backend *) mdbe->c->sqlcontext)->mvc;
1097 0 : int commit = !m->session->auto_commit && value;
1098 :
1099 0 : m->session->auto_commit = value;
1100 0 : m->session->ac_on_commit = m->session->auto_commit;
1101 0 : if (m->session->tr->active) {
1102 0 : if (commit) {
1103 0 : mdbe->msg = mvc_commit(m, 0, NULL, true);
1104 : } else {
1105 0 : mdbe->msg = mvc_rollback(m, 0, NULL, true);
1106 : }
1107 : }
1108 :
1109 0 : return mdbe->msg;
1110 : }
1111 :
1112 : int
1113 0 : monetdbe_in_transaction(monetdbe_database dbhdl)
1114 : {
1115 0 : if (!dbhdl)
1116 : return 0;
1117 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1118 :
1119 0 : if (!validate_database_handle_noerror(mdbe)) {
1120 :
1121 : return 0;
1122 : }
1123 :
1124 0 : mvc *m = ((backend *) mdbe->c->sqlcontext)->mvc;
1125 0 : int result = 0;
1126 :
1127 0 : if (m->session->tr)
1128 0 : result = m->session->tr->active;
1129 :
1130 : return result;
1131 : }
1132 :
1133 : struct callback_context {
1134 : monetdbe_database_internal *mdbe;
1135 : };
1136 :
1137 : static str
1138 1 : monetdbe_set_remote_results(backend *be, char* tblname, columnar_result* results, size_t nr_results) {
1139 :
1140 1 : char* error = NULL;
1141 :
1142 1 : if (nr_results == 0)
1143 : return MAL_SUCCEED; // No work to do.
1144 :
1145 1 : BAT* b_0 = BATdescriptor(results[0].id); // Fetch the first column to get the count.
1146 1 : if (!b_0) {
1147 0 : error = createException(MAL,"monetdbe.monetdbe_set_remote_results",SQLSTATE(HY005) "Cannot access column descriptor ");
1148 0 : return error;
1149 : }
1150 :
1151 1 : int res = mvc_result_table(be, 0, (int) nr_results, Q_TABLE);
1152 1 : if (res < 0) {
1153 0 : BBPunfix(b_0->batCacheid);
1154 0 : error = createException(MAL,"monetdbe.monetdbe_set_remote_results",SQLSTATE(HY005) "Cannot create result table");
1155 0 : return error;
1156 : }
1157 :
1158 3 : for (size_t i = 0; i < nr_results; i++) {
1159 2 : BAT *b = NULL;
1160 2 : if (i > 0)
1161 1 : b = BATdescriptor(results[i].id);
1162 : else
1163 : b = b_0; // We already fetched this first column
1164 :
1165 2 : char* colname = results[i].colname;
1166 2 : char* tpename = results[i].tpename;
1167 2 : int digits = results[i].digits;
1168 2 : int scale = results[i].scale;
1169 :
1170 2 : if (b == NULL) {
1171 0 : error = createException(MAL,"monetdbe.monetdbe_result_cb",SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
1172 0 : break;
1173 : }
1174 :
1175 2 : int res = mvc_result_column(be, tblname, colname, tpename, digits, scale, b);
1176 2 : BBPunfix(b->batCacheid);
1177 2 : if (res) {
1178 0 : error = createException(MAL,"monetdbe.monetdbe_result_cb", SQLSTATE(42000) "Cannot access column descriptor %s.%s",tblname,colname);
1179 0 : break;
1180 : }
1181 : }
1182 :
1183 0 : if (error)
1184 0 : res_tables_destroy(be->results);
1185 : return error;
1186 : }
1187 :
1188 : static str
1189 1 : monetdbe_result_cb(void* context, char* tblname, columnar_result* results, size_t nr_results) {
1190 1 : monetdbe_database_internal *mdbe = ((struct callback_context*) context)->mdbe;
1191 :
1192 1 : backend *be = NULL;
1193 1 : if ((mdbe->msg = getBackendContext(mdbe->c, &be)) != NULL)
1194 : return mdbe->msg;
1195 :
1196 1 : return monetdbe_set_remote_results(be, tblname, results, nr_results);
1197 : }
1198 :
1199 : struct prepare_callback_context {
1200 : int* prepare_id;
1201 : monetdbe_database_internal *mdbe;
1202 : };
1203 :
1204 : static str
1205 0 : monetdbe_prepare_cb(void* context, char* tblname, columnar_result* results, size_t nr_results) {
1206 0 : (void) tblname;
1207 0 : monetdbe_database_internal *mdbe = ((struct prepare_callback_context*) context)->mdbe;
1208 0 : int *prepare_id = ((struct prepare_callback_context*) context)->prepare_id;
1209 :
1210 0 : if (nr_results != 7) // 1) btype 2) bdigits 3) bscale 4) bschema 5) btable 6) bcolumn 7) bimpl
1211 0 : return createException(SQL, "monetdbe.monetdbe_prepare_cb", SQLSTATE(42000) "result table for prepared statement is wrong.");
1212 :
1213 0 : backend *be = NULL;
1214 0 : if ((mdbe->msg = getBackendContext(mdbe->c, &be)) != NULL)
1215 : return mdbe->msg;
1216 :
1217 0 : if ( (mdbe->msg = monetdbe_set_remote_results(be, tblname, results, nr_results)) != NULL)
1218 : return mdbe->msg;
1219 :
1220 0 : BAT* btype = NULL;
1221 0 : BAT* bdigits = NULL;
1222 0 : BAT* bscale = NULL;
1223 0 : BAT* bschema = NULL;
1224 0 : BAT* btable = NULL;
1225 0 : BAT* bcolumn = NULL;
1226 0 : BAT* bimpl = NULL;
1227 :
1228 0 : size_t nparams = 0;
1229 0 : BATiter btype_iter = {0};
1230 0 : BATiter bcolumn_iter = {0};
1231 0 : BATiter btable_iter = {0};
1232 0 : BATiter bimpl_iter = {0};
1233 0 : char* function = NULL;
1234 0 : Symbol prg = NULL;
1235 0 : MalBlkPtr mb = NULL;
1236 0 : InstrPtr o = NULL, e = NULL, r = NULL;
1237 0 : sql_rel* rel = NULL;
1238 0 : list *args = NULL, *rets = NULL;
1239 0 : allocator* sa = NULL;
1240 0 : ValRecord v = { .len=0 };
1241 0 : ptr vp = NULL;
1242 0 : struct callback_context* ccontext= NULL;
1243 0 : columnar_result_callback* rcb = NULL;
1244 :
1245 0 : str msg = MAL_SUCCEED;
1246 0 : if (!(btype = BATdescriptor(results[0].id)) ||
1247 0 : !(bdigits = BATdescriptor(results[1].id)) ||
1248 0 : !(bscale = BATdescriptor(results[2].id)) ||
1249 0 : !(bschema = BATdescriptor(results[3].id)) ||
1250 0 : !(btable = BATdescriptor(results[4].id)) ||
1251 0 : !(bcolumn = BATdescriptor(results[5].id)) ||
1252 0 : !(bimpl = BATdescriptor(results[6].id)))
1253 : {
1254 0 : msg = createException(SQL, "monetdbe.monetdbe_prepare_cb", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING);
1255 0 : goto cleanup;
1256 : }
1257 :
1258 0 : nparams = BATcount(btype);
1259 :
1260 0 : if (nparams != BATcount(bdigits) ||
1261 0 : nparams != BATcount(bimpl) ||
1262 0 : nparams != BATcount(bscale) ||
1263 0 : nparams != BATcount(bschema) ||
1264 0 : nparams != BATcount(btable) ||
1265 0 : nparams != BATcount(bcolumn))
1266 : {
1267 0 : msg = createException(SQL, "monetdbe.monetdbe_prepare_cb", SQLSTATE(42000) "Prepare results are incorrect");
1268 0 : goto cleanup;
1269 : }
1270 :
1271 0 : btype_iter = bat_iterator(btype);
1272 0 : bcolumn_iter = bat_iterator(bcolumn);
1273 0 : btable_iter = bat_iterator(btable);
1274 0 : bimpl_iter = bat_iterator(bimpl);
1275 0 : function = BUNtvar(btable_iter, BATcount(btable)-1);
1276 :
1277 : {
1278 0 : assert (((backend*) mdbe->c->sqlcontext)->remote < INT_MAX);
1279 0 : char nme[16] = {0};
1280 0 : const char* name = number2name(nme, sizeof(nme), ++((backend*) mdbe->c->sqlcontext)->remote);
1281 0 : prg = newFunctionArgs(userRef, putName(name), FUNCTIONsymbol, (int) nparams + 1);
1282 : }
1283 :
1284 0 : resizeMalBlk(prg->def, (int) nparams + 3 /*function declaration + remote.exec + return statement*/);
1285 0 : mb = prg->def;
1286 :
1287 0 : o = getInstrPtr(mb, 0);
1288 0 : o->retc = o->argc = 0;
1289 :
1290 0 : e = newInstructionArgs(mb, remoteRef, execRef, (int)(nparams + 5));
1291 0 : if (e == NULL) {
1292 0 : msg = createException(MAL, "monetdbe.monetdbe_prepare_cb", MAL_MALLOC_FAIL);
1293 0 : goto cleanup;
1294 : }
1295 0 : setDestVar(e, newTmpVariable(mb, TYPE_any));
1296 0 : e = pushStr(mb, e, mdbe->mid);
1297 0 : e = pushStr(mb, e, userRef);
1298 0 : e = pushStr(mb, e, function);
1299 :
1300 0 : rcb = GDKmalloc(sizeof(columnar_result_callback));
1301 0 : if (rcb == NULL) {
1302 0 : msg = createException(MAL, "monetdbe.monetdbe_prepare_cb", MAL_MALLOC_FAIL);
1303 0 : goto cleanup;
1304 : }
1305 :
1306 0 : ccontext = GDKzalloc(sizeof(struct callback_context));
1307 0 : if (ccontext == NULL) {
1308 0 : msg = createException(MAL, "monetdbe.monetdbe_prepare_cb", MAL_MALLOC_FAIL);
1309 0 : goto cleanup;
1310 : }
1311 :
1312 0 : ccontext->mdbe = mdbe;
1313 :
1314 0 : rcb->context = ccontext;
1315 0 : rcb->call = monetdbe_result_cb;
1316 :
1317 0 : vp = (ptr) rcb;
1318 :
1319 0 : VALset(&v, TYPE_ptr, &vp);
1320 0 : e = pushValue(mb, e, &v);
1321 :
1322 0 : r = newInstruction(mb, NULL, NULL);
1323 0 : if (r == NULL) {
1324 0 : msg = createException(MAL, "monetdbe.monetdbe_prepare_cb", MAL_MALLOC_FAIL);
1325 0 : goto cleanup;
1326 : }
1327 0 : r->barrier= RETURNsymbol;
1328 0 : r->argc= r->retc=0;
1329 :
1330 0 : sa = be->mvc->sa;
1331 :
1332 0 : args = new_exp_list(sa);
1333 0 : rets = new_exp_list(sa);
1334 :
1335 0 : for (size_t i = 0; i < nparams; i++) {
1336 :
1337 0 : const char *table = BUNtvar(btable_iter, i);
1338 0 : sql_type *t = SA_ZNEW(sa, sql_type);
1339 0 : const char *name = BUNtvar(btype_iter, i);
1340 0 : t->base.name = SA_STRDUP(sa, name);
1341 0 : const char *impl = BUNtvar(bimpl_iter, i);
1342 0 : t->impl = SA_STRDUP(sa, impl);
1343 0 : t->localtype = ATOMindex(t->impl);
1344 :
1345 0 : sql_subtype *st = SA_ZNEW(sa, sql_subtype);
1346 0 : sql_init_subtype(st, t, (unsigned) *(int*) Tloc(bdigits, i), (unsigned) *(int*) Tloc(bscale, i));
1347 :
1348 0 : if (strNil(table)) {
1349 : // input argument
1350 :
1351 0 : sql_arg *a = SA_ZNEW(sa, sql_arg);
1352 0 : a->type = *st;
1353 0 : append(args, a);
1354 :
1355 0 : int idx = newVariable(mb, NULL, 0, t->localtype);
1356 0 : o = pushArgument(mb, o, idx);
1357 :
1358 0 : InstrPtr p = newFcnCall(mb, remoteRef, putRef);
1359 0 : if (p == NULL) {
1360 0 : msg = createException(MAL, "monetdbe.monetdbe_prepare_cb", MAL_MALLOC_FAIL);
1361 0 : goto cleanup;
1362 : }
1363 0 : setArgType(mb, p, 0, TYPE_str);
1364 0 : p = pushStr(mb, p, mdbe->mid);
1365 0 : p = pushArgument(mb, p, idx);
1366 0 : pushInstruction(mb, p);
1367 :
1368 0 : e = pushArgument(mb, e, getArg(p, 0));
1369 : }
1370 : else {
1371 : // output argument
1372 :
1373 0 : const char *column = BUNtvar(bcolumn_iter, i);
1374 0 : sql_exp * c = exp_column(sa, table, column, st, CARD_MULTI, true, false, false);
1375 0 : append(rets, c);
1376 : }
1377 : }
1378 0 : pushInstruction(mb, e);
1379 0 : pushInstruction(mb, r);
1380 0 : e = r = NULL;
1381 :
1382 0 : if ( (mdbe->msg = chkProgram(mdbe->c->usermodule, mb)) != MAL_SUCCEED ) {
1383 0 : msg = mdbe->msg;
1384 0 : goto cleanup;
1385 : }
1386 :
1387 0 : rel = rel_project(sa, NULL, rets);
1388 0 : be->q = qc_insert(be->mvc->qc, sa, rel, NULL, args, be->mvc->type, NULL, be->no_mitosis);
1389 0 : *prepare_id = be->q->id;
1390 :
1391 : /*
1392 : * HACK: we need to rename the Symbol aka MAL function to the query cache name.
1393 : * Basically we keep the MALblock but we destroy the containing old Symbol
1394 : * and create a new one with the correct name and set its MAL block pointer to
1395 : * point to the mal block we have created in this function.
1396 : */
1397 0 : prg->def = NULL;
1398 0 : freeSymbol(prg);
1399 0 : if ((prg = newFunctionArgs(userRef, putName(be->q->name), FUNCTIONsymbol, -1)) == NULL) {
1400 0 : msg = createException(MAL, "monetdbe.monetdbe_prepare_cb", MAL_MALLOC_FAIL);
1401 0 : goto cleanup;
1402 : }
1403 0 : freeMalBlk(prg->def);
1404 0 : prg->def = mb;
1405 :
1406 : // finally add this beautiful new function to the local user module.
1407 0 : insertSymbol(mdbe->c->usermodule, prg);
1408 :
1409 0 : cleanup:
1410 0 : freeInstruction(e);
1411 0 : freeInstruction(r);
1412 0 : if (bcolumn) {
1413 0 : bat_iterator_end(&btype_iter);
1414 0 : bat_iterator_end(&bcolumn_iter);
1415 0 : bat_iterator_end(&btable_iter);
1416 0 : bat_iterator_end(&bimpl_iter);
1417 : }
1418 : // clean these up
1419 0 : BBPreclaim(btype);
1420 0 : BBPreclaim(bimpl);
1421 0 : BBPreclaim(bdigits);
1422 0 : BBPreclaim(bscale);
1423 0 : BBPreclaim(bschema);
1424 0 : BBPreclaim(btable);
1425 0 : BBPreclaim(bcolumn);
1426 :
1427 0 : if (msg && rcb) GDKfree(rcb);
1428 0 : if (msg && ccontext) GDKfree(ccontext);
1429 :
1430 : return msg;
1431 : }
1432 :
1433 : static char*
1434 1 : monetdbe_query_remote(monetdbe_database_internal *mdbe, char* query, monetdbe_result** result, monetdbe_cnt* affected_rows, int *prepare_id)
1435 : {
1436 1 : const char mod[] = "user";
1437 1 : char nme[16];
1438 :
1439 1 : Client c = mdbe->c;
1440 :
1441 1 : const char *name = number2name(nme, sizeof(nme), ++((backend*) c->sqlcontext)->remote);
1442 1 : Symbol prg = newFunction(putName(mod), putName(name), FUNCTIONsymbol);
1443 :
1444 1 : if (prg == NULL) {
1445 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", MAL_MALLOC_FAIL));
1446 0 : return mdbe->msg;
1447 : }
1448 :
1449 1 : MalBlkPtr mb = prg->def;
1450 1 : ValRecord v;
1451 1 : ptr vp;
1452 1 : columnar_result_callback* rcb;
1453 1 : InstrPtr f = getInstrPtr(mb, 0), r, p, e, o;
1454 :
1455 1 : f->retc = f->argc = 0;
1456 1 : o = newStmt(mb, remoteRef, putRef);
1457 1 : o = pushStr(mb, o, mdbe->mid);
1458 1 : o = pushBit(mb, o, TRUE);
1459 1 : if (o == NULL) {
1460 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", MAL_MALLOC_FAIL));
1461 0 : goto finalize;
1462 : }
1463 1 : pushInstruction(mb, o);
1464 :
1465 1 : if (prepare_id) {
1466 0 : size_t query_len, input_query_len, prep_len = 0;
1467 0 : input_query_len = strlen(query);
1468 0 : query_len = input_query_len + 3;
1469 0 : const char PREPARE[] = "PREPARE ";
1470 0 : prep_len = sizeof(PREPARE)-1;
1471 0 : query_len += prep_len;
1472 0 : char *nq = NULL;
1473 0 : if (!(nq = GDKmalloc(query_len))) {
1474 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", "Could not setup query stream"));
1475 0 : goto finalize;
1476 : }
1477 0 : strcpy(nq, PREPARE);
1478 0 : strcpy(nq + prep_len, query);
1479 0 : strcpy(nq + prep_len + input_query_len, "\n;");
1480 0 : query = nq;
1481 : }
1482 :
1483 1 : p = newStmt(mb, remoteRef, putRef);
1484 1 : p = pushStr(mb, p, mdbe->mid);
1485 1 : p = pushStr(mb, p, query);
1486 1 : if (p == NULL) {
1487 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", MAL_MALLOC_FAIL));
1488 0 : goto finalize;
1489 : }
1490 1 : pushInstruction(mb, p);
1491 :
1492 :
1493 1 : e = newInstruction(mb, remoteRef, execRef);
1494 1 : if (e == NULL) {
1495 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", MAL_MALLOC_FAIL));
1496 0 : goto finalize;
1497 : }
1498 1 : setDestVar(e, newTmpVariable(mb, TYPE_any));
1499 1 : e = pushStr(mb, e, mdbe->mid);
1500 1 : e = pushStr(mb, e, sqlRef);
1501 1 : e = pushStr(mb, e, evalRef);
1502 :
1503 : /*
1504 : * prepare the call back routine and its context
1505 : * and pass it over as a pointer to remote.exec.
1506 : */
1507 1 : rcb = GDKzalloc(sizeof(columnar_result_callback));
1508 1 : if (!rcb) {
1509 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", "Could not setup query stream"));
1510 0 : goto finalize;
1511 : }
1512 :
1513 1 : if (!prepare_id) {
1514 1 : struct callback_context* ccontext;
1515 1 : ccontext = GDKzalloc(sizeof(struct callback_context));
1516 1 : if (ccontext)
1517 1 : ccontext->mdbe = mdbe;
1518 1 : rcb->context = ccontext;
1519 1 : rcb->call = monetdbe_result_cb;
1520 : }
1521 : else {
1522 0 : struct prepare_callback_context* ccontext;
1523 0 : ccontext = GDKzalloc(sizeof(struct prepare_callback_context));
1524 0 : if (ccontext) {
1525 0 : ccontext->mdbe = mdbe;
1526 0 : ccontext->prepare_id = prepare_id;
1527 : }
1528 0 : rcb->context = ccontext;
1529 0 : rcb->call = monetdbe_prepare_cb;
1530 : }
1531 1 : if (!rcb->context) {
1532 0 : GDKfree(rcb);
1533 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", "Could not setup query stream"));
1534 0 : goto finalize;
1535 : }
1536 :
1537 1 : vp = (ptr) rcb;
1538 1 : VALset(&v, TYPE_ptr, &vp);
1539 1 : e = pushValue(mb, e, &v);
1540 :
1541 1 : e = pushArgument(mb, e, getArg(p, 0));
1542 1 : e = pushArgument(mb, e, getArg(o, 0));
1543 :
1544 1 : pushInstruction(mb, e);
1545 :
1546 1 : r = newInstruction(mb, NULL, NULL);
1547 1 : if (r == NULL) {
1548 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_query_remote", MAL_MALLOC_FAIL));
1549 0 : goto finalize;
1550 : }
1551 1 : r->barrier= RETURNsymbol;
1552 1 : r->argc= r->retc=0;
1553 1 : pushInstruction(mb, r);
1554 :
1555 1 : if ( (mdbe->msg = chkProgram(c->usermodule, mb)) != MAL_SUCCEED )
1556 0 : goto finalize;
1557 :
1558 1 : if ( (mdbe->msg = runMAL(c, mb, 0, NULL)) != MAL_SUCCEED )
1559 0 : goto finalize;
1560 :
1561 1 : if (result) {
1562 1 : if ((mdbe->msg = monetdbe_get_results(result, mdbe)) != MAL_SUCCEED)
1563 0 : goto finalize;
1564 :
1565 1 : mvc* m = NULL;
1566 1 : backend * be = NULL;
1567 1 : if ((mdbe->msg = getSQLContext(c, NULL, &m, &be)) != MAL_SUCCEED)
1568 0 : goto finalize;
1569 :
1570 1 : if (m->emode & m_prepare)
1571 0 : ((monetdbe_result_internal*) *result)->type = Q_PREPARE;
1572 : else
1573 1 : ((monetdbe_result_internal*) *result)->type = (be->results) ? be->results->query_type : m->type;
1574 :
1575 1 : if (!be->results && be->rowcnt >= 0 && affected_rows)
1576 0 : *affected_rows = be->rowcnt;
1577 : }
1578 :
1579 0 : finalize:
1580 1 : freeSymbol(prg);
1581 1 : return mdbe->msg;
1582 : }
1583 :
1584 : char*
1585 1 : monetdbe_query(monetdbe_database dbhdl, char* query, monetdbe_result** result, monetdbe_cnt* affected_rows)
1586 : {
1587 1 : if (!dbhdl)
1588 : return NULL;
1589 1 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1590 :
1591 1 : assert(mdbe->c);
1592 1 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
1593 1 : if (mdbe->mid) {
1594 1 : mdbe->msg = monetdbe_query_remote(mdbe, query, result, affected_rows, NULL);
1595 : }
1596 : else {
1597 0 : mdbe->msg = monetdbe_query_internal(mdbe, query, result, affected_rows, NULL, 'S');
1598 : }
1599 :
1600 1 : return mdbe->msg;
1601 : }
1602 :
1603 : char*
1604 0 : monetdbe_prepare(monetdbe_database dbhdl, char* query, monetdbe_statement **stmt, monetdbe_result** result)
1605 : {
1606 0 : if (!dbhdl)
1607 : return NULL;
1608 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1609 :
1610 0 : int prepare_id = 0;
1611 :
1612 0 : assert(mdbe->c);
1613 0 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
1614 0 : if (!stmt) {
1615 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_prepare", "Parameter stmt is NULL"));
1616 0 : assert(mdbe->msg != MAL_SUCCEED); /* help Coverity */
1617 0 : } else if (mdbe->mid) {
1618 0 : mdbe->msg = monetdbe_query_remote(mdbe, query, result, NULL, &prepare_id);
1619 : } else {
1620 0 : *stmt = NULL;
1621 0 : mdbe->msg = monetdbe_query_internal(mdbe, query, result, NULL, &prepare_id, 'S');
1622 : }
1623 :
1624 0 : if (mdbe->msg == MAL_SUCCEED) {
1625 0 : mvc *m = ((backend *) mdbe->c->sqlcontext)->mvc;
1626 0 : monetdbe_stmt_internal *stmt_internal = (monetdbe_stmt_internal*)GDKzalloc(sizeof(monetdbe_stmt_internal));
1627 0 : cq *q = qc_find(m->qc, prepare_id);
1628 :
1629 0 : if (q && stmt_internal) {
1630 0 : Symbol s = findSymbolInModule(mdbe->c->usermodule, q->f->imp);
1631 0 : assert(s->def);
1632 0 : InstrPtr p = s->def->stmt[0];
1633 0 : stmt_internal->mdbe = mdbe;
1634 0 : stmt_internal->q = q;
1635 0 : stmt_internal->retc = p->retc;
1636 0 : stmt_internal->res.nparam = list_length(q->f->ops);
1637 0 : stmt_internal->args = (ValPtr*)GDKmalloc(sizeof(ValPtr) * (stmt_internal->res.nparam + stmt_internal->retc));
1638 0 : stmt_internal->data = (ValRecord*)GDKzalloc(sizeof(ValRecord) * (stmt_internal->res.nparam+1));
1639 0 : stmt_internal->res.type = (monetdbe_types*)GDKmalloc(sizeof(monetdbe_types) * (stmt_internal->res.nparam+1));
1640 0 : if (!stmt_internal->res.type || !stmt_internal->data || !stmt_internal->args) {
1641 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_prepare", MAL_MALLOC_FAIL));
1642 0 : } else if (q->f->ops) {
1643 0 : int i = 0;
1644 0 : for (node *n = q->f->ops->h; n; n = n->next, i++) {
1645 0 : sql_arg *a = n->data;
1646 0 : sql_subtype *t = &a->type;
1647 0 : stmt_internal->res.type[i] = embedded_type(t->type->localtype);
1648 0 : stmt_internal->args[i+stmt_internal->retc] = &stmt_internal->data[i];
1649 : }
1650 : }
1651 0 : } else if (!stmt_internal)
1652 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_prepare", MAL_MALLOC_FAIL));
1653 :
1654 0 : if (mdbe->msg == MAL_SUCCEED)
1655 0 : *stmt = (monetdbe_statement*)stmt_internal;
1656 0 : else if (stmt_internal) {
1657 0 : GDKfree(stmt_internal->data);
1658 0 : GDKfree(stmt_internal->args);
1659 0 : GDKfree(stmt_internal->res.type);
1660 0 : GDKfree(stmt_internal);
1661 0 : *stmt = NULL;
1662 : }
1663 : }
1664 :
1665 0 : return mdbe->msg;
1666 : }
1667 :
1668 : char*
1669 0 : monetdbe_bind(monetdbe_statement *stmt, void *data, size_t i)
1670 : {
1671 0 : monetdbe_stmt_internal *stmt_internal = (monetdbe_stmt_internal*)stmt;
1672 :
1673 0 : if (i >= stmt->nparam)
1674 0 : return createException(MAL, "monetdbe.monetdbe_bind", "Parameter %zu not bound to a value", i);
1675 0 : sql_arg *a = (sql_arg*)list_fetch(stmt_internal->q->f->ops, (int) i);
1676 0 : assert(a);
1677 0 : int tpe = a->type.type->localtype;
1678 0 : stmt_internal->data[i].vtype = tpe;
1679 :
1680 0 : const void* nil = (tpe>=0)?ATOMnilptr(tpe):NULL;
1681 0 : if (!data) {
1682 0 : VALset(&stmt_internal->data[i], tpe, (ptr)nil);
1683 0 : } else if (tpe == TYPE_timestamp) {
1684 0 : monetdbe_data_timestamp* ts = (monetdbe_data_timestamp*)data;
1685 0 : timestamp t = *(timestamp*) nil;
1686 0 : if(!timestamp_is_null(ts))
1687 0 : t = timestamp_from_data(ts);
1688 0 : VALset(&stmt_internal->data[i], tpe, &t);
1689 : } else if (tpe == TYPE_date) {
1690 0 : monetdbe_data_date* de = (monetdbe_data_date*)data;
1691 0 : date d = *(date*) nil;
1692 0 : if(!date_is_null(de))
1693 0 : d = date_from_data(de);
1694 0 : VALset(&stmt_internal->data[i], tpe, &d);
1695 : } else if (tpe == TYPE_daytime) {
1696 0 : monetdbe_data_time* t = (monetdbe_data_time*)data;
1697 0 : daytime dt = *(daytime*) nil;
1698 :
1699 0 : if(!time_is_null(t))
1700 0 : dt = time_from_data(t);
1701 0 : VALset(&stmt_internal->data[i], tpe, &dt);
1702 : } else if (tpe == TYPE_blob) {
1703 0 : monetdbe_data_blob *be = (monetdbe_data_blob*)data;
1704 0 : blob *b = (blob*)nil;
1705 0 : if (!blob_is_null(be)) {
1706 0 : size_t len = be->size;
1707 0 : b = (blob*) GDKmalloc(blobsize(len));
1708 0 : if (b == NULL) {
1709 0 : set_error(stmt_internal->mdbe, createException(MAL, "monetdbe.monetdbe_bind", MAL_MALLOC_FAIL));
1710 0 : return stmt_internal->mdbe->msg;
1711 : }
1712 0 : b->nitems = len;
1713 0 : memcpy(b->data, be->data, len);
1714 : }
1715 0 : VALset(&stmt_internal->data[i], tpe, b);
1716 : } else if (tpe == TYPE_str) {
1717 0 : char *val = GDKstrdup(data);
1718 :
1719 0 : if (val == NULL) {
1720 0 : set_error(stmt_internal->mdbe, createException(MAL, "monetdbe.monetdbe_bind", MAL_MALLOC_FAIL));
1721 0 : return stmt_internal->mdbe->msg;
1722 : }
1723 0 : VALset(&stmt_internal->data[i], tpe, val);
1724 : } else {
1725 0 : VALset(&stmt_internal->data[i], tpe, data);
1726 : }
1727 : return MAL_SUCCEED;
1728 : }
1729 :
1730 : char*
1731 0 : monetdbe_execute(monetdbe_statement *stmt, monetdbe_result **result, monetdbe_cnt *affected_rows)
1732 : {
1733 0 : monetdbe_result_internal *res_internal = NULL;
1734 0 : monetdbe_stmt_internal *stmt_internal = (monetdbe_stmt_internal*)stmt;
1735 0 : backend *b = (backend *) stmt_internal->mdbe->c->sqlcontext;
1736 0 : mvc *m = b->mvc;
1737 0 : monetdbe_database_internal *mdbe = stmt_internal->mdbe;
1738 0 : MalStkPtr glb = NULL;
1739 0 : cq *q = stmt_internal->q;
1740 0 : Symbol s = NULL;
1741 :
1742 0 : assert(mdbe->c);
1743 0 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
1744 0 : if ((mdbe->msg = SQLtrans(m)) != MAL_SUCCEED)
1745 : return mdbe->msg;
1746 :
1747 : /* check if all inputs are bound */
1748 0 : for(int i = 0; i< list_length(stmt_internal->q->f->ops); i++){
1749 0 : if (!stmt_internal->data[i].vtype) {
1750 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_execute", "Parameter %d not bound to a value", i));
1751 0 : goto cleanup;
1752 : }
1753 : }
1754 :
1755 0 : s = findSymbolInModule(mdbe->c->usermodule, q->f->imp);
1756 0 : if ((mdbe->msg = callMAL(mdbe->c, s->def, &glb, stmt_internal->args)) != MAL_SUCCEED)
1757 0 : goto cleanup;
1758 :
1759 0 : if (b->rowcnt >= 0 && affected_rows)
1760 0 : *affected_rows = b->rowcnt;
1761 :
1762 0 : if (result) {
1763 0 : if ((mdbe->msg = monetdbe_get_results(result, mdbe)) != MAL_SUCCEED) {
1764 0 : goto cleanup;
1765 : }
1766 :
1767 0 : res_internal = *(monetdbe_result_internal**)result;
1768 0 : res_internal->type = (b->results) ? Q_TABLE : Q_UPDATE;
1769 0 : if (res_internal->monetdbe_resultset && res_internal->monetdbe_resultset->query_type == Q_TABLE) {
1770 0 : res_internal->type = Q_TABLE;
1771 0 : if (affected_rows)
1772 0 : *affected_rows = res_internal->monetdbe_resultset->nr_rows;
1773 : }
1774 : }
1775 :
1776 0 : cleanup:
1777 0 : GDKfree(glb);
1778 0 : return commit_action(m, stmt_internal->mdbe, result, res_internal);
1779 : }
1780 :
1781 : char*
1782 0 : monetdbe_cleanup_statement(monetdbe_database dbhdl, monetdbe_statement *stmt)
1783 : {
1784 0 : monetdbe_stmt_internal *stmt_internal = (monetdbe_stmt_internal*)stmt;
1785 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1786 0 : mvc *m = ((backend *) mdbe->c->sqlcontext)->mvc;
1787 0 : cq *q = stmt_internal->q;
1788 :
1789 0 : assert(!stmt_internal->mdbe || mdbe == stmt_internal->mdbe);
1790 :
1791 0 : assert(mdbe->c);
1792 0 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
1793 0 : for (size_t i = 0; i < stmt_internal->res.nparam + 1; i++) {
1794 0 : ValPtr data = &stmt_internal->data[i];
1795 0 : VALclear(data);
1796 : }
1797 :
1798 0 : GDKfree(stmt_internal->data);
1799 0 : GDKfree(stmt_internal->args);
1800 0 : GDKfree(stmt_internal->res.type);
1801 0 : GDKfree(stmt_internal);
1802 :
1803 0 : if (q)
1804 0 : qc_delete(m->qc, q);
1805 0 : return MAL_SUCCEED;
1806 : }
1807 :
1808 : char*
1809 1 : monetdbe_cleanup_result(monetdbe_database dbhdl, monetdbe_result* result)
1810 : {
1811 1 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1812 1 : monetdbe_result_internal* res = (monetdbe_result_internal *) result;
1813 :
1814 1 : assert(mdbe->c);
1815 1 : MT_thread_set_qry_ctx(&mdbe->c->qryctx);
1816 1 : if (!result) {
1817 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_cleanup_result", "Parameter result is NULL"));
1818 : } else {
1819 1 : mdbe->msg = monetdbe_cleanup_result_internal(mdbe, res);
1820 : }
1821 :
1822 1 : return mdbe->msg;
1823 : }
1824 :
1825 : static inline void
1826 0 : cleanup_get_columns_result(size_t column_count, monetdbe_column* columns)
1827 : {
1828 0 : if (columns) {
1829 0 : for (size_t c = 0; c < column_count; c++) {
1830 0 : GDKfree(columns[c].name);
1831 0 : GDKfree(columns[c].sql_type.name);
1832 : }
1833 0 : GDKfree(columns);
1834 0 : columns = NULL;
1835 : }
1836 0 : }
1837 :
1838 : static char *
1839 0 : escape_identifier(const char *s) /* Escapes a SQL identifier string, ie the " and \ characters */
1840 : {
1841 0 : char *ret = NULL, *q;
1842 0 : const char *p = s;
1843 :
1844 : /* At most we will need 2*strlen(s) + 1 characters */
1845 0 : if (!(ret = (char *)GDKmalloc(2*strlen(s) + 1)))
1846 : return NULL;
1847 :
1848 0 : for (q = ret; *p; p++, q++) {
1849 0 : *q = *p;
1850 0 : if (*p == '"')
1851 0 : *(++q) = '"';
1852 0 : else if (*p == '\\')
1853 0 : *(++q) = '\\';
1854 : }
1855 :
1856 0 : *q = '\0';
1857 0 : return ret;
1858 : }
1859 :
1860 : static char*
1861 0 : monetdbe_get_columns_remote(monetdbe_database_internal *mdbe, const char* schema_name, const char *table_name, size_t *column_count,
1862 : monetdbe_column **columns)
1863 : {
1864 0 : char buf[1024], *escaped_schema_name = NULL, *escaped_table_name = NULL;
1865 :
1866 0 : if (schema_name && !(escaped_schema_name = escape_identifier(schema_name))) {
1867 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", MAL_MALLOC_FAIL));
1868 0 : return mdbe->msg;
1869 : }
1870 0 : if (!(escaped_table_name = escape_identifier(table_name))) {
1871 0 : GDKfree(escaped_schema_name);
1872 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", MAL_MALLOC_FAIL));
1873 0 : return mdbe->msg;
1874 : }
1875 :
1876 0 : int len = snprintf(buf, 1024, "SELECT * FROM %s%s%s\"%s\" WHERE FALSE;",
1877 : escaped_schema_name ? "\"" : "", escaped_schema_name ? escaped_schema_name : "",
1878 : escaped_schema_name ? escaped_schema_name : "\".", escaped_table_name);
1879 0 : GDKfree(escaped_schema_name);
1880 0 : GDKfree(escaped_table_name);
1881 0 : if (len == -1 || len >= 1024) {
1882 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", "Schema and table path is too large"));
1883 0 : return mdbe->msg;
1884 : }
1885 :
1886 0 : monetdbe_result* result = NULL;
1887 :
1888 0 : if ((mdbe->msg = monetdbe_query_remote(mdbe, buf, &result, NULL, NULL)) != MAL_SUCCEED) {
1889 : return mdbe->msg;
1890 : }
1891 :
1892 0 : *column_count = result->ncols;
1893 0 : *columns = GDKzalloc(sizeof(monetdbe_column) * result->ncols);
1894 :
1895 :
1896 0 : if (*columns == NULL)
1897 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", MAL_MALLOC_FAIL));
1898 :
1899 0 : if (!mdbe->msg)
1900 0 : for (size_t c = 0; c < result->ncols; c++) {
1901 0 : monetdbe_column* rcol;
1902 0 : if ((mdbe->msg = monetdbe_result_fetch(result, &rcol, c)) != NULL) {
1903 : break;
1904 : }
1905 :
1906 0 : if (((*columns)[c].name = GDKstrdup(rcol->name)) == NULL) {
1907 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", MAL_MALLOC_FAIL));
1908 0 : break;
1909 : }
1910 0 : if (((*columns)[c].sql_type.name = GDKstrdup(rcol->sql_type.name)) == NULL) {
1911 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", MAL_MALLOC_FAIL));
1912 0 : break;
1913 : }
1914 0 : (*columns)[c].type = rcol->type;
1915 0 : (*columns)[c].sql_type.scale = rcol->sql_type.scale;
1916 0 : (*columns)[c].sql_type.digits = rcol->sql_type.digits;
1917 : }
1918 :
1919 : // cleanup
1920 0 : char* msg = monetdbe_cleanup_result_internal(mdbe, (monetdbe_result_internal*) result);
1921 :
1922 0 : if (msg && mdbe->msg) {
1923 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", "multiple errors: %s; %s", mdbe->msg, msg));
1924 : }
1925 0 : else if (msg) {
1926 0 : mdbe->msg = msg;
1927 : }
1928 :
1929 0 : if (mdbe->msg ) {
1930 0 : cleanup_get_columns_result(*column_count, *columns);
1931 : }
1932 :
1933 0 : return mdbe->msg;
1934 : }
1935 :
1936 : char*
1937 0 : monetdbe_get_columns(monetdbe_database dbhdl, const char* schema_name, const char *table_name, size_t *column_count, monetdbe_column **columns)
1938 : {
1939 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
1940 0 : mvc *m = NULL;
1941 0 : sql_table *t = NULL;
1942 :
1943 0 : if ((mdbe->msg = validate_database_handle(mdbe, "monetdbe.monetdbe_get_columns")) != MAL_SUCCEED) {
1944 : return mdbe->msg;
1945 : }
1946 0 : if (!column_count) {
1947 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", "Parameter column_count is NULL"));
1948 0 : return mdbe->msg;
1949 : }
1950 0 : if (!columns) {
1951 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", "Parameter columns is NULL"));
1952 0 : return mdbe->msg;
1953 : }
1954 0 : if (!table_name) {
1955 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", "Parameter table_name is NULL"));
1956 0 : return mdbe->msg;
1957 : }
1958 :
1959 0 : if (mdbe->mid) {
1960 0 : return monetdbe_get_columns_remote(mdbe, schema_name, table_name, column_count, columns);
1961 : }
1962 :
1963 0 : if ((mdbe->msg = getSQLContext(mdbe->c, NULL, &m, NULL)) != MAL_SUCCEED)
1964 : return mdbe->msg;
1965 0 : if ((mdbe->msg = SQLtrans(m)) != MAL_SUCCEED)
1966 : return mdbe->msg;
1967 0 : if (!(t = find_table_or_view_on_scope(m, NULL, schema_name, table_name, "CATALOG", false))) {
1968 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_get_columns", "%s", m->errstr + 6)); /* Skip error code */
1969 0 : goto cleanup;
1970 : }
1971 :
1972 0 : *column_count = ol_length(t->columns);
1973 0 : if ((*columns = GDKzalloc(*column_count * sizeof(monetdbe_column))) == NULL ) {
1974 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_get_columns", MAL_MALLOC_FAIL));
1975 0 : goto cleanup;
1976 : }
1977 :
1978 0 : for (node *n = ol_first_node(t->columns); n; n = n->next) {
1979 0 : sql_column *col = n->data;
1980 0 : (*columns)[col->colnr].name = col->base.name; // TODO Shouldn't this be string dupped?
1981 0 : if (((*columns)[col->colnr].sql_type.name = GDKstrdup(col->type.type->base.name)) == NULL) goto cleanup;
1982 0 : (*columns)[col->colnr].type = embedded_type(col->type.type->localtype);
1983 0 : (*columns)[col->colnr].sql_type.digits = col->type.type->digits;
1984 0 : (*columns)[col->colnr].sql_type.scale = col->type.type->scale;
1985 : }
1986 :
1987 0 : cleanup:
1988 0 : mdbe->msg = commit_action(m, mdbe, NULL, NULL);
1989 :
1990 0 : return mdbe->msg;
1991 : }
1992 :
1993 : #define GENERATE_BASE_HEADERS(type, tpename) \
1994 : static int tpename##_is_null(type *value)
1995 :
1996 : #define GENERATE_BASE_FUNCTIONS(tpe, tpename, mname) \
1997 : GENERATE_BASE_HEADERS(tpe, tpename); \
1998 : static int tpename##_is_null(tpe *value) { return *value == mname##_nil; }
1999 :
2000 : #ifdef bool
2001 : #undef bool
2002 : #endif
2003 :
2004 0 : GENERATE_BASE_FUNCTIONS(int8_t, bool, bit)
2005 0 : GENERATE_BASE_FUNCTIONS(int8_t, int8_t, bte)
2006 0 : GENERATE_BASE_FUNCTIONS(int16_t, int16_t, sht)
2007 0 : GENERATE_BASE_FUNCTIONS(int32_t, int32_t, int)
2008 0 : GENERATE_BASE_FUNCTIONS(int64_t, int64_t, lng)
2009 : #ifdef HAVE_HGE
2010 0 : GENERATE_BASE_FUNCTIONS(__int128, int128_t, hge)
2011 : #endif
2012 0 : GENERATE_BASE_FUNCTIONS(size_t, size_t, oid)
2013 :
2014 0 : GENERATE_BASE_FUNCTIONS(float, float, flt)
2015 0 : GENERATE_BASE_FUNCTIONS(double, double, dbl)
2016 :
2017 : GENERATE_BASE_HEADERS(char*, str);
2018 : GENERATE_BASE_HEADERS(monetdbe_data_blob, blob);
2019 :
2020 : GENERATE_BASE_HEADERS(monetdbe_data_date, date);
2021 : GENERATE_BASE_HEADERS(monetdbe_data_time, time);
2022 : GENERATE_BASE_HEADERS(monetdbe_data_timestamp, timestamp);
2023 :
2024 : #define GENERATE_BAT_INPUT_BASE(tpe) \
2025 : monetdbe_column_##tpe *bat_data = GDKzalloc(sizeof(monetdbe_column_##tpe)); \
2026 : if (!bat_data) { \
2027 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL)); \
2028 : goto cleanup; \
2029 : } \
2030 : bat_data->type = monetdbe_##tpe; \
2031 : if ((bat_data->sql_type.name = GDKstrdup(sqltpe->type->base.name)) == NULL) {\
2032 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL)); \
2033 : goto cleanup; \
2034 : } \
2035 : bat_data->sql_type.scale = sqltpe->scale; \
2036 : bat_data->sql_type.digits = sqltpe->digits; \
2037 : bat_data->is_null = tpe##_is_null; \
2038 : if (sqltpe->type->radix == 10) bat_data->scale = pow(10, sqltpe->scale); \
2039 : column_result = (monetdbe_column*) bat_data;
2040 :
2041 : #define GENERATE_BAT_INPUT(b, tpe, tpe_name, mtype) \
2042 : { \
2043 : GENERATE_BAT_INPUT_BASE(tpe_name); \
2044 : bat_data->count = (size_t) mres->nrows; \
2045 : bat_data->null_value = mtype##_nil; \
2046 : if (bat_data->count) { \
2047 : bat_data->data = GDKzalloc(bat_data->count * sizeof(bat_data->null_value)); \
2048 : if (!bat_data->data) { \
2049 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL)); \
2050 : goto cleanup; \
2051 : } \
2052 : } \
2053 : size_t it = 0; \
2054 : mtype* val = (mtype*)Tloc(b, 0); \
2055 : /* bat is dense, materialize it */ \
2056 : for (it = 0; it < bat_data->count; it++, val++) \
2057 : bat_data->data[it] = (tpe) *val; \
2058 : }
2059 :
2060 : static char*
2061 0 : append_create_remote_append_mal_program(
2062 : Symbol* prg,
2063 : sql_schema **s,
2064 : sql_table **t,
2065 : Client c, const char* schema, const char* table, size_t ccount, monetdbe_column* columns) {
2066 :
2067 0 : char* msg = MAL_SUCCEED;
2068 0 : char buf[16] = {0};
2069 0 : char* remote_program_name = number2name(buf, sizeof(buf), ++((backend*) c->sqlcontext)->remote);
2070 :
2071 0 : assert(s && t);
2072 0 : assert(c->sqlcontext && ((backend *) c->sqlcontext)->mvc);
2073 0 : mvc* m = ((backend *) c->sqlcontext)->mvc;
2074 :
2075 0 : Symbol _prg;
2076 0 : MalBlkPtr mb = NULL;
2077 0 : InstrPtr f = NULL, v = NULL, a = NULL, r = NULL;
2078 0 : int mvc_id = -1;
2079 :
2080 0 : if (!(*s = mvc_bind_schema(m, "tmp"))) {
2081 0 : return createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2082 : }
2083 :
2084 0 : switch (sql_trans_create_table(t, m->session->tr, *s, table, NULL, tt_table, false, SQL_DECLARED_TABLE, CA_COMMIT, -1, 0)) {
2085 0 : case -1:
2086 0 : return createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2087 0 : case -2:
2088 : case -3:
2089 0 : return createException(SQL, "monetdbe.monetdbe_append", "Table name '%s' conflicts", table);
2090 : default:
2091 0 : break;
2092 : }
2093 :
2094 0 : assert(prg);
2095 :
2096 0 : *prg = NULL;
2097 0 : _prg = newFunctionArgs(userRef, putName(remote_program_name), FUNCTIONsymbol, (int) ccount + 1); // remote program
2098 0 : mb = _prg->def;
2099 :
2100 0 : f = getInstrPtr(mb, 0);
2101 0 : f->retc = f->argc = 0;
2102 0 : f = pushReturn(mb, f, newTmpVariable(mb, TYPE_int));
2103 0 : v = newFcnCall(mb, sqlRef, mvcRef);
2104 0 : if (v == NULL) {
2105 0 : msg = createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2106 0 : goto cleanup;
2107 : }
2108 0 : pushInstruction(mb, v);
2109 0 : setArgType(mb, v, 0, TYPE_int);
2110 :
2111 0 : mvc_id = getArg(v, 0);
2112 :
2113 0 : sqlstore *store;
2114 0 : store = m->session->tr->store;
2115 0 : for (size_t i = 0; i < ccount; i++) {
2116 0 : sql_column *col = NULL;
2117 0 : sql_type *tpe = SA_ZNEW(m->sa, sql_type);
2118 0 : sql_subtype *st = SA_ZNEW(m->sa, sql_subtype);
2119 0 : if (tpe == NULL || st == NULL) {
2120 0 : msg = createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2121 0 : goto cleanup;
2122 : }
2123 0 : tpe->base.name = sa_strdup(m->sa, columns[i].name);
2124 0 : tpe->localtype = monetdbe_2_gdk_type((monetdbe_types) columns[i].type);
2125 0 : tpe->digits = columns[i].sql_type.digits;
2126 0 : tpe->scale = columns[i].sql_type.scale;
2127 0 : sql_init_subtype(st, tpe, columns[i].sql_type.digits, columns[i].sql_type.scale);
2128 :
2129 0 : switch (mvc_create_column(&col, m, *t, columns[i].name, st)) {
2130 0 : case -1:
2131 0 : msg = createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2132 0 : goto cleanup;
2133 0 : case -2:
2134 : case -3:
2135 0 : msg = createException(SQL, "monetdbe.monetdbe_append", "Column name '%s' conflicts", columns[i].name);
2136 0 : goto cleanup;
2137 : default:
2138 0 : break;
2139 : }
2140 :
2141 0 : if (store->storage_api.create_col(m->session->tr, col) != LOG_OK) {
2142 0 : msg = createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2143 0 : goto cleanup;
2144 : }
2145 :
2146 0 : int idx = newTmpVariable(mb, newBatType(tpe->localtype));
2147 0 : f = pushArgument(mb, f, idx);
2148 :
2149 0 : a = newFcnCall(mb, sqlRef, appendRef);
2150 0 : if (a == NULL) {
2151 0 : msg = createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2152 0 : goto cleanup;
2153 : }
2154 0 : setArgType(mb, a, 0, TYPE_int);
2155 0 : a = pushArgument(mb, a, mvc_id);
2156 0 : a = pushStr(mb, a, schema ? schema : "sys"); /* TODO this should be better */
2157 0 : a = pushStr(mb, a, table);
2158 0 : a = pushStr(mb, a, columns[i].name);
2159 0 : a = pushArgument(mb, a, idx);
2160 0 : pushInstruction(mb, a);
2161 :
2162 0 : mvc_id = getArg(a, 0);
2163 : }
2164 :
2165 0 : r = newInstruction(mb, NULL, NULL);
2166 0 : if (r == NULL) {
2167 0 : msg = createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL);
2168 0 : goto cleanup;
2169 : }
2170 0 : r->barrier= RETURNsymbol;
2171 0 : r->retc = r->argc = 0;
2172 0 : r = pushReturn(mb, r, mvc_id);
2173 0 : r = pushArgument(mb, r, mvc_id);
2174 0 : pushInstruction(mb, r);
2175 :
2176 0 : pushEndInstruction(mb);
2177 :
2178 0 : if ( (msg = chkProgram(c->usermodule, mb)) != MAL_SUCCEED ) {
2179 0 : goto cleanup;
2180 : }
2181 :
2182 0 : assert(msg == MAL_SUCCEED);
2183 0 : *prg = _prg;
2184 0 : return msg;
2185 :
2186 : cleanup:
2187 0 : assert(msg != MAL_SUCCEED);
2188 0 : freeSymbol(_prg);
2189 0 : *prg = NULL;
2190 0 : return msg;
2191 : }
2192 :
2193 : char*
2194 0 : monetdbe_append(monetdbe_database dbhdl, const char *schema, const char *table, monetdbe_column **input, size_t column_count)
2195 : {
2196 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
2197 0 : mvc *m = NULL;
2198 0 : sql_table *t = NULL;
2199 0 : size_t i, cnt;
2200 0 : node *n;
2201 0 : Symbol remote_prg = NULL;
2202 0 : BAT *pos = NULL;
2203 0 : BUN offset;
2204 :
2205 0 : if ((mdbe->msg = validate_database_handle(mdbe, "monetdbe.monetdbe_append")) != MAL_SUCCEED) {
2206 : return mdbe->msg;
2207 : }
2208 :
2209 0 : if ((mdbe->msg = getSQLContext(mdbe->c, NULL, &m, NULL)) != MAL_SUCCEED) {
2210 0 : mdbe->msg = commit_action(m, mdbe, NULL, NULL);
2211 0 : return mdbe->msg;
2212 : }
2213 0 : sqlstore *store = m->session->tr->store;
2214 :
2215 0 : if (table == NULL) {
2216 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", "table parameter is NULL"));
2217 0 : goto cleanup;
2218 : }
2219 0 : if (input == NULL) {
2220 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", "input parameter is NULL"));
2221 0 : goto cleanup;
2222 : }
2223 0 : if (column_count < 1) {
2224 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", "column_count must be higher than 0"));
2225 0 : goto cleanup;
2226 : }
2227 :
2228 0 : if (mdbe->mid) {
2229 : // We are going to insert the data into a temporary table which is used in the coming remote logic.
2230 :
2231 0 : size_t actual_column_count = 0;
2232 0 : monetdbe_column* columns = NULL;
2233 0 : sql_schema* s = NULL;
2234 :
2235 0 : if ((mdbe->msg = monetdbe_get_columns_remote(
2236 : mdbe,
2237 : schema,
2238 : table,
2239 : &actual_column_count,
2240 : &columns)) != MAL_SUCCEED) {
2241 0 : goto remote_cleanup;
2242 : }
2243 :
2244 0 : if ((mdbe->msg = SQLtrans(m)) != MAL_SUCCEED) {
2245 0 : goto remote_cleanup;
2246 : }
2247 :
2248 0 : if ((mdbe->msg = append_create_remote_append_mal_program
2249 : (&remote_prg,
2250 : &s,
2251 : &t,
2252 : mdbe->c,
2253 : schema,
2254 : table,
2255 : actual_column_count,
2256 : columns)) != MAL_SUCCEED) {
2257 0 : goto remote_cleanup;
2258 : }
2259 :
2260 0 : insertSymbol(mdbe->c->usermodule, remote_prg);
2261 :
2262 0 : remote_cleanup:
2263 0 : if (mdbe->msg) {
2264 0 : cleanup_get_columns_result(actual_column_count, columns);
2265 0 : freeSymbol(remote_prg);
2266 0 : goto cleanup;
2267 : }
2268 : } else {
2269 : // !mdbe->mid
2270 : // inserting into existing local table.
2271 0 : sql_part *pt = NULL;
2272 :
2273 0 : if ((mdbe->msg = SQLtrans(m)) != MAL_SUCCEED)
2274 0 : goto cleanup;
2275 0 : if (!(t = find_table_or_view_on_scope(m, NULL, schema, table, "CATALOG", false))) {
2276 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "%s", m->errstr + 6)); /* Skip error code */
2277 0 : goto cleanup;
2278 : }
2279 0 : if (!insert_allowed(m, t, t->base.name, "APPEND", "append")) {
2280 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "%s", m->errstr + 6)); /* Skip error code */
2281 0 : goto cleanup;
2282 : }
2283 0 : if ((t->s && t->s->parts && (pt = partition_find_part(m->session->tr, t, NULL))) || isRangePartitionTable(t) || isListPartitionTable(t)) {
2284 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Appending to a table from a merge table hierarchy via 'monetdbe_append' is not possible at the moment"));
2285 0 : goto cleanup;
2286 : }
2287 0 : if (t->idxs) {
2288 0 : for (node *n = ol_first_node(t->idxs); n; n = n->next) {
2289 0 : sql_idx *i = n->data;
2290 :
2291 0 : if (i->key) {
2292 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append",
2293 : "Appending to a table with key constraints via 'monetdbe_append' is not possible at the moment"));
2294 0 : goto cleanup;
2295 0 : } else if (hash_index(i->type) && list_length(i->columns) > 1) {
2296 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append",
2297 : "Appending to a table with hash indexes referring to more than one column via 'monetdbe_append' is not possible at the moment"));
2298 0 : goto cleanup;
2299 0 : } else if (i->type == join_idx) {
2300 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append",
2301 : "Appending to a table with join indexes via 'monetdbe_append' is not possible at the moment"));
2302 0 : goto cleanup;
2303 : }
2304 : }
2305 : }
2306 0 : if (t->triggers) {
2307 0 : for (n = ol_first_node(t->triggers); n; n = n->next) {
2308 0 : sql_trigger *trigger = n->data;
2309 :
2310 0 : if (trigger->event == 0) { /* insert event */
2311 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append",
2312 : "Appending to a table with triggers at the insert event via 'monetdbe_append' is not possible at the moment"));
2313 0 : goto cleanup;
2314 : }
2315 : }
2316 : }
2317 : }
2318 :
2319 : /* for now no default values, ie user should supply all columns */
2320 0 : if (column_count != (size_t)ol_length(t->columns)) {
2321 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Incorrect number of columns"));
2322 0 : goto cleanup;
2323 : }
2324 :
2325 0 : cnt = input[0]->count;
2326 0 : if (store->storage_api.claim_tab(m->session->tr, t, cnt, &offset, &pos) != LOG_OK) {
2327 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", "Claim failed"));
2328 0 : goto cleanup;
2329 : }
2330 : /* signal an insert was made on the table */
2331 0 : if (!isNew(t) && isGlobal(t) && !isGlobalTemp(t) && sql_trans_add_dependency_change(m->session->tr, t->base.id, dml) != LOG_OK) {
2332 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2333 0 : goto cleanup;
2334 : }
2335 :
2336 0 : for (i = 0, n = ol_first_node(t->columns); i < column_count && n; i++, n = n->next) {
2337 0 : sql_column *c = n->data;
2338 0 : int mtype = monetdbe_2_gdk_type(input[i]->type);
2339 0 : const void* nil = (mtype>=0)?ATOMnilptr(mtype):NULL;
2340 0 : char *v = input[i]->data;
2341 :
2342 0 : if (mtype < 0) {
2343 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot find type for column %zu", i));
2344 0 : goto cleanup;
2345 0 : } else if (input[i]->count != cnt) {
2346 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Number of values don't match between columns"));
2347 0 : goto cleanup;
2348 : }
2349 0 : if (mtype >= TYPE_bit && mtype <=
2350 : #ifdef HAVE_HGE
2351 : TYPE_hge
2352 : #else
2353 : TYPE_lng
2354 : #endif
2355 : ) {
2356 : //-------------------------------------
2357 0 : BAT *bn = NULL;
2358 :
2359 0 : if (mtype != c->type.type->localtype) {
2360 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append %d into column '%s'", input[i]->type, c->base.name));
2361 0 : goto cleanup;
2362 : }
2363 :
2364 0 : if ((bn = COLnew(0, mtype, 0, TRANSIENT)) == NULL) {
2365 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot create append column"));
2366 0 : goto cleanup;
2367 : }
2368 :
2369 : //save prev heap pointer
2370 0 : char *prev_base;
2371 0 : size_t prev_size;
2372 0 : prev_base = bn->theap->base;
2373 0 : prev_size = bn->theap->free;
2374 :
2375 : //BAT heap base to input[i]->data
2376 0 : bn->theap->base = input[i]->data;
2377 0 : bn->theap->free = tailsize(bn, cnt);
2378 :
2379 : //BATsetdims(bn); called in COLnew
2380 0 : BATsetcapacity(bn, cnt);
2381 0 : BATsetcount(bn, cnt);
2382 :
2383 : //set default flags
2384 0 : BATsettrivprop(bn);
2385 :
2386 0 : if (cnt > 1) {
2387 0 : bn->tsorted = bn->trevsorted = false;
2388 0 : bn->tnosorted = bn->tnorevsorted = 0;
2389 0 : bn->tkey = false;
2390 0 : bn->tnonil = false;
2391 0 : bn->tnil = false;
2392 : }
2393 :
2394 0 : if (store->storage_api.append_col(m->session->tr, c, offset, pos, bn, cnt, true, bn->ttype) != 0) {
2395 0 : bn->theap->base = prev_base;
2396 0 : bn->theap->free = prev_size;
2397 0 : BBPreclaim(bn);
2398 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append BAT"));
2399 0 : goto cleanup;
2400 : }
2401 :
2402 0 : bn->theap->base = prev_base;
2403 0 : bn->theap->free = prev_size;
2404 0 : BBPreclaim(bn);
2405 : } else if (mtype == TYPE_str) {
2406 : char **d = (char**)v;
2407 :
2408 0 : for (size_t j=0; j<cnt; j++) {
2409 0 : if (!d[j]) {
2410 0 : d[j] = (char*) nil;
2411 0 : } else if (!checkUTF8(d[j])) {
2412 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Incorrectly encoded UTF-8"));
2413 0 : goto cleanup;
2414 : }
2415 : }
2416 0 : if (store->storage_api.append_col(m->session->tr, c, offset, pos, d, cnt, false, mtype) != 0) {
2417 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2418 0 : goto cleanup;
2419 : }
2420 : } else if (mtype == TYPE_timestamp) {
2421 0 : int err = 0;
2422 0 : timestamp *d = GDKmalloc(sizeof(timestamp)*cnt);
2423 0 : if (!d) {
2424 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2425 0 : goto cleanup;
2426 : }
2427 : monetdbe_data_timestamp* ts = (monetdbe_data_timestamp*)v;
2428 :
2429 0 : for (size_t j=0; j<cnt; j++){
2430 0 : monetdbe_data_timestamp mdt = ts[j];
2431 :
2432 0 : if (timestamp_is_null(&mdt)) {
2433 0 : d[j] = *(timestamp*) nil;
2434 : } else {
2435 0 : d[j] = timestamp_from_data(&mdt);
2436 : }
2437 : }
2438 0 : if (store->storage_api.append_col(m->session->tr, c, offset, pos, d, cnt, false, mtype) != 0) {
2439 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2440 0 : err = 1;
2441 : }
2442 0 : GDKfree(d);
2443 0 : if (err)
2444 0 : goto cleanup;
2445 : } else if (mtype == TYPE_date) {
2446 0 : int err = 0;
2447 0 : date *d = GDKmalloc(sizeof(date)*cnt);
2448 0 : if (!d) {
2449 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2450 0 : goto cleanup;
2451 : }
2452 : monetdbe_data_date* de = (monetdbe_data_date*)v;
2453 :
2454 0 : for (size_t j=0; j<cnt; j++){
2455 0 : monetdbe_data_date mdt = de[j];
2456 :
2457 0 : if (date_is_null(&mdt)) {
2458 0 : d[j] = *(date*) nil;
2459 : } else {
2460 0 : d[j] = date_from_data(&mdt);
2461 : }
2462 : }
2463 0 : if (store->storage_api.append_col(m->session->tr, c, offset, pos, d, cnt, false, mtype) != 0) {
2464 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2465 0 : err = 1;
2466 : }
2467 0 : GDKfree(d);
2468 0 : if (err)
2469 0 : goto cleanup;
2470 : } else if (mtype == TYPE_daytime) {
2471 0 : int err = 0;
2472 0 : daytime *d = GDKmalloc(sizeof(daytime)*cnt);
2473 0 : if (!d) {
2474 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2475 0 : goto cleanup;
2476 : }
2477 : monetdbe_data_time* t = (monetdbe_data_time*)v;
2478 :
2479 0 : for (size_t j=0; j<cnt; j++){
2480 0 : monetdbe_data_time mdt = t[j];
2481 :
2482 0 : if (time_is_null(&mdt)) {
2483 0 : d[j] = *(daytime*) nil;
2484 : } else {
2485 0 : d[j] = time_from_data(&mdt);
2486 : }
2487 : }
2488 0 : if (store->storage_api.append_col(m->session->tr, c, offset, pos, d, cnt, false, mtype) != 0) {
2489 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2490 0 : err = 1;
2491 : }
2492 0 : GDKfree(d);
2493 0 : if (err)
2494 0 : goto cleanup;
2495 0 : } else if (mtype == TYPE_blob) {
2496 0 : int err = 0;
2497 0 : size_t j = 0;
2498 0 : blob **d = GDKmalloc(sizeof(blob*)*cnt);
2499 0 : if (!d) {
2500 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2501 0 : goto cleanup;
2502 : }
2503 : monetdbe_data_blob* be = (monetdbe_data_blob*)v;
2504 :
2505 0 : for (j=0; j<cnt; j++){
2506 0 : if (blob_is_null(be+j)) {
2507 0 : d[j] = (blob*)nil;
2508 : } else {
2509 0 : size_t len = be[j].size;
2510 0 : size_t nlen = blobsize(len);
2511 0 : blob *b = (blob*)GDKmalloc(nlen);
2512 0 : if (!b) {
2513 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2514 0 : err = 1;
2515 0 : break;
2516 : }
2517 0 : b->nitems = len;
2518 0 : memcpy(b->data, be[j].data, len);
2519 0 : d[j] = b;
2520 : }
2521 : }
2522 0 : if (!err && store->storage_api.append_col(m->session->tr, c, offset, pos, d, cnt, false, mtype) != 0) {
2523 0 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "Cannot append values"));
2524 0 : err = 1;
2525 : }
2526 0 : for (size_t k=0; k<j; k++){
2527 0 : if (d[k] != nil)
2528 0 : GDKfree(d[k]);
2529 : }
2530 0 : GDKfree(d);
2531 0 : if (err)
2532 0 : goto cleanup;
2533 : } else {
2534 : set_error(mdbe, createException(SQL, "monetdbe.monetdbe_append", "The internal type '%s' is not supported on monetdbe append at the moment", ATOMname(mtype)));
2535 : goto cleanup;
2536 : }
2537 : }
2538 :
2539 0 : if (mdbe->mid) {
2540 0 : char nme[16];
2541 0 : const char *name = number2name(nme, sizeof(nme), ++((backend*) mdbe->c->sqlcontext)->remote);
2542 0 : Symbol prg; // local program
2543 :
2544 0 : if ( (prg = newFunction(userRef, putName(name), FUNCTIONsymbol)) == NULL ) {
2545 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2546 0 : goto cleanup;
2547 : }
2548 :
2549 0 : MalBlkPtr mb = prg->def;
2550 0 : InstrPtr f = getInstrPtr(mb, 0);
2551 0 : f->retc = f->argc = 0;
2552 :
2553 0 : InstrPtr r = newFcnCall(mb, remoteRef, registerRef);
2554 0 : if (r == NULL) {
2555 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2556 0 : freeSymbol(prg);
2557 0 : goto cleanup;
2558 : }
2559 :
2560 0 : setArgType(mb, r, 0, TYPE_str);
2561 0 : r = pushStr(mb, r, mdbe->mid);
2562 0 : r = pushStr(mb, r, userRef);
2563 0 : r = pushStr(mb, r, putName(remote_prg->name));
2564 0 : pushInstruction(mb, r);
2565 :
2566 0 : InstrPtr e = newInstructionArgs(mb, remoteRef, execRef, 4 + ol_length(t->columns));
2567 0 : if (e == NULL) {
2568 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2569 0 : freeSymbol(prg);
2570 0 : goto cleanup;
2571 : }
2572 0 : setDestVar(e, newTmpVariable(mb, TYPE_any));
2573 0 : e = pushStr(mb, e, mdbe->mid);
2574 0 : e = pushStr(mb, e, userRef);
2575 0 : e = pushArgument(mb, e, getArg(r, 0));
2576 :
2577 0 : for (i = 0, n = ol_first_node(t->columns); i < (unsigned) ol_length(t->columns); i++, n = n->next) {
2578 0 : sql_column *c = n->data;
2579 0 : BAT* b = store->storage_api.bind_col(m->session->tr, c, RDONLY);
2580 0 : if (b == NULL) {
2581 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2582 0 : freeSymbol(prg);
2583 0 : goto cleanup;
2584 : }
2585 :
2586 0 : int idx = newTmpVariable(mb, newBatType(c->type.type->localtype));
2587 0 : ValRecord v = { .bat = true, .vtype = b->ttype, .len = sizeof(int), .val.bval = b->batCacheid};
2588 0 : getVarConstant(mb, idx) = v;
2589 0 : setVarConstant(mb, idx);
2590 0 : BBPunfix(b->batCacheid);
2591 :
2592 0 : InstrPtr p = newFcnCall(mb, remoteRef, putRef);
2593 0 : if (p == NULL) {
2594 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2595 0 : freeInstruction(e);
2596 0 : freeSymbol(prg);
2597 0 : goto cleanup;
2598 : }
2599 0 : setArgType(mb, p, 0, TYPE_str);
2600 0 : p = pushStr(mb, p, mdbe->mid);
2601 0 : p = pushArgument(mb, p, idx);
2602 0 : pushInstruction(mb, p);
2603 :
2604 0 : e = pushArgument(mb, e, getArg(p, 0));
2605 : }
2606 :
2607 0 : pushInstruction(mb, e);
2608 :
2609 0 : InstrPtr ri = newInstruction(mb, NULL, NULL);
2610 0 : if (ri == NULL) {
2611 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_append", MAL_MALLOC_FAIL));
2612 0 : freeSymbol(prg);
2613 0 : goto cleanup;
2614 : }
2615 0 : ri->barrier= RETURNsymbol;
2616 0 : ri->retc = ri->argc = 0;
2617 0 : pushInstruction(mb, ri);
2618 :
2619 0 : if ( (mdbe->msg = chkProgram(mdbe->c->usermodule, mb)) != MAL_SUCCEED ) {
2620 0 : freeSymbol(prg);
2621 0 : goto cleanup;
2622 : }
2623 :
2624 0 : mdbe->msg = runMAL(mdbe->c, mb, 0, NULL);
2625 0 : freeSymbol(prg);
2626 : }
2627 :
2628 0 : cleanup:
2629 0 : if (pos)
2630 0 : BBPreclaim(pos);
2631 0 : mdbe->msg = commit_action(m, mdbe, NULL, NULL);
2632 0 : return mdbe->msg;
2633 : }
2634 :
2635 : const void *
2636 0 : monetdbe_null(monetdbe_database dbhdl, monetdbe_types t)
2637 : {
2638 0 : monetdbe_database_internal *mdbe = (monetdbe_database_internal*)dbhdl;
2639 0 : int mtype = monetdbe_2_gdk_type(t);
2640 :
2641 0 : if (mtype < 0)
2642 : return NULL;
2643 :
2644 0 : if ((mtype >= TYPE_bit && mtype <=
2645 : #ifdef HAVE_HGE
2646 : TYPE_hge
2647 : #else
2648 : TYPE_lng
2649 : #endif
2650 : ))
2651 0 : return ATOMnilptr(mtype);
2652 : else if (mtype == TYPE_str)
2653 : return NULL;
2654 : else if (mtype == TYPE_blob)
2655 0 : return &mdbe->blob_null;
2656 : else if (mtype == TYPE_date)
2657 0 : return &mdbe->date_null;
2658 : else if (mtype == TYPE_daytime)
2659 0 : return &mdbe->time_null;
2660 : else if (mtype == TYPE_timestamp)
2661 0 : return &mdbe->timestamp_null;
2662 : return NULL;
2663 : }
2664 :
2665 : char*
2666 4 : monetdbe_result_fetch(monetdbe_result* mres, monetdbe_column** res, size_t column_index)
2667 : {
2668 4 : BAT* b = NULL;
2669 4 : int bat_type;
2670 4 : mvc* m;
2671 4 : monetdbe_result_internal* result = (monetdbe_result_internal*) mres;
2672 4 : sql_subtype* sqltpe = NULL;
2673 4 : monetdbe_column* column_result = NULL;
2674 4 : size_t j = 0;
2675 4 : monetdbe_database_internal *mdbe = result->mdbe;
2676 4 : Client c = mdbe->c;
2677 :
2678 :
2679 4 : if ((mdbe->msg = validate_database_handle(mdbe, "monetdbe.monetdbe_result_fetch")) != MAL_SUCCEED) {
2680 :
2681 : return mdbe->msg;
2682 : }
2683 :
2684 4 : if ((mdbe->msg = getSQLContext(c, NULL, &m, NULL)) != MAL_SUCCEED)
2685 0 : goto cleanup;
2686 4 : if (!res) {
2687 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", "Parameter res is NULL"));
2688 0 : goto cleanup;
2689 : }
2690 4 : if (column_index >= mres->ncols) {
2691 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", "Index out of range"));
2692 0 : goto cleanup;
2693 : }
2694 : // check if we have the column converted already
2695 4 : if (result->converted_columns[column_index]) {
2696 2 : *res = result->converted_columns[column_index];
2697 :
2698 2 : return MAL_SUCCEED;
2699 : }
2700 :
2701 : // otherwise we have to convert the column
2702 2 : b = BATdescriptor(result->monetdbe_resultset->cols[column_index].b);
2703 2 : if (!b) {
2704 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", SQLSTATE(HY002) RUNTIME_OBJECT_MISSING));
2705 0 : goto cleanup;
2706 : }
2707 2 : bat_type = b->ttype;
2708 2 : sqltpe = &result->monetdbe_resultset->cols[column_index].type;
2709 :
2710 2 : if (bat_type == TYPE_bit) {
2711 0 : GENERATE_BAT_INPUT(b, int8_t, bool, bit);
2712 : } else if (bat_type == TYPE_bte) {
2713 0 : GENERATE_BAT_INPUT(b, int8_t, int8_t, bte);
2714 : } else if (bat_type == TYPE_sht) {
2715 0 : GENERATE_BAT_INPUT(b, int16_t, int16_t, sht);
2716 : } else if (bat_type == TYPE_int) {
2717 3 : GENERATE_BAT_INPUT(b, int32_t, int32_t, int);
2718 : } else if (bat_type == TYPE_oid) {
2719 0 : GENERATE_BAT_INPUT(b, size_t, size_t, oid);
2720 : } else if (bat_type == TYPE_lng) {
2721 0 : GENERATE_BAT_INPUT(b, int64_t, int64_t, lng);
2722 : #ifdef HAVE_HGE
2723 : } else if (bat_type == TYPE_hge) {
2724 0 : GENERATE_BAT_INPUT(b, __int128, int128_t, hge);
2725 : #endif
2726 : } else if (bat_type == TYPE_flt) {
2727 0 : GENERATE_BAT_INPUT(b, float, float, flt);
2728 : } else if (bat_type == TYPE_dbl) {
2729 0 : GENERATE_BAT_INPUT(b, double, double, dbl);
2730 : } else if (bat_type == TYPE_str) {
2731 1 : BATiter li;
2732 1 : BUN p = 0, q = 0;
2733 1 : GENERATE_BAT_INPUT_BASE(str);
2734 1 : bat_data->count = (size_t) mres->nrows;
2735 1 : if (bat_data->count) {
2736 1 : bat_data->data = GDKzalloc(sizeof(char *) * bat_data->count);
2737 1 : bat_data->null_value = NULL;
2738 1 : if (!bat_data->data) {
2739 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2740 0 : goto cleanup;
2741 : }
2742 : }
2743 :
2744 1 : j = 0;
2745 1 : li = bat_iterator(b);
2746 3 : BATloop(b, p, q)
2747 : {
2748 2 : const char *t = (const char*)BUNtvar(li, p);
2749 2 : if (strcmp(t, str_nil) == 0) {
2750 0 : bat_data->data[j] = NULL;
2751 : } else {
2752 2 : bat_data->data[j] = GDKstrdup(t);
2753 2 : if (!bat_data->data[j]) {
2754 0 : bat_iterator_end(&li);
2755 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2756 0 : goto cleanup;
2757 : }
2758 : }
2759 2 : j++;
2760 : }
2761 1 : bat_iterator_end(&li);
2762 : } else if (bat_type == TYPE_date) {
2763 0 : date *baseptr;
2764 0 : GENERATE_BAT_INPUT_BASE(date);
2765 0 : bat_data->count = (size_t) mres->nrows;
2766 0 : if (bat_data->count) {
2767 0 : bat_data->data = GDKmalloc(sizeof(bat_data->null_value) * bat_data->count);
2768 0 : if (!bat_data->data) {
2769 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2770 0 : goto cleanup;
2771 : }
2772 : }
2773 :
2774 0 : baseptr = (date *)Tloc(b, 0);
2775 0 : for (j = 0; j < bat_data->count; j++)
2776 0 : data_from_date(baseptr[j], bat_data->data + j);
2777 0 : memcpy(&bat_data->null_value, &mdbe->date_null, sizeof(monetdbe_data_date));
2778 : } else if (bat_type == TYPE_daytime) {
2779 0 : daytime *baseptr;
2780 0 : GENERATE_BAT_INPUT_BASE(time);
2781 0 : bat_data->count = (size_t) mres->nrows;
2782 0 : if (bat_data->count) {
2783 0 : bat_data->data = GDKmalloc(sizeof(bat_data->null_value) * bat_data->count);
2784 0 : if (!bat_data->data) {
2785 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2786 0 : goto cleanup;
2787 : }
2788 : }
2789 :
2790 0 : baseptr = (daytime *)Tloc(b, 0);
2791 0 : for (j = 0; j < bat_data->count; j++)
2792 0 : data_from_time(baseptr[j], bat_data->data + j);
2793 0 : memcpy(&bat_data->null_value, &mdbe->time_null, sizeof(monetdbe_data_time));
2794 : } else if (bat_type == TYPE_timestamp) {
2795 0 : timestamp *baseptr;
2796 0 : GENERATE_BAT_INPUT_BASE(timestamp);
2797 0 : bat_data->count = (size_t) mres->nrows;
2798 0 : if (bat_data->count) {
2799 0 : bat_data->data = GDKmalloc(sizeof(bat_data->null_value) * bat_data->count);
2800 0 : if (!bat_data->data) {
2801 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2802 0 : goto cleanup;
2803 : }
2804 : }
2805 :
2806 0 : baseptr = (timestamp *)Tloc(b, 0);
2807 0 : for (j = 0; j < bat_data->count; j++)
2808 0 : data_from_timestamp(baseptr[j], bat_data->data + j);
2809 0 : memcpy(&bat_data->null_value, &mdbe->timestamp_null, sizeof(monetdbe_data_timestamp));
2810 : } else if (bat_type == TYPE_blob) {
2811 0 : BATiter li;
2812 0 : BUN p = 0, q = 0;
2813 0 : GENERATE_BAT_INPUT_BASE(blob);
2814 0 : bat_data->count = (size_t) mres->nrows;
2815 0 : if (bat_data->count) {
2816 0 : bat_data->data = GDKmalloc(sizeof(monetdbe_data_blob) * bat_data->count);
2817 0 : if (!bat_data->data) {
2818 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2819 0 : goto cleanup;
2820 : }
2821 : }
2822 0 : j = 0;
2823 :
2824 0 : li = bat_iterator(b);
2825 0 : BATloop(b, p, q)
2826 : {
2827 0 : const blob *t = (const blob *)BUNtvar(li, p);
2828 0 : if (t->nitems == ~(size_t)0) {
2829 0 : bat_data->data[j].size = 0;
2830 0 : bat_data->data[j].data = NULL;
2831 : } else {
2832 0 : bat_data->data[j].size = t->nitems;
2833 0 : bat_data->data[j].data = GDKmalloc(t->nitems);
2834 0 : if (!bat_data->data[j].data) {
2835 0 : bat_iterator_end(&li);
2836 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2837 0 : goto cleanup;
2838 : }
2839 0 : memcpy(bat_data->data[j].data, t->data, t->nitems);
2840 : }
2841 0 : j++;
2842 : }
2843 0 : bat_iterator_end(&li);
2844 0 : bat_data->null_value.size = 0;
2845 0 : bat_data->null_value.data = NULL;
2846 : } else {
2847 : // unsupported type: convert to string
2848 0 : BATiter li;
2849 0 : BUN p = 0, q = 0;
2850 0 : GENERATE_BAT_INPUT_BASE(str);
2851 0 : bat_data->count = (size_t) mres->nrows;
2852 0 : if (bat_data->count) {
2853 0 : bat_data->null_value = NULL;
2854 0 : bat_data->data = GDKzalloc(sizeof(char *) * bat_data->count);
2855 0 : if (!bat_data->data) {
2856 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", MAL_MALLOC_FAIL));
2857 0 : goto cleanup;
2858 : }
2859 : }
2860 0 : j = 0;
2861 :
2862 0 : li = bat_iterator(b);
2863 0 : BATloop(b, p, q)
2864 : {
2865 0 : const void *t = BUNtail(li, p);
2866 0 : if (BATatoms[bat_type].atomCmp(t, BATatoms[bat_type].atomNull) == 0) {
2867 0 : bat_data->data[j] = NULL;
2868 : } else {
2869 0 : char *sresult = NULL;
2870 0 : size_t length = 0;
2871 0 : if (BATatoms[bat_type].atomToStr(&sresult, &length, t, true) == 0) {
2872 0 : bat_iterator_end(&li);
2873 0 : set_error(mdbe, createException(MAL, "monetdbe.monetdbe_result_fetch", "Failed to convert element to string"));
2874 0 : goto cleanup;
2875 : }
2876 0 : bat_data->data[j] = sresult;
2877 : }
2878 0 : j++;
2879 : }
2880 0 : bat_iterator_end(&li);
2881 : }
2882 0 : if (column_result)
2883 2 : column_result->name = result->monetdbe_resultset->cols[column_index].name;
2884 2 : cleanup:
2885 2 : BBPreclaim(b);
2886 2 : if (mdbe->msg) {
2887 0 : if (res)
2888 0 : *res = NULL;
2889 0 : monetdbe_destroy_column(column_result);
2890 2 : } else if (res) {
2891 2 : result->converted_columns[column_index] = column_result;
2892 2 : *res = result->converted_columns[column_index];
2893 : }
2894 2 : mdbe->msg = commit_action(m, mdbe, NULL, NULL);
2895 :
2896 2 : return mdbe->msg;
2897 : }
2898 :
2899 : static void
2900 1 : data_from_date(date d, monetdbe_data_date *ptr)
2901 : {
2902 1 : ptr->day = date_day(d);
2903 1 : ptr->month = date_month(d);
2904 1 : ptr->year = date_year(d);
2905 1 : }
2906 :
2907 : static date
2908 0 : date_from_data(monetdbe_data_date *ptr)
2909 : {
2910 0 : return date_create(ptr->year, ptr->month, ptr->day);
2911 : }
2912 :
2913 : static void
2914 1 : data_from_time(daytime d, monetdbe_data_time *ptr)
2915 : {
2916 1 : ptr->hours = daytime_hour(d);
2917 1 : ptr->minutes = daytime_min(d);
2918 1 : ptr->seconds = daytime_sec(d);
2919 1 : ptr->ms = daytime_usec(d) / 1000;
2920 1 : }
2921 :
2922 : static daytime
2923 0 : time_from_data(monetdbe_data_time *ptr)
2924 : {
2925 0 : return daytime_create(ptr->hours, ptr->minutes, ptr->seconds, ptr->ms * 1000);
2926 : }
2927 :
2928 : static void
2929 1 : data_from_timestamp(timestamp d, monetdbe_data_timestamp *ptr)
2930 : {
2931 1 : daytime tm = timestamp_daytime(d);
2932 1 : date dt = timestamp_date(d);
2933 :
2934 1 : ptr->date.day = date_day(dt);
2935 1 : ptr->date.month = date_month(dt);
2936 1 : ptr->date.year = date_year(dt);
2937 1 : ptr->time.hours = daytime_hour(tm);
2938 1 : ptr->time.minutes = daytime_min(tm);
2939 1 : ptr->time.seconds = daytime_sec(tm);
2940 1 : ptr->time.ms = daytime_usec(tm) / 1000;
2941 1 : }
2942 :
2943 : static timestamp
2944 0 : timestamp_from_data(monetdbe_data_timestamp *ptr)
2945 : {
2946 0 : return timestamp_create(
2947 0 : date_create(ptr->date.year, ptr->date.month, ptr->date.day),
2948 0 : daytime_create(ptr->time.hours, ptr->time.minutes, ptr->time.seconds, ptr->time.ms * 1000));
2949 : }
2950 :
2951 : const char*
2952 0 : monetdbe_get_mapi_port(void) {
2953 0 : return GDKgetenv("mapi_port");
2954 : }
|