/* * Copyright (c) 1998-2003 Carnegie Mellon University. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * 3. The name "Carnegie Mellon University" must not be used to * endorse or promote products derived from this software without * prior written permission. For permission or any other legal * details, please contact * Office of Technology Transfer * Carnegie Mellon University * 5000 Forbes Avenue * Pittsburgh, PA 15213-3890 * (412) 268-4387, fax: (412) 268-7395 * tech-transfer@andrew.cmu.edu * * 4. Redistributions of any form whatsoever must retain the following * acknowledgment: * "This product includes software developed by Computing Services * at Carnegie Mellon University (http://www.cmu.edu/computing/)." * * CARNEGIE MELLON UNIVERSITY DISCLAIMS ALL WARRANTIES WITH REGARD TO * THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY * AND FITNESS, IN NO EVENT SHALL CARNEGIE MELLON UNIVERSITY BE LIABLE * FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN * AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ /* $Id: cyrusdb.c,v 1.5 2005/03/05 00:37:12 dasenbro Exp $ */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "cyrusdb.h" #include "exitcodes.h" #include "libcyr_cfg.h" #include "retry.h" #include "xmalloc.h" struct cyrusdb_backend *cyrusdb_backends[] = { #ifdef HAVE_BDB &cyrusdb_berkeley, &cyrusdb_berkeley_nosync, #endif &cyrusdb_flat, &cyrusdb_skiplist, &cyrusdb_quotalegacy, NULL }; void cyrusdb_init() { int i, r; char dbdir[1024]; const char *confdir = libcyrus_config_getstring(CYRUSOPT_CONFIG_DIR); int initflags = libcyrus_config_getint(CYRUSOPT_DB_INIT_FLAGS); strcpy(dbdir, confdir); strcat(dbdir, FNAME_DBDIR); for(i=0; cyrusdb_backends[i]; i++) { r = (cyrusdb_backends[i])->init(dbdir, initflags); if(r) { syslog(LOG_ERR, "DBERROR: init() on %s", cyrusdb_backends[i]->name); } } } void cyrusdb_done() { int i; for(i=0; cyrusdb_backends[i]; i++) { (cyrusdb_backends[i])->done(); } } int cyrusdb_copyfile(const char *srcname, const char *dstname) { int srcfd, dstfd; struct stat sbuf; char *buf; int bufsize, n; if ((srcfd = open(srcname, O_RDONLY)) < 0) { syslog(LOG_DEBUG, "error opening %s for reading", srcname); return -1; } if (fstat(srcfd, &sbuf) < 0) { syslog(LOG_DEBUG, "error fstating %s", srcname); close(srcfd); return -1; } if ((dstfd = open(dstname, O_WRONLY | O_CREAT, sbuf.st_mode)) < 0) { syslog(LOG_DEBUG, "error opening %s for writing (%d)", dstname, sbuf.st_mode); close(srcfd); return -1; } bufsize = sbuf.st_blksize; if ((buf = (char*) xmalloc(bufsize)) == NULL) { syslog(LOG_DEBUG, "error allocing buf (%d)", bufsize); close(srcfd); close(dstfd); return -1; } for (;;) { n = read(srcfd, buf, bufsize); if (n < 0) { if (errno == EINTR) continue; syslog(LOG_DEBUG, "error reading buf (%d)", bufsize); close(srcfd); close(dstfd); unlink(dstname); return -1; } if (n == 0) break; if (retry_write(dstfd, buf, n) != n) { syslog(LOG_DEBUG, "error writing buf (%d)", n); close(srcfd); close(dstfd); unlink(dstname); return -1; } } close(srcfd); close(dstfd); return 0; } struct cyrusdb_backend *cyrusdb_fromname(const char *name) { int i; struct cyrusdb_backend *db = NULL; for (i = 0; cyrusdb_backends[i]; i++) { if (!strcmp(cyrusdb_backends[i]->name, name)) { db = cyrusdb_backends[i]; break; } } if (!db) { char errbuf[1024]; snprintf(errbuf, sizeof(errbuf), "cyrusdb backend %s not supported", name); fatal(errbuf, EC_CONFIG); } return db; }