The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
rlm_sql_sqlite.c
Go to the documentation of this file.
1/*
2 * This program is is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or (at
5 * your option) any later version.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 * You should have received a copy of the GNU General Public License
13 * along with this program; if not, write to the Free Software
14 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17/**
18 * $Id: 8d4af25aa295b1147ba12cc8b8c1f1150d21e45d $
19 * @file rlm_sql_sqlite.c
20 * @brief SQLite driver.
21 *
22 * @copyright 2013 Network RADIUS SAS (legal@networkradius.com)
23 * @copyright 2007 Apple Inc.
24 */
25RCSID("$Id: 8d4af25aa295b1147ba12cc8b8c1f1150d21e45d $")
26
27#define LOG_PREFIX "sql - sqlite"
28#include <freeradius-devel/server/base.h>
29#include <freeradius-devel/util/debug.h>
30
31#include <fcntl.h>
32#include <sys/stat.h>
33
34#include <sqlite3.h>
35
36#include "rlm_sql.h"
37#include "rlm_sql_trunk.h"
38#include "config.h"
39
40#define BOOTSTRAP_MAX (1048576 * 10)
41
42/*
43 * Allow us to use versions < 3.6.0 beta0
44 */
45#ifndef SQLITE_OPEN_NOMUTEX
46# define SQLITE_OPEN_NOMUTEX 0
47#endif
48
49#ifndef HAVE_SQLITE3_INT64
50typedef sqlite_int64 sqlite3_int64;
51#endif
52
53typedef struct {
54 sqlite3 *db;
55 sqlite3_stmt *statement;
58
59typedef struct {
60 char const *filename;
63
68
69/** Convert an sqlite status code to an sql_rcode_t
70 *
71 * @param status to convert.
72 * @return
73 * - RLM_SQL_OK - If no errors found.
74 * - RLM_SQL_ERROR - If a known, non-fatal, error occurred.
75 * - RLM_SQL_ALT_QUERY - If a constraints violation occurred.
76 * - RLM_SQL_RECONNECT - Anything else, we assume the connection can no longer be used.
77 */
79{
80 /*
81 * Lowest byte is error category, other byte may contain
82 * the extended error, depending on version.
83 */
84 switch (status & 0xff) {
85 /*
86 * Not errors
87 */
88 case SQLITE_OK:
89 case SQLITE_DONE:
90 case SQLITE_ROW:
91 return RLM_SQL_OK;
92 /*
93 * User/transient errors
94 */
95 case SQLITE_ERROR: /* SQL error or missing database */
96 case SQLITE_FULL:
97 case SQLITE_MISMATCH:
98 case SQLITE_BUSY: /* Can be caused by database locking */
99 return RLM_SQL_ERROR;
100
101 /*
102 * Constraints violations
103 */
104 case SQLITE_CONSTRAINT:
105 return RLM_SQL_ALT_QUERY;
106
107 /*
108 * Errors with the handle, that probably require reinitialisation
109 */
110 default:
111 return RLM_SQL_RECONNECT;
112 }
113}
114
115/** Determine if an error occurred, and what type of error it was
116 *
117 * @param db handle to extract error from (may be NULL).
118 * @param status to check (if unused, set to SQLITE_OK).
119 * @return
120 * - RLM_SQL_OK - If no errors found.
121 * - RLM_SQL_ERROR - If a known, non-fatal, error occurred.
122 * - RLM_SQL_ALT_QUERY - If a constraints violation occurred.
123 * - RLM_SQL_RECONNECT - Anything else. We assume the connection can no longer be used.
124 */
125static sql_rcode_t sql_check_error(sqlite3 *db, int status)
126{
127 int hstatus = SQLITE_OK;
128
129 if (db) {
130 hstatus = sqlite3_errcode(db);
131 switch (hstatus & 0xff) {
132 case SQLITE_OK:
133 case SQLITE_DONE:
134 case SQLITE_ROW:
135 hstatus = SQLITE_OK;
136 break;
137
138 default:
139 break;
140 }
141 }
142
143 switch (status & 0xff) {
144 case SQLITE_OK:
145 case SQLITE_DONE:
146 case SQLITE_ROW:
147 status = SQLITE_OK;
148 break;
149
150 default:
151 break;
152 }
153
154 if (status != SQLITE_OK) return sql_error_to_rcode(status);
155 if (hstatus != SQLITE_OK) return sql_error_to_rcode(hstatus);
156
157 return RLM_SQL_OK;
158}
159
160/** Print an error to the global debug log
161 *
162 * If status does not indicate success, write an error to the global error log.
163 *
164 * @note The error code will be appended to the fmt string in the format ": code 0x<hex> (<int>)[: <string>]".
165 *
166 * @param db handle to extract error from (may be NULL).
167 * @param status to check (if unused, set to SQLITE_OK).
168 * @param fmt to prepend.
169 * @param ... arguments to fmt.
170 */
171static void sql_print_error(sqlite3 *db, int status, char const *fmt, ...)
172 CC_HINT(format (printf, 3, 4)) CC_HINT(nonnull (3));
173static void sql_print_error(sqlite3 *db, int status, char const *fmt, ...)
174{
175 va_list ap;
176 char *p;
177 int hstatus = SQLITE_OK;
178
179 if (db) {
180 hstatus = sqlite3_errcode(db);
181 switch (hstatus & 0xff) {
182 case SQLITE_OK:
183 case SQLITE_DONE:
184 case SQLITE_ROW:
185 hstatus = SQLITE_OK;
186 break;
187
188 default:
189 break;
190 }
191 }
192
193 switch (status & 0xff) {
194 case SQLITE_OK:
195 case SQLITE_DONE:
196 case SQLITE_ROW:
197 status = SQLITE_OK;
198 break;
199
200 default:
201 break;
202 }
203
204 /*
205 * No errors!
206 */
207 if ((hstatus == SQLITE_OK) && (status == SQLITE_OK)) return;
208
209 /*
210 * At least one error...
211 */
212 va_start(ap, fmt);
213 MEM(p = talloc_vasprintf(NULL, fmt, ap));
214 va_end(ap);
215
216 /*
217 * Disagreement between handle, and function return code,
218 * print them both.
219 */
220 if ((status != SQLITE_OK) && (status != hstatus)) {
221 fr_strerror_printf("%s: Code 0x%04x (%i): %s", p, status, status, sqlite3_errstr(status));
222 }
223
224 if (hstatus != SQLITE_OK) fr_strerror_printf("%s: Code 0x%04x (%i): %s",
225 p, hstatus, hstatus, sqlite3_errmsg(db));
226}
227
228static int sql_loadfile(TALLOC_CTX *ctx, sqlite3 *db, char const *filename)
229{
230 ssize_t len;
231 int statement_len, statement_cnt = 0;
232 char *buffer;
233 char const *p;
234 int cl;
235 FILE *f;
236 struct stat finfo;
237
238 int status;
239 sqlite3_stmt *statement;
240 char const *z_tail;
241
242 INFO("Executing SQL statements from file \"%s\"", filename);
243
244 f = fopen(filename, "r");
245 if (!f) {
246 fr_strerror_printf("Failed opening SQL file \"%s\": %s", filename,
247 fr_syserror(errno));
248 return -1;
249 }
250
251 if (fstat(fileno(f), &finfo) < 0) {
252 fr_strerror_printf("Failed stating SQL file \"%s\": %s", filename,
253 fr_syserror(errno));
254 error:
255 fclose(f);
256 return -1;
257 }
258
259 if (finfo.st_size > BOOTSTRAP_MAX) {
260 too_big:
261 fr_strerror_printf("Size of SQL (%zu) file exceeds limit (%uk)",
262 (size_t) finfo.st_size / 1024, BOOTSTRAP_MAX / 1024);
263 goto error;
264 }
265
266 MEM(buffer = talloc_array(ctx, char, finfo.st_size + 1));
267 len = fread(buffer, sizeof(char), finfo.st_size, f);
268
269 if (len > finfo.st_size) {
271 goto too_big;
272 }
273
274 if (!len) {
276 if (ferror(f)) {
277 fr_strerror_printf("Failed reading from SQL file: %s", fr_syserror(errno));
278 goto error;
279 }
280
281 DEBUG("Ignoring empty SQL file");
282
283 fclose(f);
284 return 0;
285 }
286
287 buffer[len] = '\0';
288 fclose(f);
289
290 /*
291 * Check if input data is UTF-8. Allow CR/LF \t, too.
292 */
293 for (p = buffer; p < (buffer + len); p += cl) {
294 if (*p < ' ') {
295 if ((*p != 0x0a) && (*p != 0x0d) && (*p != '\t')) break;
296 cl = 1;
297 } else {
298 cl = fr_utf8_char((uint8_t const *) p, -1);
299 if (!cl) break;
300 }
301 }
302
303 if ((p - buffer) != len) {
304 fr_strerror_printf("Bootstrap file contains non-UTF8 char at offset %zu", p - buffer);
306 return -1;
307 }
308
309 p = buffer;
310 while (*p) {
311 statement_len = len - (p - buffer);
312 status = sqlite3_prepare_v2(db, p, statement_len, &statement, &z_tail);
313
314 if (sql_check_error(db, status) != RLM_SQL_OK) {
315 sql_print_error(db, status, "Failed preparing statement %i", statement_cnt);
317 return -1;
318 }
319
320 /*
321 * No SQL statement was found
322 */
323 if (!statement) break;
324
325 status = sqlite3_step(statement);
326 if (sql_check_error(db, status) != RLM_SQL_OK) {
327 sql_print_error(db, status, "Failed executing statement %i", statement_cnt);
328 sqlite3_finalize(statement);
330 return -1;
331 }
332
333 status = sqlite3_finalize(statement);
334 if (sql_check_error(db, status) != RLM_SQL_OK) {
335 sql_print_error(db, status, "Failed finalizing statement %i", statement_cnt);
337 return -1;
338 }
339
340 statement_cnt++;
341 p = z_tail;
342 }
343
345 return 0;
346}
347
348static void _sql_greatest(sqlite3_context *ctx, int num_values, sqlite3_value **values)
349{
350 int i;
351 sqlite3_int64 value, max = 0;
352
353 for (i = 0; i < num_values; i++) {
354 value = sqlite3_value_int64(values[i]);
355 if (value > max) {
356 max = value;
357 }
358 }
359
360 sqlite3_result_int64(ctx, max);
361}
362
363CC_NO_UBSAN(function) /* UBSAN: false positive - public vs private connection_t trips --fsanitize=function*/
364static connection_state_t _sql_connection_init(void **h, connection_t *conn, void *uctx)
365{
367 rlm_sql_sqlite_t *inst = talloc_get_type_abort(sql->driver_submodule->data, rlm_sql_sqlite_t);
369 rlm_sql_config_t const *config = &sql->config;
370 int status;
371
372 MEM(c = talloc_zero(conn, rlm_sql_sqlite_conn_t));
373
374 INFO("Opening SQLite database \"%s\"", inst->filename);
375 status = sqlite3_open_v2(inst->filename, &(c->db), SQLITE_OPEN_READWRITE | SQLITE_OPEN_NOMUTEX, NULL);
376
377 if (!c->db || (sql_check_error(c->db, status) != RLM_SQL_OK)) {
378 int fd;
379
380 sql_print_error(c->db, status, "Failed opening SQLite database \"%s\"", inst->filename);
381 ERROR("%s", fr_strerror());
382
383 fd = open(inst->filename, O_RDWR);
384 if (fd < 0) {
385 if (!inst->bootstrap && (errno == ENOENT)) {
386 WARN("Perhaps use the sqlite driver 'bootstrap' option to create the database file?");
387 }
388
389 fr_strerror_printf("Cannot open \"%s\" in read/write mode - %s",
390 inst->filename, fr_syserror(errno));
391 } else {
392 close(fd);
393 fr_strerror_printf("Failed in SQLite while opening \"%s\" - the file exists, but is the database corrupted?",
394 inst->filename);
395 }
396
397 error:
398 ERROR("%s", fr_strerror());
399 talloc_free(c);
401 }
402 status = sqlite3_busy_timeout(c->db, fr_time_delta_to_sec(config->query_timeout));
403 if (sql_check_error(c->db, status) != RLM_SQL_OK) {
404 sql_print_error(c->db, status, "Failed setting busy timeout");
405 goto error;
406 }
407
408 /*
409 * Enable extended return codes for extra debugging info.
410 */
411 status = sqlite3_extended_result_codes(c->db, 1);
412 if (sql_check_error(c->db, status) != RLM_SQL_OK) {
413 sql_print_error(c->db, status, "Failed enabling extended result codes");
414 goto error;
415 }
416
417 status = sqlite3_create_function_v2(c->db, "GREATEST", -1, SQLITE_ANY, NULL,
418 _sql_greatest, NULL, NULL, NULL);
419 if (sql_check_error(c->db, status) != RLM_SQL_OK) {
420 sql_print_error(c->db, status, "Failed registering 'GREATEST' sql function");
421 goto error;
422 }
423
424 *h = c;
425
427}
428
429static void _sql_connection_close(UNUSED fr_event_list_t *el, void *h, UNUSED void *uctx)
430{
431 rlm_sql_sqlite_conn_t *c = talloc_get_type_abort(h, rlm_sql_sqlite_conn_t);
432 int status = 0;
433
434 DEBUG2("Socket destructor called, closing socket");
435
436 if (c->db) {
437 status = sqlite3_close(c->db);
438 if (status != SQLITE_OK) WARN("Got SQLite error when closing socket: %s",
439 sqlite3_errmsg(c->db));
440 }
441}
442
443static sql_rcode_t sql_fields(char const **out[], fr_sql_query_t *query_ctx, UNUSED rlm_sql_config_t const *config)
444{
445 rlm_sql_sqlite_conn_t *conn = talloc_get_type_abort(query_ctx->tconn->conn->h, rlm_sql_sqlite_conn_t);
446
447 int fields, i;
448 char const **names;
449
450 fields = sqlite3_column_count(conn->statement);
451 if (fields <= 0) return RLM_SQL_ERROR;
452
453 MEM(names = talloc_array(query_ctx, char const *, fields));
454
455 for (i = 0; i < fields; i++) names[i] = sqlite3_column_name(conn->statement, i);
456 *out = names;
457
458 return RLM_SQL_OK;
459}
460
461static unlang_action_t sql_fetch_row(rlm_rcode_t *p_result, UNUSED int *priority, UNUSED request_t *request, void *uctx)
462{
463 fr_sql_query_t *query_ctx = talloc_get_type_abort(uctx, fr_sql_query_t);
464 int status, i = 0;
465 rlm_sql_sqlite_conn_t *conn = talloc_get_type_abort(query_ctx->tconn->conn->h, rlm_sql_sqlite_conn_t);
466 char **row;
467
468 TALLOC_FREE(query_ctx->row);
469
470 /*
471 * Executes the SQLite query and iterates over the results
472 */
473 status = sqlite3_step(conn->statement);
474
475 /*
476 * Error getting next row
477 */
478 if (sql_check_error(conn->db, status) != RLM_SQL_OK) {
479 error:
480 query_ctx->rcode = RLM_SQL_ERROR;
482 }
483
484 /*
485 * No more rows to process (we're done)
486 */
487 if (status == SQLITE_DONE) {
488 query_ctx->rcode = RLM_SQL_NO_MORE_ROWS;
490 }
491
492 /*
493 * We only need to do this once per result set, because
494 * the number of columns won't change.
495 */
496 if (conn->col_count == 0) {
497 conn->col_count = sqlite3_column_count(conn->statement);
498 if (conn->col_count == 0) goto error;
499 }
500
501 /*
502 * Free the previous result (also gets called on finish_query)
503 */
504 MEM(row = query_ctx->row = talloc_zero_array(query_ctx, char *, conn->col_count + 1));
505
506 for (i = 0; i < conn->col_count; i++) {
507 switch (sqlite3_column_type(conn->statement, i)) {
508 case SQLITE_INTEGER:
509 MEM(row[i] = talloc_typed_asprintf(row, "%d", sqlite3_column_int(conn->statement, i)));
510 break;
511
512 case SQLITE_FLOAT:
513 MEM(row[i] = talloc_typed_asprintf(row, "%f", sqlite3_column_double(conn->statement, i)));
514 break;
515
516 case SQLITE_TEXT:
517 {
518 char const *p;
519 p = (char const *) sqlite3_column_text(conn->statement, i);
520
521 if (p) MEM(row[i] = talloc_typed_strdup(row, p));
522 }
523 break;
524
525 case SQLITE_BLOB:
526 {
527 uint8_t const *p;
528 size_t len;
529
530 p = sqlite3_column_blob(conn->statement, i);
531 if (p) {
532 len = sqlite3_column_bytes(conn->statement, i);
533
534 MEM(row[i] = talloc_zero_array(row, char, len + 1));
535 memcpy(row[i], p, len);
536 }
537 }
538 break;
539
540 default:
541 break;
542 }
543 }
544
545 query_ctx->rcode = RLM_SQL_OK;
547}
548
550{
551 rlm_sql_sqlite_conn_t *conn = talloc_get_type_abort(query_ctx->tconn->conn->h, rlm_sql_sqlite_conn_t);
552
553 if (conn->statement) {
554 TALLOC_FREE(query_ctx->row);
555
556 (void) sqlite3_finalize(conn->statement);
557 conn->statement = NULL;
558 conn->col_count = 0;
559 }
560
561 /*
562 * There's no point in checking the code returned by finalize
563 * as it'll have already been encountered elsewhere in the code.
564 *
565 * It's just the last error that occurred processing the
566 * statement.
567 */
568 return RLM_SQL_OK;
569}
570
571/** Retrieves any errors associated with the query context
572 *
573 * @note Caller will free any memory allocated in ctx.
574 *
575 * @param ctx to allocate temporary error buffers in.
576 * @param out Array of sql_log_entrys to fill.
577 * @param outlen Length of out array.
578 * @param query_ctx Query context to retrieve error for.
579 * @return number of errors written to the #sql_log_entry_t array.
580 */
581static size_t sql_error(UNUSED TALLOC_CTX *ctx, sql_log_entry_t out[], NDEBUG_UNUSED size_t outlen,
582 fr_sql_query_t *query_ctx)
583{
584 rlm_sql_sqlite_conn_t *conn = talloc_get_type_abort(query_ctx->tconn->conn->h, rlm_sql_sqlite_conn_t);
585 char const *error;
586
587 fr_assert(outlen > 0);
588
589 error = sqlite3_errmsg(conn->db);
590 if (!error) return 0;
591
592 out[0].type = L_ERR;
593 out[0].msg = error;
594
595 return 1;
596}
597
599{
600 return sql_free_result(query_ctx, config);
601}
602
603static int sql_affected_rows(fr_sql_query_t *query_ctx,
605{
606 rlm_sql_sqlite_conn_t *conn = talloc_get_type_abort(query_ctx->tconn->conn->h, rlm_sql_sqlite_conn_t);
607
608 if (conn->db) return sqlite3_changes(conn->db);
609
610 return -1;
611}
612
614
615CC_NO_UBSAN(function) /* UBSAN: false positive - public vs private connection_t trips --fsanitize=function*/
617 connection_t *conn, UNUSED void *uctx)
618{
619 rlm_sql_sqlite_conn_t *sql_conn = talloc_get_type_abort(conn->h, rlm_sql_sqlite_conn_t);
620 trunk_request_t *treq;
621 request_t *request;
622 fr_sql_query_t *query_ctx;
623 int status;
624 char const *z_tail;
625
626 if (trunk_connection_pop_request(&treq, tconn) != 0) return;
627 if (!treq) return;
628
629 query_ctx = talloc_get_type_abort(treq->preq, fr_sql_query_t);
630 request = query_ctx->request;
631 query_ctx->tconn = tconn;
632
633 ROPTIONAL(RDEBUG2, DEBUG2, "Executing query: %s", query_ctx->query_str);
634 status = sqlite3_prepare_v2(sql_conn->db, query_ctx->query_str, strlen(query_ctx->query_str),
635 &sql_conn->statement, &z_tail);
636 query_ctx->rcode = sql_check_error(sql_conn->db, status);
637 if (query_ctx->rcode != RLM_SQL_OK) {
638 error:
639 query_ctx->status = SQL_QUERY_FAILED;
641 return;
642 }
643
644 if (query_ctx->type == SQL_QUERY_OTHER) {
645 status = sqlite3_step(sql_conn->statement);
646 query_ctx->rcode = sql_check_error(sql_conn->db, status);
647 if (query_ctx->rcode == RLM_SQL_ERROR) goto error;
648 }
649
651}
652
654
655static void sql_request_fail(UNUSED request_t *request, void *preq, UNUSED void *rctx,
656 UNUSED trunk_request_state_t state, UNUSED void *uctx)
657{
658 fr_sql_query_t *query_ctx = talloc_get_type_abort(preq, fr_sql_query_t);
659
660 query_ctx->treq = NULL;
661 if (query_ctx->rcode == RLM_SQL_OK) query_ctx->rcode = RLM_SQL_ERROR;
662}
663
664static void sql_request_complete(UNUSED request_t *request, void *preq, UNUSED void *rctx, UNUSED void *uctx)
665{
666 fr_sql_query_t *query_ctx = talloc_get_type_abort(preq, fr_sql_query_t);
667
668 sql_free_result(query_ctx, NULL);
669}
670
671static int mod_instantiate(module_inst_ctx_t const *mctx)
672{
673 rlm_sql_t const *parent = talloc_get_type_abort(mctx->mi->parent->data, rlm_sql_t);
674 rlm_sql_config_t const *config = &parent->config;
675 rlm_sql_sqlite_t *inst = talloc_get_type_abort(mctx->mi->data, rlm_sql_sqlite_t);
676 bool exists;
677 struct stat buf;
678 int fd;
679 char const *r;
680
681 if (!inst->filename) {
682 MEM(inst->filename = talloc_typed_asprintf(inst, "%s/%s",
683 main_config->raddb_dir, config->sql_db));
684 }
685
686 /*
687 * We will try to create the database if it doesn't exist, up to and
688 * including creating the directory it should live in, in which case
689 * we get to call fr_dirfd() again. Hence failing this first fr_dirfd()
690 * just means the database isn't there.
691 */
692 if (fr_dirfd(&fd, &r, inst->filename) < 0) {
693 exists = false;
694 } else if (fstatat(fd, r, &buf, 0) == 0) {
695 exists = true;
696 } else if (errno == ENOENT) {
697 exists = false;
698 } else {
699 ERROR("Database exists, but couldn't be opened: %s", fr_syserror(errno));
700 close(fd);
701 return -1;
702 }
703
704 if (cf_pair_find(mctx->mi->conf, "bootstrap")) {
705 inst->bootstrap = true;
706 }
707
708 if (inst->bootstrap && !exists) {
709 int status;
710 int ret;
711 char const *p;
712 char *buff;
713 sqlite3 *db = NULL;
714 CONF_PAIR *cp;
715
716 INFO("Database \"%s\" doesn't exist, creating it and loading schema", inst->filename);
717
718 p = strrchr(inst->filename, '/');
719 if (p) {
720 size_t len = (p - inst->filename) + 1;
721
722 buff = talloc_array(mctx->mi->conf, char, len);
723 strlcpy(buff, inst->filename, len);
724 } else {
725 MEM(buff = talloc_typed_strdup(mctx->mi->conf, inst->filename));
726 }
727
728 ret = fr_mkdir(NULL, buff, -1, 0700, NULL, NULL);
730 if (ret < 0) {
731 PERROR("Failed creating directory for SQLite database");
732
733 return -1;
734 }
735 (void) fr_dirfd(&fd, &r, inst->filename);
736
737 status = sqlite3_open_v2(inst->filename, &db, SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE, NULL);
738 if (!db) {
739 ERROR("Failed creating opening/creating SQLite database: %s",
740 sqlite3_errstr(status));
741 goto unlink;
742 }
743
744 if (sql_check_error(db, status) != RLM_SQL_OK) {
745 (void) sqlite3_close(db);
746
747 goto unlink;
748 }
749
750 /*
751 * Execute multiple bootstrap SQL files in order
752 */
753 for (cp = cf_pair_find(mctx->mi->conf, "bootstrap");
754 cp;
755 cp = cf_pair_find_next(mctx->mi->conf, cp, "bootstrap")) {
756 p = cf_pair_value(cp);
757 if (!p) continue;
758
759 ret = sql_loadfile(mctx->mi->conf, db, p);
760 if (ret < 0) {
761 ERROR("%s", fr_strerror());
762 (void) sqlite3_close(db);
763 goto unlink;
764 }
765 }
766
767 status = sqlite3_close(db);
768 if (status != SQLITE_OK) {
769 /*
770 * Safer to use sqlite3_errstr here, just in case the handle is in a weird state
771 */
772 ERROR("Failed closing SQLite handle: %s", sqlite3_errstr(status));
773 goto unlink;
774 }
775
776 if (ret < 0) {
777 unlink:
778 if ((unlinkat(fd, r, 0) < 0) && (errno != ENOENT)) {
779 ERROR("Failed to remove partially initialised database: %s",
780 fr_syserror(errno));
781 }
782 close(fd);
783 return -1;
784 }
785 }
786
787 close(fd);
788 return 0;
789}
790
791static int mod_load(void)
792{
793 if (sqlite3_libversion_number() != SQLITE_VERSION_NUMBER) {
794 WARN("libsqlite version changed since the server was built");
795 WARN("linked: %s built: %s", sqlite3_libversion(), SQLITE_VERSION);
796 }
797 INFO("libsqlite version: %s", sqlite3_libversion());
798
799 return 0;
800}
801
802/* Exported to rlm_sql */
805 .common = {
806 .name = "sql_sqlite",
807 .magic = MODULE_MAGIC_INIT,
808 .inst_size = sizeof(rlm_sql_sqlite_t),
810 .onload = mod_load,
812 },
814 .sql_query_resume = sql_query_resume,
815 .sql_select_query_resume = sql_query_resume,
816 .sql_affected_rows = sql_affected_rows,
817 .sql_fetch_row = sql_fetch_row,
818 .sql_fields = sql_fields,
819 .sql_free_result = sql_free_result,
820 .sql_error = sql_error,
821 .sql_finish_query = sql_finish_query,
822 .sql_finish_select_query = sql_finish_query,
823 .trunk_io_funcs = {
824 .connection_alloc = sql_trunk_connection_alloc,
825 .request_mux = sql_trunk_request_mux,
826 .request_complete = sql_request_complete,
827 .request_fail = sql_request_fail
828 }
829};
unlang_action_t
Returned by unlang_op_t calls, determine the next action of the interpreter.
Definition action.h:35
static int const char char buffer[256]
Definition acutest.h:576
va_end(args)
static int const char * fmt
Definition acutest.h:573
va_start(args, fmt)
#define RCSID(id)
Definition build.h:485
#define NDEBUG_UNUSED
Definition build.h:328
#define CC_NO_UBSAN(_sanitize)
Definition build.h:428
#define UNUSED
Definition build.h:317
#define CONF_PARSER_TERMINATOR
Definition cf_parse.h:658
conf_parser_flags_t flags
Flags which control parsing behaviour.
Definition cf_parse.h:601
#define FR_CONF_OFFSET_FLAGS(_name, _flags, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:272
@ CONF_FLAG_REQUIRED
Error out if no matching CONF_PAIR is found, and no dflt value is set.
Definition cf_parse.h:434
@ CONF_FLAG_FILE_OUTPUT
File matching value must exist, and must be writable.
Definition cf_parse.h:442
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:595
Configuration AVP similar to a fr_pair_t.
Definition cf_priv.h:70
CONF_PAIR * cf_pair_find_next(CONF_SECTION const *cs, CONF_PAIR const *prev, char const *attr)
Find a pair with a name matching attr, after specified pair.
Definition cf_util.c:1453
CONF_PAIR * cf_pair_find(CONF_SECTION const *cs, char const *attr)
Search for a CONF_PAIR with a specific name.
Definition cf_util.c:1439
char const * cf_pair_value(CONF_PAIR const *pair)
Return the value of a CONF_PAIR.
Definition cf_util.c:1594
connection_state_t
Definition connection.h:45
@ CONNECTION_STATE_FAILED
Connection has failed.
Definition connection.h:54
@ CONNECTION_STATE_CONNECTED
File descriptor is open (ready for writing).
Definition connection.h:52
#define MEM(x)
Definition debug.h:36
#define ERROR(fmt,...)
Definition dhcpclient.c:41
#define DEBUG(fmt,...)
Definition dhcpclient.c:39
Test enumeration values.
Definition dict_test.h:92
#define MODULE_MAGIC_INIT
Stop people using different module/library/server versions together.
Definition dl_module.h:63
ssize_t fr_mkdir(int *fd_out, char const *path, ssize_t len, mode_t mode, fr_mkdir_func_t func, void *uctx)
Create directories that are missing in the specified path.
Definition file.c:219
int fr_dirfd(int *dirfd, char const **filename, char const *pathname)
From a pathname, return fd and filename needed for *at() functions.
Definition file.c:412
#define PERROR(_fmt,...)
Definition log.h:228
#define ROPTIONAL(_l_request, _l_global, _fmt,...)
Use different logging functions depending on whether request is NULL or not.
Definition log.h:528
talloc_free(reap)
Stores all information relating to an event list.
Definition event.c:380
@ L_ERR
Error message.
Definition log.h:56
main_config_t const * main_config
Main server configuration.
Definition main_config.c:69
char const * raddb_dir
Path to raddb directory.
Definition main_config.h:78
long int ssize_t
unsigned char uint8_t
module_instance_t * mi
Instance of the module being instantiated.
Definition module_ctx.h:51
Temporary structure to hold arguments for instantiation calls.
Definition module_ctx.h:50
size_t fr_utf8_char(uint8_t const *str, ssize_t inlen)
Checks for utf-8, taken from http://www.w3.org/International/questions/qa-forms-utf-8.
Definition print.c:39
static const conf_parser_t config[]
Definition base.c:183
#define fr_assert(_expr)
Definition rad_assert.h:38
#define RDEBUG2(fmt,...)
Definition radclient.h:54
#define DEBUG2(fmt,...)
Definition radclient.h:43
#define WARN(fmt,...)
Definition radclient.h:47
#define INFO(fmt,...)
Definition radict.c:54
#define RETURN_MODULE_OK
Definition rcode.h:57
#define RETURN_MODULE_FAIL
Definition rcode.h:56
rlm_rcode_t
Return codes indicating the result of the module call.
Definition rcode.h:40
static int instantiate(module_inst_ctx_t const *mctx)
Definition rlm_rest.c:1313
Prototypes and functions for the SQL module.
fr_sql_query_status_t status
Status of the query.
Definition rlm_sql.h:140
trunk_connection_t * tconn
Trunk connection this query is being run on.
Definition rlm_sql.h:136
fr_sql_query_type_t type
Type of query.
Definition rlm_sql.h:139
char const * query_str
Query string to run.
Definition rlm_sql.h:138
request_t * request
Request this query relates to.
Definition rlm_sql.h:134
sql_rcode_t
Action to take at end of an SQL query.
Definition rlm_sql.h:44
@ RLM_SQL_ALT_QUERY
Key constraint violation, use an alternative query.
Definition rlm_sql.h:49
@ RLM_SQL_RECONNECT
Stale connection, should reconnect.
Definition rlm_sql.h:48
@ RLM_SQL_ERROR
General connection/server error.
Definition rlm_sql.h:46
@ RLM_SQL_OK
Success.
Definition rlm_sql.h:47
@ RLM_SQL_NO_MORE_ROWS
No more rows available.
Definition rlm_sql.h:50
@ SQL_QUERY_OTHER
Definition rlm_sql.h:117
#define RLM_SQL_RCODE_FLAGS_ALT_QUERY
Can distinguish between other errors and those resulting from a unique key violation.
Definition rlm_sql.h:167
rlm_sql_row_t row
Row data from the last query.
Definition rlm_sql.h:142
sql_rcode_t rcode
Result code.
Definition rlm_sql.h:141
trunk_request_t * treq
Trunk request for this query.
Definition rlm_sql.h:137
@ SQL_QUERY_FAILED
Failed to submit.
Definition rlm_sql.h:123
Definition rlm_sql.h:61
static void sql_request_complete(UNUSED request_t *request, void *preq, UNUSED void *rctx, UNUSED void *uctx)
static size_t sql_error(UNUSED TALLOC_CTX *ctx, sql_log_entry_t out[], NDEBUG_UNUSED size_t outlen, fr_sql_query_t *query_ctx)
Retrieves any errors associated with the query context.
sqlite_int64 sqlite3_int64
static int mod_load(void)
static sql_rcode_t sql_fields(char const **out[], fr_sql_query_t *query_ctx, UNUSED rlm_sql_config_t const *config)
static SQL_TRUNK_CONNECTION_ALLOC void sql_trunk_request_mux(UNUSED fr_event_list_t *el, trunk_connection_t *tconn, connection_t *conn, UNUSED void *uctx)
static void sql_print_error(sqlite3 *db, int status, char const *fmt,...))
Print an error to the global debug log.
rlm_sql_driver_t rlm_sql_sqlite
static connection_state_t _sql_connection_init(void **h, connection_t *conn, void *uctx)
char const * filename
static int sql_loadfile(TALLOC_CTX *ctx, sqlite3 *db, char const *filename)
static unlang_action_t sql_fetch_row(rlm_rcode_t *p_result, UNUSED int *priority, UNUSED request_t *request, void *uctx)
sqlite3_stmt * statement
static void _sql_connection_close(UNUSED fr_event_list_t *el, void *h, UNUSED void *uctx)
static const conf_parser_t driver_config[]
static sql_rcode_t sql_error_to_rcode(int status)
Convert an sqlite status code to an sql_rcode_t.
#define SQLITE_OPEN_NOMUTEX
#define BOOTSTRAP_MAX
static SQL_QUERY_RESUME void sql_request_fail(UNUSED request_t *request, void *preq, UNUSED void *rctx, UNUSED trunk_request_state_t state, UNUSED void *uctx)
static sql_rcode_t sql_finish_query(fr_sql_query_t *query_ctx, rlm_sql_config_t const *config)
static sql_rcode_t sql_free_result(fr_sql_query_t *query_ctx, UNUSED rlm_sql_config_t const *config)
static int sql_affected_rows(fr_sql_query_t *query_ctx, UNUSED rlm_sql_config_t const *config)
static int mod_instantiate(module_inst_ctx_t const *mctx)
static sql_rcode_t sql_check_error(sqlite3 *db, int status)
Determine if an error occurred, and what type of error it was.
static void _sql_greatest(sqlite3_context *ctx, int num_values, sqlite3_value **values)
Macros to reduce boilerplate in trunk SQL drivers.
#define SQL_QUERY_RESUME
#define SQL_TRUNK_CONNECTION_ALLOC
Allocate an SQL trunk connection.
CONF_SECTION * conf
Module's instance configuration.
Definition module.h:329
size_t inst_size
Size of the module's instance data.
Definition module.h:203
void * data
Module's instance data.
Definition module.h:271
module_instance_t const * parent
Parent module's instance (if any).
Definition module.h:337
static char buff[sizeof("18446744073709551615")+3]
Definition size_tests.c:41
eap_aka_sim_process_conf_t * inst
size_t strlcpy(char *dst, char const *src, size_t siz)
Definition strlcpy.c:34
module_t common
Common fields for all loadable modules.
Definition rlm_sql.h:197
module_instance_t * driver_submodule
Driver's submodule.
Definition rlm_sql.h:230
rlm_sql_config_t config
Definition rlm_sql.h:224
char const * fr_syserror(int num)
Guaranteed to be thread-safe version of strerror.
Definition syserror.c:243
char * talloc_typed_asprintf(TALLOC_CTX *ctx, char const *fmt,...)
Call talloc vasprintf, setting the type on the new chunk correctly.
Definition talloc.c:492
char * talloc_typed_strdup(TALLOC_CTX *ctx, char const *p)
Call talloc_strdup, setting the type on the new chunk correctly.
Definition talloc.c:445
#define talloc_get_type_abort_const
Definition talloc.h:282
static const char * names[8]
Definition time.c:628
static int64_t fr_time_delta_to_sec(fr_time_delta_t delta)
Definition time.h:647
void trunk_request_signal_fail(trunk_request_t *treq)
Signal that a trunk request failed.
Definition trunk.c:2132
int trunk_connection_pop_request(trunk_request_t **treq_out, trunk_connection_t *tconn)
Pop a request off a connection's pending queue.
Definition trunk.c:3883
void trunk_request_signal_reapable(trunk_request_t *treq)
Signal that the request was written to a connection successfully, but no response is expected.
Definition trunk.c:2072
Associates request queues with a connection.
Definition trunk.c:134
Wraps a normal request.
Definition trunk.c:100
trunk_request_state_t
Used for sanity checks and to simplify freeing.
Definition trunk.h:161
close(uq->fd)
static fr_event_list_t * el
static fr_slen_t parent
Definition pair.h:845
char const * fr_strerror(void)
Get the last library error.
Definition strerror.c:554
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
int nonnull(2, 5))
static size_t char ** out
Definition value.h:1012