/* frontend.c - Database fronend code for keyboxd * Copyright (C) 2019 g10 Code GmbH * * This file is part of GnuPG. * * GnuPG is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * GnuPG is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, see <https://www.gnu.org/licenses/>. * SPDX-License-Identifier: GPL-3.0+ */ #include <config.h> #include <stdio.h> #include <stdlib.h> #include <stddef.h> #include <string.h> #include "keyboxd.h" #include <assuan.h> #include "../common/i18n.h" #include "../common/userids.h" #include "backend.h" #include "frontend.h" /* An object to keep infos about the database. */ struct { enum database_types db_type; backend_handle_t backend_handle; } the_database; /* Take a lock for reading the databases. */ static void take_read_lock (ctrl_t ctrl) { /* FIXME */ (void)ctrl; } /* Take a lock for reading and writing the databases. */ static void take_read_write_lock (ctrl_t ctrl) { /* FIXME */ (void)ctrl; } /* Release a lock. It is valid to call this even if no lock has been * taken in which case this is a nop. */ static void release_lock (ctrl_t ctrl) { /* FIXME */ (void)ctrl; } /* Set the database to use. Depending on the FILENAME suffix we * decide which one to use. This function must be called at daemon * startup because it employs no locking. If FILENAME has no * directory separator, the file is expected or created below * "$GNUPGHOME/public-keys.d/". In READONLY mode the file must exists; * otherwise it is created. */ gpg_error_t kbxd_set_database (ctrl_t ctrl, const char *filename_arg, int readonly) { gpg_error_t err; char *filename; enum database_types db_type = 0; backend_handle_t handle = NULL; unsigned int n; /* Do tilde expansion etc. */ if (strchr (filename_arg, DIRSEP_C) #ifdef HAVE_W32_SYSTEM || strchr (filename_arg, '/') /* Windows also accepts a slash. */ #endif ) filename = make_filename (filename_arg, NULL); else filename = make_filename (gnupg_homedir (), GNUPG_PUBLIC_KEYS_DIR, filename_arg, NULL); /* 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 * always be the first to be queried. */ if (the_database.db_type) { log_error ("error: only one database allowed\n"); err = gpg_error (GPG_ERR_CONFLICT); goto leave; } /* Init the cache. */ err = be_cache_initialize (); if (err) goto leave; n = strlen (filename); if (db_type) ; /* We already know it. */ else if (n > 4 && !strcmp (filename + n - 4, ".kbx")) db_type = DB_TYPE_KBX; else if (n > 3 && !strcmp (filename + n - 3, ".db")) db_type = DB_TYPE_SQLITE; else { log_error (_("can't use file '%s': %s\n"), filename, _("unknown suffix")); err = gpg_error (GPG_ERR_NOT_SUPPORTED); goto leave; } err = gpg_error (GPG_ERR_BUG); switch (db_type) { case DB_TYPE_NONE: /* NOTREACHED */ break; case DB_TYPE_CACHE: err = be_cache_add_resource (ctrl, &handle); break; case DB_TYPE_KBX: err = be_kbx_add_resource (ctrl, &handle, filename, readonly); break; case DB_TYPE_SQLITE: err = be_sqlite_add_resource (ctrl, &handle, filename, readonly); break; } if (err) goto leave; the_database.db_type = db_type; the_database.backend_handle = handle; handle = NULL; leave: if (err) { log_error ("error setting database '%s': %s\n", filename, gpg_strerror (err)); be_generic_release_backend (ctrl, handle); } xfree (filename); return err; } /* Release all per session objects. */ void kbxd_release_session_info (ctrl_t ctrl) { if (!ctrl) return; be_release_request (ctrl->db_req); ctrl->db_req = NULL; } gpg_error_t kbxd_rollback (void) { return be_sqlite_rollback (); } gpg_error_t kbxd_commit (void) { return be_sqlite_commit (); } static void dump_search_desc (struct keydb_search_desc *desc) { switch (desc->mode) { case KEYDB_SEARCH_MODE_EXACT: log_printf ("EXACT: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_SUBSTR: log_printf ("SUBSTR: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_MAIL: log_printf ("MAIL: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_MAILSUB: log_printf ("MAILSUB: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_MAILEND: log_printf ("MAILEND: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_WORDS: log_printf ("WORDS: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_SHORT_KID: log_printf ("SHORT_KID: 0x%08lX\n", (ulong)desc->u.kid[1]); break; case KEYDB_SEARCH_MODE_LONG_KID: log_printf ("LONG_KID: 0x%08lX%08lX\n", (ulong)desc->u.kid[0], (ulong)desc->u.kid[1]); break; case KEYDB_SEARCH_MODE_FPR: log_printf ("FPR%02d: ", desc->fprlen); log_printhex (desc->u.fpr, desc->fprlen, ""); break; case KEYDB_SEARCH_MODE_ISSUER: log_printf ("ISSUER: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_ISSUER_SN: log_printf ("ISSUER_SN: '#%.*s/%s'\n", (int)desc->snlen, desc->sn, desc->u.name); break; case KEYDB_SEARCH_MODE_SN: log_printf ("SN: '%.*s'\n", (int)desc->snlen, desc->sn); break; case KEYDB_SEARCH_MODE_SUBJECT: log_printf ("SUBJECT: '%s'\n", desc->u.name); break; case KEYDB_SEARCH_MODE_KEYGRIP: log_printf ("KEYGRIP: "); log_printhex (desc[0].u.grip, KEYGRIP_LEN, ""); break; case KEYDB_SEARCH_MODE_UBID: log_printf ("UBID: "); log_printhex (desc[0].u.ubid, UBID_LEN, ""); break; case KEYDB_SEARCH_MODE_FIRST: log_printf ("FIRST\n"); break; case KEYDB_SEARCH_MODE_NEXT: log_printf ("NEXT\n"); break; default: log_printf ("Bad search mode (%d)\n", desc->mode); } } /* Search for the keys described by (DESC,NDESC) and return them to * the caller. If RESET is set, the search state is first reset. * Only a reset guarantees that changed search description in DESC are * considered. */ gpg_error_t kbxd_search (ctrl_t ctrl, KEYDB_SEARCH_DESC *desc, unsigned int ndesc, int reset) { gpg_error_t err; int i; db_request_t request; if (DBG_CLOCK) log_clock ("%s: enter", __func__); if (DBG_LOOKUP) { log_debug ("%s: %u search descriptions:\n", __func__, ndesc); for (i = 0; i < ndesc; i ++) { log_debug ("%s %d: ", __func__, i); dump_search_desc (&desc[i]); } } take_read_lock (ctrl); /* Allocate a handle object if none exists for this context. */ if (!ctrl->db_req) { ctrl->db_req = xtrycalloc (1, sizeof *ctrl->db_req); if (!ctrl->db_req) { err = gpg_error_from_syserror (); goto leave; } } request = ctrl->db_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 * letting the caller do a separate call for an initial reset. */ if (!desc || reset) { switch (the_database.db_type) { case DB_TYPE_CACHE: err = 0; /* Nothing to do. */ break; case DB_TYPE_KBX: err = be_kbx_search (ctrl, the_database.backend_handle, request, NULL, 0); break; case DB_TYPE_SQLITE: err = be_sqlite_search (ctrl, the_database.backend_handle, request, NULL, 0); break; default: err = gpg_error (GPG_ERR_INTERNAL); break; } if (err) { log_error ("error during the %ssearch reset: %s\n", reset? "initial ":"", gpg_strerror (err)); goto leave; } request->any_search = 0; request->any_found = 0; request->next_dbidx = 0; if (!desc) /* Reset only mode */ { err = 0; goto leave; } } /* Divert to the backend for the actual search. */ switch (the_database.db_type) { case DB_TYPE_CACHE: err = be_cache_search (ctrl, the_database.backend_handle, request, desc, ndesc); /* Expected error codes from the cache lookup are: * 0 - found and returned via the cache * GPG_ERR_NOT_FOUND - marked in the cache as not available * GPG_ERR_EOF - cache miss. */ break; case DB_TYPE_KBX: err = be_kbx_search (ctrl, the_database.backend_handle, request, desc, ndesc); break; case DB_TYPE_SQLITE: err = be_sqlite_search (ctrl, the_database.backend_handle, request, desc, ndesc); 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) log_debug ("%s: searched %s => %s\n", __func__, strdbtype (the_database.db_type), gpg_strerror (err)); request->any_search = 1; if (!err) { request->any_found = 1; } else if (gpg_err_code (err) == GPG_ERR_EOF) { if (the_database.db_type == DB_TYPE_CACHE && request->last_cached_valid) { if (request->last_cached_final) goto leave; } request->next_dbidx++; /* 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; } leave: release_lock (ctrl); if (DBG_CLOCK) log_clock ("%s: leave (%s)", __func__, err? "not found" : "found"); return err; } /* Store; that is insert or update the key (BLOB,BLOBLEN). MODE * controls whether only updates or only inserts are allowed. */ gpg_error_t kbxd_store (ctrl_t ctrl, const void *blob, size_t bloblen, enum kbxd_store_modes mode) { gpg_error_t err; db_request_t request; char ubid[UBID_LEN]; enum pubkey_types pktype; int insert = 0; if (DBG_CLOCK) log_clock ("%s: enter", __func__); take_read_write_lock (ctrl); /* Allocate a handle object if none exists for this context. */ if (!ctrl->db_req) { ctrl->db_req = xtrycalloc (1, sizeof *ctrl->db_req); if (!ctrl->db_req) { err = gpg_error_from_syserror (); goto leave; } } request = ctrl->db_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. */ err = be_ubid_from_blob (blob, bloblen, &pktype, ubid); if (err) goto leave; if (the_database.db_type == DB_TYPE_KBX) { err = be_kbx_seek (ctrl, the_database.backend_handle, request, ubid); if (!err) ; /* Found - need to update. */ else if (gpg_err_code (err) == GPG_ERR_EOF) insert = 1; /* Not found - need to insert. */ else { log_debug ("%s: searching fingerprint failed: %s\n", __func__, gpg_strerror (err)); goto leave; } if (insert) { if (mode == KBXD_STORE_UPDATE) err = gpg_error (GPG_ERR_CONFLICT); else err = be_kbx_insert (ctrl, the_database.backend_handle, request, pktype, blob, bloblen); } else /* Update. */ { if (mode == KBXD_STORE_INSERT) err = gpg_error (GPG_ERR_CONFLICT); else err = be_kbx_update (ctrl, the_database.backend_handle, request, pktype, blob, bloblen); } } else if (the_database.db_type == DB_TYPE_SQLITE) { err = be_sqlite_store (ctrl, the_database.backend_handle, request, mode, pktype, ubid, blob, bloblen); } else { log_error ("%s: unsupported database type %d\n", __func__, the_database.db_type); err = gpg_error (GPG_ERR_INTERNAL); } leave: release_lock (ctrl); if (DBG_CLOCK) log_clock ("%s: leave", __func__); return err; } /* Delete; remove the blob identified by UBID. */ gpg_error_t kbxd_delete (ctrl_t ctrl, const unsigned char *ubid) { gpg_error_t err; db_request_t request; if (DBG_CLOCK) log_clock ("%s: enter", __func__); take_read_write_lock (ctrl); /* Allocate a handle object if none exists for this context. */ if (!ctrl->db_req) { ctrl->db_req = xtrycalloc (1, sizeof *ctrl->db_req); if (!ctrl->db_req) { err = gpg_error_from_syserror (); goto leave; } } request = ctrl->db_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 (the_database.db_type == DB_TYPE_KBX) { err = be_kbx_seek (ctrl, the_database.backend_handle, request, ubid); if (!err) ; /* Found - we can delete. */ else if (gpg_err_code (err) == GPG_ERR_EOF) { err = gpg_error (GPG_ERR_NOT_FOUND); goto leave; } else { log_debug ("%s: searching primary fingerprint failed: %s\n", __func__, gpg_strerror (err)); goto leave; } err = be_kbx_delete (ctrl, the_database.backend_handle, request); } else if (the_database.db_type == DB_TYPE_SQLITE) { err = be_sqlite_delete (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); } leave: release_lock (ctrl); if (DBG_CLOCK) log_clock ("%s: leave", __func__); return err; }