/* ldb database library Copyright (C) Andrew Tridgell 2004 ** NOTE! The following LGPL license applies to the ldb ** library. This does NOT imply that all of Samba is released ** under the LGPL This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /* * Name: ldb * * Component: ldb tdb backend * * Description: core functions for tdb backend * * Author: Andrew Tridgell */ #include "includes.h" #include "ldb/ldb_tdb/ldb_tdb.h" /* form a TDB_DATA for a record key caller frees note that the key for a record can depend on whether the dn refers to a case sensitive index record or not */ struct TDB_DATA ltdb_key(struct ldb_context *ldb, const char *dn) { TDB_DATA key; char *key_str = NULL; char *dn_folded = NULL; const char *prefix = LTDB_INDEX ":"; const char *s; int flags; /* most DNs are case insensitive. The exception is index DNs for case sensitive attributes there are 3 cases dealt with in this code: 1) if the dn doesn't start with @INDEX: then uppercase whole dn 2) if the dn starts with @INDEX:attr and 'attr' is a case insensitive attribute then uppercase whole dn 3) if the dn starts with @INDEX:attr and 'attr' is a case sensitive attribute then uppercase up to the value of the attribute, but not the value itself */ if (strncmp(dn, prefix, strlen(prefix)) == 0 && (s = strchr(dn+strlen(prefix), ':'))) { char *attr_name, *attr_name_folded; attr_name = ldb_strndup(ldb, dn+strlen(prefix), (s-(dn+strlen(prefix)))); if (!attr_name) { goto failed; } flags = ltdb_attribute_flags(ldb, attr_name); if (flags & LTDB_FLAG_CASE_INSENSITIVE) { dn_folded = ldb_casefold(ldb, dn); } else { attr_name_folded = ldb_casefold(ldb, attr_name); if (!attr_name_folded) { goto failed; } ldb_asprintf(ldb, &dn_folded, "%s:%s:%s", prefix, attr_name_folded, s+1); ldb_free(ldb, attr_name_folded); } ldb_free(ldb, attr_name); } else { dn_folded = ldb_casefold(ldb, dn); } if (!dn_folded) { goto failed; } ldb_asprintf(ldb, &key_str, "DN=%s", dn_folded); ldb_free(ldb, dn_folded); if (!key_str) { goto failed; } key.dptr = key_str; key.dsize = strlen(key_str)+1; return key; failed: errno = ENOMEM; key.dptr = NULL; key.dsize = 0; return key; } /* lock the database for write - currently a single lock is used */ static int ltdb_lock(struct ldb_context *ldb) { struct ltdb_private *ltdb = ldb->private_data; TDB_DATA key; int ret; key = ltdb_key(ldb, "LDBLOCK"); if (!key.dptr) { return -1; } ret = tdb_chainlock(ltdb->tdb, key); ldb_free(ldb, key.dptr); return ret; } /* unlock the database after a ltdb_lock() */ static void ltdb_unlock(struct ldb_context *ldb) { struct ltdb_private *ltdb = ldb->private_data; TDB_DATA key; key = ltdb_key(ldb, "LDBLOCK"); if (!key.dptr) { return; } tdb_chainunlock(ltdb->tdb, key); ldb_free(ldb, key.dptr); } /* we've made a modification to a dn - possibly reindex and update sequence number */ static int ltdb_modified(struct ldb_context *ldb, const char *dn) { int ret = 0; if (strcmp(dn, LTDB_INDEXLIST) == 0 || strcmp(dn, LTDB_ATTRIBUTES) == 0) { ret = ltdb_reindex(ldb); } if (ret == 0 && strcmp(dn, LTDB_BASEINFO) != 0) { ret = ltdb_increase_sequence_number(ldb); } return ret; } /* store a record into the db */ int ltdb_store(struct ldb_context *ldb, const struct ldb_message *msg, int flgs) { struct ltdb_private *ltdb = ldb->private_data; TDB_DATA tdb_key, tdb_data; int ret; tdb_key = ltdb_key(ldb, msg->dn); if (!tdb_key.dptr) { return -1; } ret = ltdb_pack_data(ldb, msg, &tdb_data); if (ret == -1) { ldb_free(ldb, tdb_key.dptr); return -1; } ret = tdb_store(ltdb->tdb, tdb_key, tdb_data, flgs); if (ret == -1) { goto done; } ret = ltdb_index_add(ldb, msg); if (ret == -1) { tdb_delete(ltdb->tdb, tdb_key); } done: ldb_free(ldb, tdb_key.dptr); ldb_free(ldb, tdb_data.dptr); return ret; } /* add a record to the database */ static int ltdb_add(struct ldb_context *ldb, const struct ldb_message *msg) { struct ltdb_private *ltdb = ldb->private_data; int ret; ltdb->last_err_string = NULL; if (ltdb_lock(ldb) != 0) { return -1; } if (ltdb_cache_load(ldb) != 0) { ltdb_unlock(ldb); return -1; } ret = ltdb_store(ldb, msg, TDB_INSERT); if (ret == 0) { ltdb_modified(ldb, msg->dn); } ltdb_unlock(ldb); return ret; } /* delete a record from the database, not updating indexes (used for deleting index records) */ int ltdb_delete_noindex(struct ldb_context *ldb, const char *dn) { struct ltdb_private *ltdb = ldb->private_data; TDB_DATA tdb_key; int ret; tdb_key = ltdb_key(ldb, dn); if (!tdb_key.dptr) { return -1; } ret = tdb_delete(ltdb->tdb, tdb_key); ldb_free(ldb, tdb_key.dptr); return ret; } /* delete a record from the database */ static int ltdb_delete(struct ldb_context *ldb, const char *dn) { struct ltdb_private *ltdb = ldb->private_data; int ret; struct ldb_message msg; ltdb->last_err_string = NULL; if (ltdb_lock(ldb) != 0) { return -1; } if (ltdb_cache_load(ldb) != 0) { ltdb_unlock(ldb); return -1; } /* in case any attribute of the message was indexed, we need to fetch the old record */ ret = ltdb_search_dn1(ldb, dn, &msg); if (ret != 1) { /* not finding the old record is an error */ goto failed; } ret = ltdb_delete_noindex(ldb, dn); if (ret == -1) { ltdb_search_dn1_free(ldb, &msg); goto failed; } /* remove any indexed attributes */ ret = ltdb_index_del(ldb, &msg); ltdb_search_dn1_free(ldb, &msg); if (ret == 0) { ltdb_modified(ldb, dn); } ltdb_unlock(ldb); return ret; failed: ltdb_unlock(ldb); return -1; } /* find an element by attribute name. At the moment this does a linear search, it should be re-coded to use a binary search once all places that modify records guarantee sorted order return the index of the first matching element if found, otherwise -1 */ static int find_element(const struct ldb_message *msg, const char *name) { int i; for (i=0;i<msg->num_elements;i++) { if (ldb_attr_cmp(msg->elements[i].name, name) == 0) { return i; } } return -1; } /* add an element to an existing record. Assumes a elements array that we can call re-alloc on, and assumed that we can re-use the data pointers from the passed in additional values. Use with care! returns 0 on success, -1 on failure (and sets errno) */ static int msg_add_element(struct ldb_context *ldb, struct ldb_message *msg, struct ldb_message_element *el) { struct ldb_message_element *e2; int i; e2 = ldb_realloc_p(ldb, msg->elements, struct ldb_message_element, msg->num_elements+1); if (!e2) { errno = ENOMEM; return -1; } msg->elements = e2; e2 = &msg->elements[msg->num_elements]; e2->name = el->name; e2->flags = el->flags; e2->values = NULL; if (el->num_values != 0) { e2->values = ldb_malloc_array_p(ldb, struct ldb_val, el->num_values); if (!e2->values) { free(e2->name); errno = ENOMEM; return -1; } } for (i=0;i<el->num_values;i++) { e2->values[i] = el->values[i]; } e2->num_values = el->num_values; msg->num_elements++; return 0; } /* delete all elements having a specified attribute name */ static int msg_delete_attribute(struct ldb_context *ldb, struct ldb_message *msg, const char *name) { int i, count=0; struct ldb_message_element *el2; el2 = ldb_malloc_array_p(ldb, struct ldb_message_element, msg->num_elements); if (!el2) { errno = ENOMEM; return -1; } for (i=0;i<msg->num_elements;i++) { if (ldb_attr_cmp(msg->elements[i].name, name) != 0) { el2[count++] = msg->elements[i]; } else { ldb_free(ldb, msg->elements[i].values); } } msg->num_elements = count; ldb_free(ldb, msg->elements); msg->elements = el2; return 0; } /* delete all elements matching an attribute name/value return 0 on success, -1 on failure */ static int msg_delete_element(struct ldb_context *ldb, struct ldb_message *msg, const char *name, const struct ldb_val *val) { int i; struct ldb_message_element *el; i = find_element(msg, name); if (i == -1) { return -1; } el = &msg->elements[i]; for (i=0;i<el->num_values;i++) { if (ldb_val_equal(ldb, msg->elements[i].name, &el->values[i], val)) { if (i<el->num_values-1) { memmove(&el->values[i], &el->values[i+1], sizeof(el->values[i])*(el->num_values-(i+1))); } el->num_values--; if (el->num_values == 0) { return msg_delete_attribute(ldb, msg, name); } return 0; } } return -1; } /* modify a record - internal interface yuck - this is O(n^2). Luckily n is usually small so we probably get away with it, but if we ever have really large attribute lists then we'll need to look at this again */ int ltdb_modify_internal(struct ldb_context *ldb, const struct ldb_message *msg) { struct ltdb_private *ltdb = ldb->private_data; TDB_DATA tdb_key, tdb_data; struct ldb_message msg2; int ret, i, j; tdb_key = ltdb_key(ldb, msg->dn); if (!tdb_key.dptr) { return -1; } tdb_data = tdb_fetch(ltdb->tdb, tdb_key); if (!tdb_data.dptr) { free(tdb_key.dptr); return -1; } ret = ltdb_unpack_data(ldb, &tdb_data, &msg2); if (ret == -1) { ldb_free(ldb, tdb_key.dptr); free(tdb_data.dptr); return -1; } if (!msg2.dn) { msg2.dn = msg->dn; } for (i=0;i<msg->num_elements;i++) { switch (msg->elements[i].flags & LDB_FLAG_MOD_MASK) { case LDB_FLAG_MOD_ADD: /* add this element to the message. fail if it already exists */ ret = find_element(&msg2, msg->elements[i].name); if (ret != -1) { ltdb->last_err_string = "Attribute exists"; goto failed; } if (msg_add_element(ldb, &msg2, &msg->elements[i]) != 0) { goto failed; } break; case LDB_FLAG_MOD_REPLACE: /* replace all elements of this attribute name with the elements listed. The attribute not existing is not an error */ msg_delete_attribute(ldb, &msg2, msg->elements[i].name); /* add the replacement element, if not empty */ if (msg->elements[i].num_values != 0 && msg_add_element(ldb, &msg2, &msg->elements[i]) != 0) { goto failed; } break; case LDB_FLAG_MOD_DELETE: /* we could be being asked to delete all values or just some values */ if (msg->elements[i].num_values == 0) { if (msg_delete_attribute(ldb, &msg2, msg->elements[i].name) != 0) { ltdb->last_err_string = "No such attribute"; goto failed; } break; } for (j=0;j<msg->elements[i].num_values;j++) { if (msg_delete_element(ldb, &msg2, msg->elements[i].name, &msg->elements[i].values[j]) != 0) { ltdb->last_err_string = "No such attribute"; goto failed; } } break; } } /* we've made all the mods - save the modified record back into the database */ ret = ltdb_store(ldb, &msg2, TDB_MODIFY); ldb_free(ldb, tdb_key.dptr); free(tdb_data.dptr); ltdb_unpack_data_free(ldb, &msg2); return ret; failed: ldb_free(ldb, tdb_key.dptr); free(tdb_data.dptr); ltdb_unpack_data_free(ldb, &msg2); return -1; } /* modify a record */ static int ltdb_modify(struct ldb_context *ldb, const struct ldb_message *msg) { struct ltdb_private *ltdb = ldb->private_data; int ret; ltdb->last_err_string = NULL; if (ltdb_lock(ldb) != 0) { return -1; } if (ltdb_cache_load(ldb) != 0) { ltdb_unlock(ldb); return -1; } ret = ltdb_modify_internal(ldb, msg); if (ret == 0) { ltdb_modified(ldb, msg->dn); } ltdb_unlock(ldb); return ret; } /* close database */ static int ltdb_close(struct ldb_context *ldb) { struct ltdb_private *ltdb = ldb->private_data; int ret; ltdb->last_err_string = NULL; ltdb_cache_free(ldb); ldb_set_alloc(ldb, NULL, NULL); ret = tdb_close(ltdb->tdb); ldb_free(ldb, ltdb); free(ldb); return ret; } /* return extended error information */ static const char *ltdb_errstring(struct ldb_context *ldb) { struct ltdb_private *ltdb = ldb->private_data; if (ltdb->last_err_string) { return ltdb->last_err_string; } return tdb_errorstr(ltdb->tdb); } static const struct ldb_backend_ops ltdb_ops = { ltdb_close, ltdb_search, ltdb_search_free, ltdb_add, ltdb_modify, ltdb_delete, ltdb_errstring, ltdb_cache_free }; /* connect to the database */ struct ldb_context *ltdb_connect(const char *url, unsigned int flags, const char *options[]) { const char *path; int tdb_flags, open_flags; struct ltdb_private *ltdb; TDB_CONTEXT *tdb; struct ldb_context *ldb; ldb = calloc(1, sizeof(struct ldb_context)); if (!ldb) { errno = ENOMEM; return NULL; } /* parse the url */ if (strchr(url, ':')) { if (strncmp(url, "tdb://", 6) != 0) { errno = EINVAL; return NULL; } path = url+6; } else { path = url; } tdb_flags = TDB_DEFAULT; if (flags & LDB_FLG_RDONLY) { open_flags = O_RDONLY; } else { open_flags = O_CREAT | O_RDWR; } /* note that we use quite a large default hash size */ tdb = tdb_open(path, 10000, tdb_flags, open_flags, 0666); if (!tdb) { free(ldb); return NULL; } ltdb = ldb_malloc_p(ldb, struct ltdb_private); if (!ltdb) { tdb_close(tdb); free(ldb); errno = ENOMEM; return NULL; } ltdb->tdb = tdb; ltdb->sequence_number = 0; memset(<db->cache, 0, sizeof(ltdb->cache)); ldb->private_data = ltdb; ldb->ops = <db_ops; return ldb; }