[Date Prev][Date Next] [Chronological] [Thread] [Top]

Removing deleted entry's id from Parent's children list



I realize that Blake Binkley and I are the only people using LDAP
that seem to consider this an issue, but I made a change to remove
an entry from its parent's children ID list when it's deleted.  It
involved changes to delete.c, id2children.c and idl.c in the back-ldbm
directory.

The diff is off the latest devel snapshot.


--- delete.c.orig	Mon Nov  9 13:53:29 1998
+++ delete.c	Tue Nov 10 08:00:11 1998
@@ -23,7 +23,8 @@
 {
 	struct ldbminfo	*li = (struct ldbminfo *) be->be_private;
 	char		*matched = NULL;
-	Entry		*e;
+        char            *pdn = NULL;
+	Entry		*e, *p;
 
 	Debug(LDAP_DEBUG_ARGS, "==> ldbm_back_delete: %s\n", dn, 0, 0);
 
@@ -66,6 +67,13 @@
 		e->e_rdwr.readers_reading, e->e_rdwr.writer_writing, 0);
 
 	/* XXX delete from parent's id2children entry XXX */
+	pdn = dn_parent( be, dn );
+	matched = NULL;
+	p = dn2entry_w( be, pdn, &matched );
+	if ( id2children_remove( be, p, e ) != 0 ) {
+		send_ldap_result( conn, op, LDAP_OPERATIONS_ERROR, "","" );
+                goto error_return;
+	}
 
 	/* delete from dn2id mapping */
 	if ( dn2id_delete( be, e->e_dn ) != 0 ) {

--- id2children.c.orig	Mon Nov  9 15:32:01 1998
+++ id2children.c	Mon Nov  9 15:23:05 1998
@@ -60,6 +60,49 @@
 	return( 0 );
 }
 
+
+int
+id2children_remove(
+    Backend	*be,
+    Entry	*p,
+    Entry	*e
+)
+{
+	struct dbcache	*db;
+	Datum		key;
+	int		len, rc;
+	IDList		*idl;
+	char		buf[20];
+
+	Debug( LDAP_DEBUG_TRACE, "=> id2children_remove( %d, %d )\n", p ? p->e_id
+	    : 0, e->e_id, 0 );
+
+	if ( (db = ldbm_cache_open( be, "id2children", LDBM_SUFFIX,
+	    LDBM_WRCREAT )) == NULL ) {
+		Debug( LDAP_DEBUG_ANY,
+		    "<= id2children_add -1 could not open \"id2children%s\"\n",
+		    LDBM_SUFFIX, 0, 0 );
+		return( -1 );
+	}
+
+	memset( &key, 0, sizeof(key) );
+	sprintf( buf, "%c%d", EQ_PREFIX, p ? p->e_id : 0 );
+	key.dptr = buf;
+	key.dsize = strlen( buf ) + 1;
+
+	if ( idl_delete_key( be, db, key, e->e_id ) != 0 ) {
+		Debug( LDAP_DEBUG_TRACE, "<= id2children_add -1 (idl_insert)\n",
+		    0, 0, 0 );
+		ldbm_cache_close( be, db );
+		return( -1 );
+	}
+
+	ldbm_cache_close( be, db );
+
+	Debug( LDAP_DEBUG_TRACE, "<= id2children_add 0\n", 0, 0, 0 );
+	return( 0 );
+}
+
 int
 has_children(
     Backend	*be,

--- idl.c.orig	Mon Nov  9 15:32:32 1998
+++ idl.c	Mon Nov  9 15:36:50 1998
@@ -618,6 +618,93 @@
 	return( i == 0 ? 1 : 0 );	/* inserted - first id changed or not */
 }
 
+int
+idl_delete_key (
+	Backend         *be,
+	struct dbcache  *db,
+	Datum           key,
+	ID              id
+)
+{
+	Datum  k2;
+	IDList *idl, *tmp;
+	int i, j, nids;
+	char	*kstr;
+
+	if ( (idl = idl_fetch_one( be, db, key ) ) == NULL )
+	{
+		/* It wasn't found.  Hmm... */
+		return -1;
+	}
+
+	if ( ! INDIRECT_BLOCK( idl ) )
+	{
+		for ( i=0; i < idl->b_nids; i++ )
+		{
+			if ( idl->b_ids[i] == id )
+			{
+				memcpy ( &idl->b_ids[i], &idl->b_ids[i+1], sizeof(ID)*(idl->b_nids-(i+1)));
+				idl->b_ids[idl->b_nids-1] = NOID;
+				idl->b_nids--;
+				if ( idl->b_nids )
+					idl_store( be, db, key, idl );
+				else
+					ldbm_cache_delete( db, key );
+				return 0;
+			}
+			/*  We didn't find the ID.  Hmmm... */
+		}
+		return -1;
+	}
+	
+	/* We have to go through an indirect block and find the ID
+	   in the list of IDL's
+	   */
+	for ( nids = 0; idl->b_ids[nids] != NOID; nids++ )
+		;	/* NULL */
+	kstr = (char *) ch_malloc( key.dsize + 20 );
+	for ( j = 0; idl->b_ids[j] != NOID; j++ ) 
+	{
+		memset( &k2, 0, sizeof(k2) );
+		sprintf( kstr, "%c%s%d", CONT_PREFIX, key.dptr, idl->b_ids[j] );
+		k2.dptr = kstr;
+		k2.dsize = strlen( kstr ) + 1;
+
+		if ( (tmp = idl_fetch_one( be, db, k2 )) == NULL ) {
+			Debug( LDAP_DEBUG_ANY,
+			    "idl_fetch of (%s) returns NULL\n", k2.dptr, 0, 0 );
+			continue;
+		}
+		/*
+		   Now try to find the ID in tmp
+		*/
+		for ( i=0; i < tmp->b_nids; i++ )
+		{
+			if ( tmp->b_ids[i] == id )
+			{
+				memcpy ( &tmp->b_ids[i], &tmp->b_ids[i+1], sizeof(ID)*(tmp->b_nids-(i+1)));
+				tmp->b_ids[tmp->b_nids-1] = NOID;
+				tmp->b_nids--;
+				if ( tmp->b_nids )
+					idl_store ( be, db, k2, tmp );
+				else
+				{
+					ldbm_cache_delete( db, k2 );
+					memcpy ( &idl->b_ids[j], &idl->b_ids[j+1], sizeof(ID)*(nids-(j+1)));
+					idl->b_ids[nids-1] = NOID;
+					nids--;
+					if ( ! nids )
+						ldbm_cache_delete( db, key );
+					else
+						idl_store( be, db, key, idl );
+				}
+				return 0;
+			}
+		}
+	}
+	return -1;
+}
+
 static IDList *
 idl_dup( IDList *idl )
 {