1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-05 12:31:50 +01:00

kbx: Change keyboxd to work only with one database.

* kbx/frontend.c (the_database): New var.
(db_desc_t): Remove.
(kbxd_add_resource): Renamed to ...
(kbxd_set_database): this.  Simplify.
(kbxd_search): Change to use only one database.
(kbxd_store): Ditto.
(kbxd_delete): Ditto.
--

The original implementation was way to complicated and would have only
brought back the problems deciding which database to use for each key.
The new scheme used one configured database and only that.  That
database needs to be set right at the start.

Signed-off-by: Werner Koch <wk@gnupg.org>
This commit is contained in:
Werner Koch 2019-12-10 09:35:33 +01:00
parent b2a2df174b
commit 8a556c23a2
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B
3 changed files with 110 additions and 160 deletions

View File

@ -32,19 +32,12 @@
#include "frontend.h" #include "frontend.h"
/* An object to describe a single database. */ /* An object to keep infos about the database. */
struct db_desc_s struct
{ {
enum database_types db_type; enum database_types db_type;
backend_handle_t backend_handle; backend_handle_t backend_handle;
}; } the_database;
typedef struct db_desc_s *db_desc_t;
/* The table of databases and the size of that table. */
static db_desc_t databases;
static unsigned int no_of_databases;
@ -76,28 +69,23 @@ release_lock (ctrl_t ctrl)
} }
/* Add a new resource to the database. Depending on the FILENAME /* Set the database to use. Depending on the FILENAME suffix we
* suffix we decide which one to use. This function must be called at * decide which one to use. This function must be called at daemon
* daemon startup because it employs no locking. If FILENAME has no * startup because it employs no locking. If FILENAME has no
* directory separator, the file is expected or created below * directory separator, the file is expected or created below
* "$GNUPGHOME/public-keys-v1.d/". In READONLY mode the file must * "$GNUPGHOME/public-keys.d/". In READONLY mode the file must exists;
* exists; otherwise it is created. */ * otherwise it is created. */
gpg_error_t gpg_error_t
kbxd_add_resource (ctrl_t ctrl, const char *filename_arg, int readonly) kbxd_set_database (ctrl_t ctrl, const char *filename_arg, int readonly)
{ {
gpg_error_t err; gpg_error_t err;
char *filename; char *filename;
enum database_types db_type = 0; enum database_types db_type = 0;
backend_handle_t handle = NULL; backend_handle_t handle = NULL;
unsigned int n, dbidx; unsigned int n;
/* Do tilde expansion etc. */ /* Do tilde expansion etc. */
if (!strcmp (filename_arg, "[cache]")) if (strchr (filename_arg, DIRSEP_C)
{
filename = xstrdup (filename_arg);
db_type = DB_TYPE_CACHE;
}
else if (strchr (filename_arg, DIRSEP_C)
#ifdef HAVE_W32_SYSTEM #ifdef HAVE_W32_SYSTEM
|| strchr (filename_arg, '/') /* Windows also accepts a slash. */ || strchr (filename_arg, '/') /* Windows also accepts a slash. */
#endif #endif
@ -110,14 +98,18 @@ kbxd_add_resource (ctrl_t ctrl, const char *filename_arg, int readonly)
/* If this is the first call to the function and the request is not /* If this is the first call to the function and the request is not
* for the cache backend, add the cache backend so that it will * for the cache backend, add the cache backend so that it will
* always be the first to be queried. */ * always be the first to be queried. */
if (!no_of_databases && !db_type) if (the_database.db_type)
{ {
err = be_cache_initialize (); log_error ("error: only one database allowed\n");
/* err = kbxd_add_resource (ctrl, "[cache]", 0); */ err = gpg_error (GPG_ERR_CONFLICT);
if (err)
goto leave; goto leave;
} }
/* Init the cache. */
err = be_cache_initialize ();
if (err)
goto leave;
n = strlen (filename); n = strlen (filename);
if (db_type) if (db_type)
; /* We already know it. */ ; /* We already know it. */
@ -147,56 +139,14 @@ kbxd_add_resource (ctrl_t ctrl, const char *filename_arg, int readonly)
if (err) if (err)
goto leave; goto leave;
/* All good, create an entry in the table. */ the_database.db_type = db_type;
for (dbidx = 0; dbidx < no_of_databases; dbidx++) the_database.backend_handle = handle;
if (!databases[dbidx].db_type)
break;
if (dbidx == no_of_databases)
{
/* No table yet or table is full. */
if (!databases)
{
/* Create first set of databases. Note that the initial
* type for all entries is DB_TYPE_NONE. */
dbidx = 4;
databases = xtrycalloc (dbidx, sizeof *databases);
if (!databases)
{
err = gpg_error_from_syserror ();
goto leave;
}
no_of_databases = dbidx;
dbidx = 0; /* Put into first slot. */
}
else
{
db_desc_t newdb;
dbidx = no_of_databases + (no_of_databases == 4? 12 : 16);
newdb = xtrycalloc (dbidx, sizeof *databases);
if (!databases)
{
err = gpg_error_from_syserror ();
goto leave;
}
for (n=0; n < no_of_databases; n++)
newdb[n] = databases[n];
xfree (databases);
databases = newdb;
n = no_of_databases;
no_of_databases = dbidx;
dbidx = n; /* Put into first new slot. */
}
}
databases[dbidx].db_type = db_type;
databases[dbidx].backend_handle = handle;
handle = NULL; handle = NULL;
leave: leave:
if (err) if (err)
{ {
log_error ("error adding resource '%s': %s\n", log_error ("error setting database '%s': %s\n",
filename, gpg_strerror (err)); filename, gpg_strerror (err));
be_generic_release_backend (ctrl, handle); be_generic_release_backend (ctrl, handle);
} }
@ -227,8 +177,6 @@ kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc,
{ {
gpg_error_t err; gpg_error_t err;
int i; int i;
unsigned int dbidx;
db_desc_t db;
db_request_t request; db_request_t request;
int start_at_ubid = 0; int start_at_ubid = 0;
@ -260,27 +208,30 @@ kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc,
} }
request = ctrl->opgp_req; request = ctrl->opgp_req;
if (!the_database.db_type)
{
log_error ("%s: error: no database configured\n", __func__);
err = gpg_error (GPG_ERR_NOT_INITIALIZED);
goto leave;
}
/* If requested do a reset. Using the reset flag is faster than /* If requested do a reset. Using the reset flag is faster than
* letting the caller do a separate call for an intial reset. */ * letting the caller do a separate call for an intial reset. */
if (!desc || reset) if (!desc || reset)
{ {
for (dbidx=0; dbidx < no_of_databases; dbidx++) switch (the_database.db_type)
{ {
db = databases + dbidx;
if (!db->db_type)
continue; /* Empty slot. */
switch (db->db_type)
{
case DB_TYPE_NONE: /* NOTREACHED */
break;
case DB_TYPE_CACHE: case DB_TYPE_CACHE:
err = 0; /* Nothing to do. */ err = 0; /* Nothing to do. */
break; break;
case DB_TYPE_KBX: case DB_TYPE_KBX:
err = be_kbx_search (ctrl, db->backend_handle, request, NULL, 0); err = be_kbx_search (ctrl, the_database.backend_handle,
request, NULL, 0);
break;
default:
err = gpg_error (GPG_ERR_INTERNAL);
break; break;
} }
if (err) if (err)
@ -289,7 +240,6 @@ kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc,
reset? "initial ":"", gpg_strerror (err)); reset? "initial ":"", gpg_strerror (err));
goto leave; goto leave;
} }
}
request->any_search = 0; request->any_search = 0;
request->any_found = 0; request->any_found = 0;
request->next_dbidx = 0; request->next_dbidx = 0;
@ -300,34 +250,11 @@ kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc,
} }
} }
/* Move to the next non-empty slot. */
next_db:
for (dbidx=request->next_dbidx; (dbidx < no_of_databases
&& !databases[dbidx].db_type); dbidx++)
;
request->next_dbidx = dbidx;
if (!(dbidx < no_of_databases))
{
/* All databases have been searched. Put the non-found mark
* into the cache for all descriptors.
* FIXME: We need to see which pubkey type we need to insert. */
be_cache_not_found (ctrl, PUBKEY_TYPE_UNKNOWN, desc, ndesc);
err = gpg_error (GPG_ERR_NOT_FOUND);
goto leave;
}
db = databases + dbidx;
/* Divert to the backend for the actual search. */ /* Divert to the backend for the actual search. */
switch (db->db_type) switch (the_database.db_type)
{ {
case DB_TYPE_NONE:
/* NOTREACHED */
err = gpg_error (GPG_ERR_INTERNAL);
break;
case DB_TYPE_CACHE: case DB_TYPE_CACHE:
err = be_cache_search (ctrl, db->backend_handle, request, err = be_cache_search (ctrl, the_database.backend_handle, request,
desc, ndesc); desc, ndesc);
/* Expected error codes from the cache lookup are: /* Expected error codes from the cache lookup are:
* 0 - found and returned via the cache * 0 - found and returned via the cache
@ -339,26 +266,31 @@ kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc,
if (start_at_ubid) if (start_at_ubid)
{ {
/* We need to set the startpoint for the search. */ /* We need to set the startpoint for the search. */
err = be_kbx_seek (ctrl, db->backend_handle, request, err = be_kbx_seek (ctrl, the_database.backend_handle, request,
request->last_cached_ubid); request->last_cached_ubid);
if (err) if (err)
{ {
log_debug ("%s: seeking %s to an UBID failed: %s\n", log_debug ("%s: seeking %s to an UBID failed: %s\n", __func__,
__func__, strdbtype (db->db_type), gpg_strerror (err)); strdbtype (the_database.db_type), gpg_strerror (err));
break; break;
} }
} }
err = be_kbx_search (ctrl, db->backend_handle, request, err = be_kbx_search (ctrl, the_database.backend_handle, request,
desc, ndesc); desc, ndesc);
if (start_at_ubid && gpg_err_code (err) == GPG_ERR_EOF) if (start_at_ubid && gpg_err_code (err) == GPG_ERR_EOF)
be_cache_mark_final (ctrl, request); be_cache_mark_final (ctrl, request);
break; break;
default:
log_error ("%s: unsupported database type %d\n",
__func__, the_database.db_type);
err = gpg_error (GPG_ERR_INTERNAL);
break;
} }
if (DBG_LOOKUP) if (DBG_LOOKUP)
log_debug ("%s: searched %s (db %u of %u) => %s\n", log_debug ("%s: searched %s => %s\n", __func__,
__func__, strdbtype (db->db_type), dbidx, no_of_databases, strdbtype (the_database.db_type), gpg_strerror (err));
gpg_strerror (err));
request->any_search = 1; request->any_search = 1;
start_at_ubid = 0; start_at_ubid = 0;
if (!err) if (!err)
@ -367,14 +299,17 @@ kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc,
} }
else if (gpg_err_code (err) == GPG_ERR_EOF) else if (gpg_err_code (err) == GPG_ERR_EOF)
{ {
if (db->db_type == DB_TYPE_CACHE && request->last_cached_valid) if (the_database.db_type == DB_TYPE_CACHE && request->last_cached_valid)
{ {
if (request->last_cached_final) if (request->last_cached_final)
goto leave; goto leave;
start_at_ubid = 1; start_at_ubid = 1;
} }
request->next_dbidx++; request->next_dbidx++;
goto next_db; /* FIXME: We need to see which pubkey type we need to insert. */
be_cache_not_found (ctrl, PUBKEY_TYPE_UNKNOWN, desc, ndesc);
err = gpg_error (GPG_ERR_NOT_FOUND);
goto leave;
} }
@ -395,8 +330,6 @@ kbxd_store (ctrl_t ctrl, const void *blob, size_t bloblen,
{ {
gpg_error_t err; gpg_error_t err;
db_request_t request; db_request_t request;
unsigned int dbidx;
db_desc_t db;
char ubid[UBID_LEN]; char ubid[UBID_LEN];
enum pubkey_types pktype; enum pubkey_types pktype;
int insert = 0; int insert = 0;
@ -418,23 +351,29 @@ kbxd_store (ctrl_t ctrl, const void *blob, size_t bloblen,
} }
request = ctrl->opgp_req; request = ctrl->opgp_req;
if (!the_database.db_type)
{
log_error ("%s: error: no database configured\n", __func__);
err = gpg_error (GPG_ERR_NOT_INITIALIZED);
goto leave;
}
/* Check whether to insert or update. */ /* Check whether to insert or update. */
err = be_ubid_from_blob (blob, bloblen, &pktype, ubid); err = be_ubid_from_blob (blob, bloblen, &pktype, ubid);
if (err) if (err)
goto leave; goto leave;
/* FIXME: We force the use of the KBX backend. */ if (the_database.db_type == DB_TYPE_KBX)
for (dbidx=0; dbidx < no_of_databases; dbidx++)
if (databases[dbidx].db_type == DB_TYPE_KBX)
break;
if (!(dbidx < no_of_databases))
{ {
err = gpg_error (GPG_ERR_NOT_INITIALIZED); err = be_kbx_seek (ctrl, the_database.backend_handle, request, ubid);
goto leave; }
else
{
log_error ("%s: unsupported database type %d\n",
__func__, the_database.db_type);
err = gpg_error (GPG_ERR_INTERNAL);
} }
db = databases + dbidx;
err = be_kbx_seek (ctrl, db->backend_handle, request, ubid);
if (!err) if (!err)
; /* Found - need to update. */ ; /* Found - need to update. */
else if (gpg_err_code (err) == GPG_ERR_EOF) else if (gpg_err_code (err) == GPG_ERR_EOF)
@ -450,17 +389,21 @@ kbxd_store (ctrl_t ctrl, const void *blob, size_t bloblen,
{ {
if (mode == KBXD_STORE_UPDATE) if (mode == KBXD_STORE_UPDATE)
err = gpg_error (GPG_ERR_CONFLICT); err = gpg_error (GPG_ERR_CONFLICT);
else else if (the_database.db_type == DB_TYPE_KBX)
err = be_kbx_insert (ctrl, db->backend_handle, request, err = be_kbx_insert (ctrl, the_database.backend_handle, request,
pktype, blob, bloblen); pktype, blob, bloblen);
else
err = gpg_error (GPG_ERR_INTERNAL);
} }
else /* Update. */ else /* Update. */
{ {
if (mode == KBXD_STORE_INSERT) if (mode == KBXD_STORE_INSERT)
err = gpg_error (GPG_ERR_CONFLICT); err = gpg_error (GPG_ERR_CONFLICT);
else else if (the_database.db_type == DB_TYPE_KBX)
err = be_kbx_update (ctrl, db->backend_handle, request, err = be_kbx_update (ctrl, the_database.backend_handle, request,
pktype, blob, bloblen); pktype, blob, bloblen);
else
err = gpg_error (GPG_ERR_INTERNAL);
} }
leave: leave:
@ -479,8 +422,6 @@ kbxd_delete (ctrl_t ctrl, const unsigned char *ubid)
{ {
gpg_error_t err; gpg_error_t err;
db_request_t request; db_request_t request;
unsigned int dbidx;
db_desc_t db;
if (DBG_CLOCK) if (DBG_CLOCK)
log_clock ("%s: enter", __func__); log_clock ("%s: enter", __func__);
@ -499,18 +440,24 @@ kbxd_delete (ctrl_t ctrl, const unsigned char *ubid)
} }
request = ctrl->opgp_req; request = ctrl->opgp_req;
/* FIXME: We force the use of the KBX backend. */ if (!the_database.db_type)
for (dbidx=0; dbidx < no_of_databases; dbidx++)
if (databases[dbidx].db_type == DB_TYPE_KBX)
break;
if (!(dbidx < no_of_databases))
{ {
log_error ("%s: error: no database configured\n", __func__);
err = gpg_error (GPG_ERR_NOT_INITIALIZED); err = gpg_error (GPG_ERR_NOT_INITIALIZED);
goto leave; goto leave;
} }
db = databases + dbidx;
err = be_kbx_seek (ctrl, db->backend_handle, request, ubid); if (the_database.db_type == DB_TYPE_KBX)
{
err = be_kbx_seek (ctrl, the_database.backend_handle, request, ubid);
}
else
{
log_error ("%s: unsupported database type %d\n",
__func__, the_database.db_type);
err = gpg_error (GPG_ERR_INTERNAL);
}
if (!err) if (!err)
; /* Found - we can delete. */ ; /* Found - we can delete. */
else if (gpg_err_code (err) == GPG_ERR_EOF) else if (gpg_err_code (err) == GPG_ERR_EOF)
@ -525,7 +472,10 @@ kbxd_delete (ctrl_t ctrl, const unsigned char *ubid)
goto leave; goto leave;
} }
err = be_kbx_delete (ctrl, db->backend_handle, request); if (the_database.db_type == DB_TYPE_KBX)
err = be_kbx_delete (ctrl, the_database.backend_handle, request);
else
err = gpg_error (GPG_ERR_INTERNAL);
leave: leave:
release_lock (ctrl); release_lock (ctrl);

View File

@ -31,7 +31,7 @@ enum kbxd_store_modes
}; };
gpg_error_t kbxd_add_resource (ctrl_t ctrl, gpg_error_t kbxd_set_database (ctrl_t ctrl,
const char *filename_arg, int readonly); const char *filename_arg, int readonly);
void kbxd_release_session_info (ctrl_t ctrl); void kbxd_release_session_info (ctrl_t ctrl);

View File

@ -736,7 +736,7 @@ main (int argc, char **argv )
} }
kbxd_init_default_ctrl (ctrl); kbxd_init_default_ctrl (ctrl);
kbxd_add_resource (ctrl, "pubring.kbx", 0); kbxd_set_database (ctrl, "pubring.kbx", 0);
kbxd_start_command_handler (ctrl, GNUPG_INVALID_FD, 0); kbxd_start_command_handler (ctrl, GNUPG_INVALID_FD, 0);
kbxd_deinit_default_ctrl (ctrl); kbxd_deinit_default_ctrl (ctrl);
@ -870,7 +870,7 @@ main (int argc, char **argv )
kbxd_exit (1); kbxd_exit (1);
} }
kbxd_init_default_ctrl (ctrl); kbxd_init_default_ctrl (ctrl);
kbxd_add_resource (ctrl, "pubring.kbx", 0); kbxd_set_database (ctrl, "pubring.kbx", 0);
kbxd_deinit_default_ctrl (ctrl); kbxd_deinit_default_ctrl (ctrl);
xfree (ctrl); xfree (ctrl);
} }