GNUnet 0.25.2-11-g84e94e98c
 
Loading...
Searching...
No Matches
plugin_namecache_sqlite.c
Go to the documentation of this file.
1/*
2 * This file is part of GNUnet
3 * Copyright (C) 2009-2013 GNUnet e.V.
4 *
5 * GNUnet is free software: you can redistribute it and/or modify it
6 * under the terms of the GNU Affero General Public License as published
7 * by the Free Software Foundation, either version 3 of the License,
8 * or (at your option) any later version.
9 *
10 * GNUnet is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Affero General Public License for more details.
14 *
15 * You should have received a copy of the GNU Affero General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17
18 SPDX-License-Identifier: AGPL3.0-or-later
19 */
20
26#include "platform.h"
27#include "gnunet_sq_lib.h"
30#include <sqlite3.h>
31
42#define BUSY_TIMEOUT_MS 1000
43
44
50#define LOG_SQLITE(db, level, cmd) do { GNUNET_log_from (level, \
51 "namecache-sqlite", _ ( \
52 "`%s' failed at %s:%d with error: %s\n"), \
53 cmd, \
54 __FILE__, __LINE__, \
55 sqlite3_errmsg ( \
56 db->dbh)); \
57} while (0)
58
59#define LOG(kind, ...) GNUNET_log_from (kind, "namecache-sqlite", __VA_ARGS__)
60
61
65struct Plugin
66{
67 const struct GNUNET_CONFIGURATION_Handle *cfg;
68
72 char *fn;
73
77 sqlite3 *dbh;
78
82 sqlite3_stmt *cache_block;
83
87 sqlite3_stmt *delete_block;
88
92 sqlite3_stmt *lookup_block;
93
97 sqlite3_stmt *expire_blocks;
98};
99
100
109static int
111{
112 struct GNUNET_SQ_ExecuteStatement es[] = {
113 GNUNET_SQ_make_try_execute ("PRAGMA temp_store=MEMORY"),
114 GNUNET_SQ_make_try_execute ("PRAGMA synchronous=NORMAL"),
115 GNUNET_SQ_make_try_execute ("PRAGMA legacy_file_format=OFF"),
116 GNUNET_SQ_make_try_execute ("PRAGMA auto_vacuum=INCREMENTAL"),
117 GNUNET_SQ_make_try_execute ("PRAGMA encoding=\"UTF-8\""),
118 GNUNET_SQ_make_try_execute ("PRAGMA locking_mode=EXCLUSIVE"),
119 GNUNET_SQ_make_try_execute ("PRAGMA page_size=4092"),
120 GNUNET_SQ_make_try_execute ("PRAGMA journal_mode=WAL"),
121 GNUNET_SQ_make_execute ("CREATE TABLE IF NOT EXISTS ns096blocks ("
122 " query BLOB NOT NULL,"
123 " block BLOB NOT NULL,"
124 " expiration_time INT8 NOT NULL"
125 ")"),
126 GNUNET_SQ_make_execute ("CREATE INDEX IF NOT EXISTS ir_query_hash "
127 "ON ns096blocks (query,expiration_time)"),
128 GNUNET_SQ_make_execute ("CREATE INDEX IF NOT EXISTS ir_block_expiration "
129 "ON ns096blocks (expiration_time)"),
131 };
132 struct GNUNET_SQ_PrepareStatement ps[] = {
134 "INSERT INTO ns096blocks (query,block,expiration_time) VALUES (?, ?, ?)",
135 &plugin->cache_block),
136 GNUNET_SQ_make_prepare ("DELETE FROM ns096blocks WHERE expiration_time<?",
137 &plugin->expire_blocks),
139 "DELETE FROM ns096blocks WHERE query=? AND expiration_time<=?",
140 &plugin->delete_block),
141 GNUNET_SQ_make_prepare ("SELECT block FROM ns096blocks WHERE query=? "
142 "ORDER BY expiration_time DESC LIMIT 1",
143 &plugin->lookup_block),
145 };
146 char *afsdir;
147
148 if (GNUNET_OK !=
150 "namecache-sqlite",
151 "FILENAME",
152 &afsdir))
153 {
155 "namecache-sqlite",
156 "FILENAME");
157 return GNUNET_SYSERR;
158 }
159 if (GNUNET_OK !=
160 GNUNET_DISK_file_test (afsdir))
161 {
162 if (GNUNET_OK !=
164 {
165 GNUNET_break (0);
166 GNUNET_free (afsdir);
167 return GNUNET_SYSERR;
168 }
169 }
170 /* afsdir should be UTF-8-encoded. If it isn't, it's a bug */
171 plugin->fn = afsdir;
172
173 /* Open database and precompile statements */
174 if (SQLITE_OK !=
175 sqlite3_open (plugin->fn, &plugin->dbh))
176 {
178 _ ("Unable to initialize SQLite: %s.\n"),
179 sqlite3_errmsg (plugin->dbh));
180 return GNUNET_SYSERR;
181 }
182 if (GNUNET_OK !=
184 es))
185 {
186 GNUNET_break (0);
188 _ ("Failed to setup database at `%s'\n"),
189 plugin->fn);
190 return GNUNET_SYSERR;
191 }
192 GNUNET_break (SQLITE_OK ==
193 sqlite3_busy_timeout (plugin->dbh,
195
196 if (GNUNET_OK !=
198 ps))
199 {
200 GNUNET_break (0);
202 _ ("Failed to setup database at `%s'\n"),
203 plugin->fn);
204 return GNUNET_SYSERR;
205 }
206
207 return GNUNET_OK;
208}
209
210
216static void
218{
219 int result;
220 sqlite3_stmt *stmt;
221
222 if (NULL != plugin->cache_block)
223 sqlite3_finalize (plugin->cache_block);
224 if (NULL != plugin->lookup_block)
225 sqlite3_finalize (plugin->lookup_block);
226 if (NULL != plugin->expire_blocks)
227 sqlite3_finalize (plugin->expire_blocks);
228 if (NULL != plugin->delete_block)
229 sqlite3_finalize (plugin->delete_block);
230 result = sqlite3_close (plugin->dbh);
231 if (result == SQLITE_BUSY)
232 {
234 _ (
235 "Tried to close sqlite without finalizing all prepared statements.\n"));
236 stmt = sqlite3_next_stmt (plugin->dbh,
237 NULL);
238 while (stmt != NULL)
239 {
241 "sqlite",
242 "Closing statement %p\n",
243 stmt);
244 result = sqlite3_finalize (stmt);
245 if (result != SQLITE_OK)
247 "sqlite",
248 "Failed to close statement %p: %d\n",
249 stmt,
250 result);
251 stmt = sqlite3_next_stmt (plugin->dbh,
252 NULL);
253 }
254 result = sqlite3_close (plugin->dbh);
255 }
256 if (SQLITE_OK != result)
259 "sqlite3_close");
260
261 GNUNET_free (plugin->fn);
262}
263
264
270static void
272{
274 struct GNUNET_SQ_QueryParam params[] = {
277 };
278 int n;
279
280 if (GNUNET_OK !=
281 GNUNET_SQ_bind (plugin->expire_blocks,
282 params))
283 {
286 "sqlite3_bind_XXXX");
288 plugin->expire_blocks);
289 return;
290 }
291 n = sqlite3_step (plugin->expire_blocks);
293 plugin->expire_blocks);
294 switch (n)
295 {
296 case SQLITE_DONE:
298 "sqlite",
299 "Records expired\n");
300 return;
301
302 case SQLITE_BUSY:
305 "sqlite3_step");
306 return;
307
308 default:
311 "sqlite3_step");
312 return;
313 }
314}
315
316
324static int
326 const struct GNUNET_GNSRECORD_Block *block)
327{
328 static struct GNUNET_TIME_Absolute last_expire;
329 struct Plugin *plugin = cls;
330 struct GNUNET_HashCode query;
332 size_t block_size = GNUNET_GNSRECORD_block_get_size (block);
333 int n;
334
335 /* run expiration of old cache entries once per hour */
336 if (GNUNET_TIME_absolute_get_duration (last_expire).rel_value_us >
338 {
339 last_expire = GNUNET_TIME_absolute_get ();
341 }
343 GNUNET_GNSRECORD_query_from_block (block, &query));
346 "Caching new version of block %s (expires %s)\n",
347 GNUNET_h2s (&query),
349 if (block_size > 64 * 65536)
350 {
351 GNUNET_break (0);
352 return GNUNET_SYSERR;
353 }
354
355 {
356 struct GNUNET_SQ_QueryParam del_params[] = {
360 };
361 /* delete old version of the block */
362 if (GNUNET_OK !=
363 GNUNET_SQ_bind (plugin->delete_block,
364 del_params))
365 {
367 "sqlite3_bind_XXXX");
369 plugin->delete_block);
370 return GNUNET_SYSERR;
371 }
372 }
373 n = sqlite3_step (plugin->delete_block);
374 switch (n)
375 {
376 case SQLITE_DONE:
378 "sqlite",
379 "Old block deleted\n");
380 break;
381
382 case SQLITE_BUSY:
385 "sqlite3_step");
386 break;
387
388 default:
391 "sqlite3_step");
392 break;
393 }
395 plugin->delete_block);
396 {
397 struct GNUNET_SQ_QueryParam ins_params[] = {
400 block_size),
403 };
404
405 /* insert new version of the block */
406 if (GNUNET_OK !=
407 GNUNET_SQ_bind (plugin->cache_block,
408 ins_params))
409 {
412 "sqlite3_bind_XXXX");
414 plugin->cache_block);
415 return GNUNET_SYSERR;
416 }
417 }
419 "Caching block under derived key `%s'\n",
420 GNUNET_h2s_full (&query));
421 n = sqlite3_step (plugin->cache_block);
423 plugin->cache_block);
424 switch (n)
425 {
426 case SQLITE_DONE:
428 "Record stored\n");
429 return GNUNET_OK;
430
431 case SQLITE_BUSY:
434 "sqlite3_step");
435 return GNUNET_NO;
436
437 default:
440 "sqlite3_step");
441 return GNUNET_SYSERR;
442 }
443}
444
445
456static int
458 const struct GNUNET_HashCode *query,
460 void *iter_cls)
461{
462 struct Plugin *plugin = cls;
463 int ret;
464 int sret;
465 size_t block_size;
466 const struct GNUNET_GNSRECORD_Block *block;
467 struct GNUNET_SQ_QueryParam params[] = {
470 };
471 struct GNUNET_SQ_ResultSpec rs[] = {
472 GNUNET_SQ_result_spec_variable_size ((void **) &block,
473 &block_size),
475 };
476
477 if (GNUNET_OK !=
478 GNUNET_SQ_bind (plugin->lookup_block,
479 params))
480 {
483 "sqlite3_bind_XXXX");
485 plugin->lookup_block);
486 return GNUNET_SYSERR;
487 }
488 ret = GNUNET_NO;
489 if (SQLITE_ROW ==
490 (sret = sqlite3_step (plugin->lookup_block)))
491 {
492 if (GNUNET_OK !=
493 GNUNET_SQ_extract_result (plugin->lookup_block,
494 rs))
495 {
496 GNUNET_break (0);
498 }
499 else if ((block_size < sizeof(struct GNUNET_GNSRECORD_Block)))
500 {
501 GNUNET_break (0);
504 }
505 else
506 {
508 "Found block under derived key `%s'\n",
509 GNUNET_h2s_full (query));
510 iter (iter_cls,
511 block);
513 ret = GNUNET_YES;
514 }
515 }
516 else
517 {
518 if (SQLITE_DONE != sret)
519 {
522 "sqlite_step");
524 }
525 else
526 {
528 "No block found under derived key `%s'\n",
529 GNUNET_h2s_full (query));
530 }
531 }
533 plugin->lookup_block);
534 return ret;
535}
536
537
538void *
540
547void *
549{
550 static struct Plugin plugin;
551 const struct GNUNET_CONFIGURATION_Handle *cfg = cls;
553
554 if (NULL != plugin.cfg)
555 return NULL; /* can only initialize once! */
556 memset (&plugin, 0, sizeof(struct Plugin));
557 plugin.cfg = cfg;
559 {
561 return NULL;
562 }
564 api->cls = &plugin;
568 _ ("Sqlite database running\n"));
569 return api;
570}
571
572
573void *
575
582void *
584{
586 struct Plugin *plugin = api->cls;
587
589 plugin->cfg = NULL;
592 "sqlite plugin is finished\n");
593 return NULL;
594}
595
596
597/* end of plugin_namecache_sqlite.c */
static int ret
Final status code.
Definition gnunet-arm.c:93
static struct GNUNET_CONFIGURATION_Handle * cfg
Our configuration.
Definition gnunet-arm.c:108
static struct GNUNET_TESTING_PluginFunctions * plugin
Plugin to dynamically load a test case.
static struct GNUNET_PEERSTORE_Handle * ps
Handle to the PEERSTORE service.
static struct GNUNET_TIME_Relative expiration
User supplied expiration value.
static int result
Global testing status.
API that can be used to manipulate GNS record data.
Plugin API for the namecache database backend.
helper functions for Sqlite3 DB interactions
struct GNUNET_SQ_PrepareStatement GNUNET_SQ_make_prepare(const char *sql, sqlite3_stmt **pstmt)
Create a struct GNUNET_SQ_PrepareStatement
Definition sq_prepare.c:37
struct GNUNET_SQ_ExecuteStatement GNUNET_SQ_make_execute(const char *sql)
Create a struct GNUNET_SQ_ExecuteStatement where errors are fatal.
Definition sq_exec.c:36
struct GNUNET_SQ_ExecuteStatement GNUNET_SQ_make_try_execute(const char *sql)
Create a struct GNUNET_SQ_ExecuteStatement where errors should be tolerated.
Definition sq_exec.c:55
struct GNUNET_SQ_QueryParam GNUNET_SQ_query_param_fixed_size(const void *ptr, size_t ptr_size)
Generate query parameter for a buffer ptr of ptr_size bytes.
#define GNUNET_SQ_PREPARE_END
Terminator for executable statement list.
enum GNUNET_GenericReturnValue GNUNET_SQ_extract_result(sqlite3_stmt *result, struct GNUNET_SQ_ResultSpec *rs)
Extract results from a query result according to the given specification.
Definition sq.c:75
void GNUNET_SQ_cleanup_result(struct GNUNET_SQ_ResultSpec *rs)
Free all memory that was allocated in rs during GNUNET_SQ_extract_result().
Definition sq.c:104
enum GNUNET_GenericReturnValue GNUNET_SQ_exec_statements(sqlite3 *dbh, const struct GNUNET_SQ_ExecuteStatement *es)
Request execution of an array of statements es from Postgres.
Definition sq_exec.c:76
enum GNUNET_GenericReturnValue GNUNET_SQ_prepare(sqlite3 *dbh, const struct GNUNET_SQ_PrepareStatement *ps)
Prepare all statements given in the (NULL,NULL)-terminated array at ps.
Definition sq_prepare.c:50
#define GNUNET_SQ_EXECUTE_STATEMENT_END
Terminator for executable statement list.
struct GNUNET_SQ_QueryParam GNUNET_SQ_query_param_absolute_time(const struct GNUNET_TIME_Absolute *x)
Generate query parameter for an absolute time value.
#define GNUNET_SQ_result_spec_end
End of result parameter specification.
struct GNUNET_SQ_ResultSpec GNUNET_SQ_result_spec_variable_size(void **dst, size_t *sptr)
Variable-size result expected.
#define GNUNET_SQ_query_param_auto_from_type(x)
Generate fixed-size query parameter with size determined by variable type.
enum GNUNET_GenericReturnValue GNUNET_SQ_bind(sqlite3_stmt *stmt, const struct GNUNET_SQ_QueryParam *params)
Execute binding operations for a prepared statement.
Definition sq.c:30
void GNUNET_SQ_reset(sqlite3 *dbh, sqlite3_stmt *stmt)
Reset stmt and log error.
Definition sq.c:119
#define GNUNET_SQ_query_param_end
End of query parameter specification.
enum GNUNET_GenericReturnValue GNUNET_CONFIGURATION_get_value_filename(const struct GNUNET_CONFIGURATION_Handle *cfg, const char *section, const char *option, char **value)
Get a configuration value that should be the name of a file or directory.
enum GNUNET_GenericReturnValue GNUNET_DISK_file_test(const char *fil)
Check that fil corresponds to a filename (of a file that exists and that is not a directory).
Definition disk.c:533
enum GNUNET_GenericReturnValue GNUNET_DISK_directory_create_for_file(const char *filename)
Create the directory structure for storing a file.
Definition disk.c:633
size_t GNUNET_GNSRECORD_block_get_size(const struct GNUNET_GNSRECORD_Block *block)
Returns the length of this block in bytes.
enum GNUNET_GenericReturnValue GNUNET_GNSRECORD_query_from_block(const struct GNUNET_GNSRECORD_Block *block, struct GNUNET_HashCode *query)
Builds the query hash from a block.
struct GNUNET_TIME_Absolute GNUNET_GNSRECORD_block_get_expiration(const struct GNUNET_GNSRECORD_Block *block)
Returns the expiration of a block.
#define GNUNET_log(kind,...)
#define GNUNET_log_from(kind, comp,...)
@ GNUNET_OK
@ GNUNET_YES
@ GNUNET_NO
@ GNUNET_SYSERR
const char * GNUNET_h2s_full(const struct GNUNET_HashCode *hc)
Convert a hash value to a string (for printing debug messages).
#define GNUNET_assert(cond)
Use this for fatal errors that cannot be handled.
#define GNUNET_break(cond)
Use this for internal assertion violations that are not fatal (can be handled) but should not occur.
const char * GNUNET_h2s(const struct GNUNET_HashCode *hc)
Convert a hash value to a string (for printing debug messages).
void GNUNET_log_config_missing(enum GNUNET_ErrorType kind, const char *section, const char *option)
Log error message about missing configuration option.
@ GNUNET_ERROR_TYPE_WARNING
@ GNUNET_ERROR_TYPE_ERROR
@ GNUNET_ERROR_TYPE_BULK
@ GNUNET_ERROR_TYPE_DEBUG
@ GNUNET_ERROR_TYPE_INFO
#define GNUNET_new(type)
Allocate a struct or union of the given type.
#define GNUNET_free(ptr)
Wrapper around free.
void(* GNUNET_NAMECACHE_BlockCallback)(void *cls, const struct GNUNET_GNSRECORD_Block *block)
Function called for matching blocks.
struct GNUNET_TIME_Relative GNUNET_TIME_absolute_get_duration(struct GNUNET_TIME_Absolute whence)
Get the duration of an operation as the difference of the current time and the given start time "henc...
Definition time.c:438
#define GNUNET_TIME_UNIT_HOURS
One hour.
struct GNUNET_TIME_Absolute GNUNET_TIME_absolute_get(void)
Get the current time.
Definition time.c:111
const char * GNUNET_STRINGS_absolute_time_to_string(struct GNUNET_TIME_Absolute t)
Like asctime, except for GNUnet time.
Definition strings.c:660
#define _(String)
GNU gettext support macro.
Definition platform.h:179
#define BUSY_TIMEOUT_MS
After how many ms "busy" should a DB operation fail for good? A low value makes sure that we are more...
static void namecache_sqlite_expire_blocks(struct Plugin *plugin)
Removes any expired block.
#define LOG_SQLITE(db, level, cmd)
Log an error message at log-level 'level' that indicates a failure of the command 'cmd' on file 'file...
void * libgnunet_plugin_namecache_sqlite_init(void *cls)
Entry point for the plugin.
void * libgnunet_plugin_namecache_sqlite_done(void *cls)
Exit point from the plugin.
static void database_shutdown(struct Plugin *plugin)
Shutdown database connection and associate data structures.
static int namecache_sqlite_cache_block(void *cls, const struct GNUNET_GNSRECORD_Block *block)
Cache a block in the datastore.
static int database_setup(struct Plugin *plugin)
Initialize the database connections and associated data structures (create tables and indices as need...
#define LOG(kind,...)
static int namecache_sqlite_lookup_block(void *cls, const struct GNUNET_HashCode *query, GNUNET_NAMECACHE_BlockCallback iter, void *iter_cls)
Get the block for a particular zone and label in the datastore.
void * cls
Closure for all of the callbacks.
A 512-bit hashcode.
struct returned by the initialization function of the plugin
int(* lookup_block)(void *cls, const struct GNUNET_HashCode *query, GNUNET_NAMECACHE_BlockCallback iter, void *iter_cls)
Get the block for a particular zone and label in the datastore.
int(* cache_block)(void *cls, const struct GNUNET_GNSRECORD_Block *block)
Cache a block in the datastore.
void * cls
Closure to pass to all plugin functions.
Information needed to run a list of SQL statements using GNUNET_SQ_exec_statements().
Information needed to run a list of SQL statements using GNUNET_SQ_exec_statements().
Description of a DB query parameter.
Description of a DB result cell.
void * cls
Closure for conv and cleaner.
Time for absolute times used by GNUnet, in microseconds.
uint64_t rel_value_us
The actual value.
Handle for a plugin.
Definition block.c:38
struct GNUNET_BLOCK_PluginFunctions * api
Plugin API.
Definition block.c:47
sqlite3_stmt * cache_block
Precompiled SQL for caching a block.
sqlite3_stmt * lookup_block
Precompiled SQL for looking up a block.
sqlite3_stmt * delete_block
Precompiled SQL for deleting an older block.
char * fn
Filename used for the DB.
sqlite3_stmt * expire_blocks
Precompiled SQL for removing expired blocks.
struct GNUNET_PQ_Context * dbh
Native Postgres database handle.
const struct GNUNET_CONFIGURATION_Handle * cfg
Our configuration.