Skip to content
Snippets Groups Projects
Commit f1fde41f authored by Kurt Zeilenga's avatar Kurt Zeilenga
Browse files

remove an entry from its parent's children ID list when it's deleted.

Provided by "Gary D. Williams" <sasgwi@unx.sas.com>
parent 14a5b7f5
No related branches found
No related tags found
No related merge requests found
......@@ -23,7 +23,8 @@ ldbm_back_delete(
{
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 @@ ldbm_back_delete(
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_r( 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 ) {
......@@ -96,5 +104,8 @@ error_return:;
/* free entry and writer lock */
cache_return_entry_w( &li->li_cache, e );
if( p )
cache_return_entry_r( &li->li_cache, p );
return( -1 );
}
......@@ -61,6 +61,49 @@ id2children_add(
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,
......
......@@ -618,6 +618,93 @@ idl_insert( IDList **idl, ID id, int maxids )
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 )
{
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment