Hello! Is there anybody interested in CDB support for auth dicts?
I've implemented simple patch for it.
diff --git a/configure.in b/configure.in index a4e5b5a..89e076e 100644 --- a/configure.in +++ b/configure.in @@ -121,6 +121,11 @@ AS_HELP_STRING([--with-vpopmail], [Build with vpopmail support (auto)]), # want_db=no) want_db=no +AC_ARG_WITH(cdb, +AS_HELP_STRING([--with-cdb], [Build with CDB support]), + TEST_WITH(cdb, $withval), + want_cdb=no) + dnl The --with-sql is useful only if Dovecot is being built with all the SQL dnl drivers as modules. If any SQL driver is built-in, this option is ignored. AC_ARG_WITH(sql, @@ -2110,6 +2115,27 @@ if test $want_db != no; then fi fi +if test $want_cdb != no; then + AC_CHECK_LIB(cdb, cdb_init, [ + AC_CHECK_HEADER(cdb.h, [ + CDB_LIBS="$CDB_LIBS -lcdb" + AC_DEFINE(BUILD_CDB,, Build with CDB support) + ], [ + if test $want_cdb = yes; then + AC_ERROR([Can't build with CDB support: cdb.h not found]) + fi + ]) + ], [ + if test $want_cdb = yes; then + AC_ERROR([Can't build with CDB support: libcdb not found]) + fi + ]) +echo "CDB_LIBS: $CDB_LIBS" + if test "$CDB_LIBS" != ""; then + LIBS="$LIBS $CDB_LIBS" + fi +fi + if test $want_pgsql != no; then AC_CHECK_PROG(PG_CONFIG, pg_config, pg_config, NO) if test $PG_CONFIG = NO; then @@ -2394,6 +2420,7 @@ AC_SUBST(SQLITE_CFLAGS) AC_SUBST(SQLITE_LIBS) AC_SUBST(DICT_LIBS) +AC_SUBST(CDB_LIBS) AC_SUBST(dict_drivers) dnl ** diff --git a/src/lib-dict/Makefile.am b/src/lib-dict/Makefile.am index 31d3522..a447a90 100644 --- a/src/lib-dict/Makefile.am +++ b/src/lib-dict/Makefile.am @@ -14,6 +14,7 @@ base_sources = \ dict.c \ dict-client.c \ dict-file.c \ + dict-cdb.c \ dict-memcached.c \ dict-redis.c diff --git a/src/lib-dict/dict-cdb.c b/src/lib-dict/dict-cdb.c new file mode 100644 index 0000000..49e5ac0 --- /dev/null +++ b/src/lib-dict/dict-cdb.c @@ -0,0 +1,137 @@ +/* Copyright (c) 2006-2012 Dovecot authors, see the included COPYING file */ + +#include "lib.h" + +#ifdef BUILD_CDB +#include "dict-private.h" +#include "safe-memset.h" + +#include <string.h> +#include <cdb.h> +#include <unistd.h> +#include <fcntl.h> + +#define CDB_WITH_NULL 1 +#define CDB_WITHOUT_NULL 2 + +struct cdb_dict { + struct dict dict; + struct cdb cdb; + char *path; + int fd, flag; +}; + +static void cdb_dict_deinit(struct dict *_dict); + +static struct dict * +cdb_dict_init(struct dict *driver, const char *uri, + enum dict_data_type value_type ATTR_UNUSED, + const char *username ATTR_UNUSED, const char *base_dir ATTR_UNUSED) +{ + struct cdb_dict *dict; + + dict = i_new(struct cdb_dict, 1); + + dict->dict = *driver; + dict->path = i_strdup(uri); + dict->flag = CDB_WITH_NULL | CDB_WITHOUT_NULL; + + /* initialize cdb to 0 (unallocated) */ + safe_memset(&dict->cdb, 0, sizeof(struct cdb)); + + dict->fd = open(dict->path, O_RDONLY); + if (dict->fd == -1) { + i_error("open(%s) failed: %m", dict->path); + cdb_dict_deinit(&dict->dict); + return NULL; + } + +#ifdef TINYCDB_VERSION + if (cdb_init(&dict->cdb, dict->fd) < 0) { + i_error("cdb_init(%s) failed: %m", dict->path); + cdb_dict_deinit(&dict->dict); + return NULL; + } +#else + cdb_init(&dict->cdb, dict->fd); +#endif + + return &dict->dict; +} +static void cdb_dict_deinit(struct dict *_dict) +{ + struct cdb_dict *dict = (struct cdb_dict *)_dict; + + /* we can safely deinit unallocated cdb */ + cdb_free(&dict->cdb); + + if (dict->fd != -1) { + if (close(dict->fd) < 0) + i_error("close(%s) failed: %m", dict->path); + } + + i_free(dict->path); + i_free(dict); +} + +static int cdb_dict_lookup(struct dict *_dict, pool_t pool, + const char *key, const char **value_r) +{ + struct cdb_dict *dict = (struct cdb_dict *)_dict; + unsigned datalen; + int ret = 0; + char *data; + + /* keys and values may be null terminated... */ + if (dict->flag & CDB_WITH_NULL) { + ret = cdb_find(&dict->cdb, key, (unsigned)strlen(key)+1); + if (ret > 0) + dict->flag &= ~CDB_WITHOUT_NULL; + } + + /* ...or not */ + if ((0 == ret) && (dict->flag & CDB_WITHOUT_NULL)) { + ret = cdb_find(&dict->cdb, key, (unsigned)strlen(key)); + if (ret > 0) + dict->flag &= ~CDB_WITH_NULL; + } + + if (ret <= 0) { + *value_r = NULL; + /* something bad with db */ + if (ret < 0) { + /*i_error("cdb_lookup(%s) on %s failed: %m", + key, dict->path);*/ + return -1; + } + /* found nothing */ + return 0; + } + + datalen = cdb_datalen(&dict->cdb); + data = p_new(pool, char, datalen + 1); + cdb_read(&dict->cdb, data, datalen, cdb_datapos(&dict->cdb)); + *value_r = data; + return 1; +} + + +struct dict dict_driver_cdb = { + .name = "cdb", + { + cdb_dict_init, + cdb_dict_deinit, + NULL, + cdb_dict_lookup, + NULL, + NULL, + NULL, + NULL, + NULL, + NULL, + NULL, + NULL, + NULL + } +}; +#endif diff --git a/src/lib-dict/dict-private.h b/src/lib-dict/dict-private.h index ae41dc8..9200c8f 100644 --- a/src/lib-dict/dict-private.h +++ b/src/lib-dict/dict-private.h @@ -55,5 +55,8 @@ extern struct dict dict_driver_client; extern struct dict dict_driver_file; extern struct dict dict_driver_memcached; extern struct dict dict_driver_redis; +#ifdef HAVE_CDB +extern struct dict dict_driver_cdb; +#endif #endif diff --git a/src/lib-dict/dict.c b/src/lib-dict/dict.c index 99d1580..a67eacc 100644 --- a/src/lib-dict/dict.c +++ b/src/lib-dict/dict.c @@ -58,6 +58,9 @@ void dict_drivers_register_builtin(void) dict_driver_register(&dict_driver_file); dict_driver_register(&dict_driver_memcached); dict_driver_register(&dict_driver_redis); +#ifdef HAVE_CDB + dict_driver_register(&dict_driver_cdb); +#endif } void dict_drivers_unregister_builtin(void) @@ -66,6 +69,10 @@ void dict_drivers_unregister_builtin(void) dict_driver_unregister(&dict_driver_file); dict_driver_unregister(&dict_driver_memcached); dict_driver_unregister(&dict_driver_redis); +#ifdef HAVE_CDB + dict_driver_unregister(&dict_driver_cdb); +#endif + } struct dict *dict_init(const char *uri, enum dict_data_type value_type, diff --git a/src/lib-dict/test-dict.c b/src/lib-dict/test-dict.c index 88164e0..0aa8e9b 100644 --- a/src/lib-dict/test-dict.c +++ b/src/lib-dict/test-dict.c @@ -8,6 +8,7 @@ struct dict dict_driver_client; struct dict dict_driver_file; struct dict dict_driver_memcached; struct dict dict_driver_redis; +struct dict dict_driver_cdb; static void test_dict_escape(void) {