diff options
Diffstat (limited to 'ldap/servers/slapd/entrywsi.c')
-rw-r--r-- | ldap/servers/slapd/entrywsi.c | 1151 |
1 files changed, 1151 insertions, 0 deletions
diff --git a/ldap/servers/slapd/entrywsi.c b/ldap/servers/slapd/entrywsi.c new file mode 100644 index 00000000..8a39b941 --- /dev/null +++ b/ldap/servers/slapd/entrywsi.c @@ -0,0 +1,1151 @@ +/** BEGIN COPYRIGHT BLOCK + * Copyright 2001 Sun Microsystems, Inc. + * Portions copyright 1999, 2001-2003 Netscape Communications Corporation. + * All rights reserved. + * END COPYRIGHT BLOCK **/ + +/* entrywsi.c - routines for dealing with entries... With State Information */ + +#include "slap.h" +#include "slapi-plugin.h" + +static void resolve_attribute_state(Slapi_Entry *e, Slapi_Attr *a, int attribute_state); + +static int +entry_present_value_to_deleted_value(Slapi_Attr *a, Slapi_Value *v) +{ + Slapi_Value *r= valueset_remove_value(a, &a->a_present_values, v); + if(r!=NULL) + { + slapi_valueset_add_value_ext(&a->a_deleted_values, r, SLAPI_VALUE_FLAG_PASSIN); + } + return LDAP_SUCCESS; +} + +static int +entry_present_value_to_zapped_value(Slapi_Attr *a, Slapi_Value *v) +{ + if(v!=NULL) + { + Slapi_Value *r= valueset_remove_value(a, &a->a_present_values, v); + if(r!=NULL) + { + slapi_value_free(&r); + } + } + return LDAP_SUCCESS; +} + +static int +entry_deleted_value_to_present_value(Slapi_Attr *a, Slapi_Value *v) +{ + Slapi_Value *r= valueset_remove_value(a, &a->a_deleted_values, v); + if(r!=NULL) + { + slapi_valueset_add_value_ext(&a->a_present_values, r, SLAPI_VALUE_FLAG_PASSIN); + } + return LDAP_SUCCESS; +} + +static int +entry_deleted_value_to_zapped_value(Slapi_Attr *a, Slapi_Value *v) +{ + if(v!=NULL) + { + Slapi_Value *r= valueset_remove_value(a, &a->a_deleted_values, v); + if(r!=NULL) + { + slapi_value_free(&r); + } + } + return LDAP_SUCCESS; +} + +static int +entry_present_attribute_to_deleted_attribute(Slapi_Entry *e, Slapi_Attr *a) +{ + attrlist_remove(&e->e_attrs,a->a_type); + attrlist_add(&e->e_deleted_attrs,a); + return LDAP_SUCCESS; +} + +static int +entry_deleted_attribute_to_present_attribute(Slapi_Entry *e, Slapi_Attr *a) +{ + attrlist_remove(&e->e_deleted_attrs,a->a_type); + attrlist_add(&e->e_attrs,a); + return LDAP_SUCCESS; +} + +/* + * Get the first deleted attribute. + * + * Return 0: Return the type and the CSN of the deleted attribute. + * Return -1: There are no deleted attributes. + */ +int +entry_first_deleted_attribute( const Slapi_Entry *e, Slapi_Attr **a) +{ + *a= e->e_deleted_attrs; + return( *a ? 0 : -1 ); +} + +/* + * Get the next deleted attribute. + * + * Return 0: the type and the CSN of the deleted attribute. + * Return -1: no deleted attributes. + */ +int +entry_next_deleted_attribute( const Slapi_Entry *e, Slapi_Attr **a) +{ + *a= (*a)->a_next; + return( *a ? 0 : -1 ); +} + +const CSN * +entry_get_maxcsn ( const Slapi_Entry *entry ) +{ + return entry->e_maxcsn; +} + +void +entry_set_maxcsn ( Slapi_Entry *entry, const CSN *csn ) +{ + if ( NULL == entry->e_maxcsn ) + { + entry->e_maxcsn = csn_dup ( csn ); + } + else if ( csn_compare ( entry->e_maxcsn, csn ) < 0 ) + { + csn_init_by_csn ( entry->e_maxcsn, csn ); + } +} + +/* + * Get the DN CSN of an entry. + */ +const CSN * +entry_get_dncsn(const Slapi_Entry *entry) +{ + return csnset_get_last_csn(entry->e_dncsnset); +} + +/* + * Get the DN CSN set of an entry. + */ +const CSNSet * +entry_get_dncsnset(const Slapi_Entry *entry) +{ + return entry->e_dncsnset; +} + +/* + * Add a DN CSN to an entry. + */ +int +entry_add_dncsn(Slapi_Entry *entry, const CSN *csn) +{ + PR_ASSERT(entry!=NULL); + if(!csnset_contains(entry->e_dncsnset,csn)) + { + csnset_add_csn(&entry->e_dncsnset, CSN_TYPE_VALUE_DISTINGUISHED, csn); + } + return 0; +} + +/* + * Add a DN CSN to an entry, but uses flags to control the behavior + * Using the ENTRY_DNCSN_INCREASING flag makes sure the csnset is in + * order of increasing csn. csnset_insert_csn may not be very fast, so + * we may have to revisit this if it becomes a performance problem. + * In most cases, storing the csn unsorted is ok since the server + * usually makes sure the csn is already in order. However, when doing + * a str2entry, the order is not preserved unless we sort it. + */ +int +entry_add_dncsn_ext(Slapi_Entry *entry, const CSN *csn, PRUint32 flags) +{ + PR_ASSERT(entry!=NULL); + if(!csnset_contains(entry->e_dncsnset,csn)) + { + if (flags & ENTRY_DNCSN_INCREASING) + { + csnset_insert_csn(&entry->e_dncsnset, CSN_TYPE_VALUE_DISTINGUISHED, csn); + } + else + { + csnset_add_csn(&entry->e_dncsnset, CSN_TYPE_VALUE_DISTINGUISHED, csn); + } + } + return 0; +} + +/* + * Set the CSN for all the present values on the entry. + * This is only intended to be used for new entries + * being added. + */ +int +entry_set_csn(Slapi_Entry *entry, const CSN *csn) +{ + Slapi_Attr *a; + + PR_ASSERT(entry!=NULL); + + slapi_entry_first_attr( entry, &a ); + while(a!=NULL) + { + /* + * JCM - it'd be more efficient if the str2entry code + * set a flag on the attribute structure. + */ + if(strcasecmp(a->a_type, SLAPI_ATTR_UNIQUEID)!=0) + { + attr_set_csn(a,csn); + } + slapi_entry_next_attr( entry, a, &a ); + } + return 0; +} + +/* + * Set the Distinguished CSN for the RDN components of the entry. + */ +void +entry_add_rdn_csn(Slapi_Entry *e, const CSN *csn) +{ + char *type; + char *value; + int index; + const Slapi_DN *dn= slapi_entry_get_sdn_const(e); + Slapi_RDN *rdn= slapi_rdn_new_sdn(dn); + index= slapi_rdn_get_first(rdn, &type, &value); + while(index!=-1) + { + Slapi_Attr *a= NULL; + Slapi_Value *v= NULL; + entry_attr_find_wsi(e, type, &a); + if(a!=NULL) + { + struct berval bv; + bv.bv_len= strlen(value); + bv.bv_val= (void*)value; + attr_value_find_wsi(a, &bv, &v); + } + if(v!=NULL) + { + value_update_csn(v,CSN_TYPE_VALUE_DISTINGUISHED,csn); + } + else + { + /* JCM RDN component isn't a present value - this is illegal. */ + } + index= slapi_rdn_get_next(rdn, index, &type, &value); + } + slapi_rdn_free(&rdn); +} + +CSN* +entry_assign_operation_csn ( Slapi_PBlock *pb, Slapi_Entry *e, Slapi_Entry *parententry ) +{ + Slapi_Operation *op; + const CSN *basecsn = NULL; + const CSN *parententry_dncsn = NULL; + CSN *opcsn = NULL; + + slapi_pblock_get ( pb, SLAPI_OPERATION, &op ); + + /* + * The replication pre-op would have set op->o_csngen_handler for + * user requests that are against a replica. + */ + if ( op->o_csngen_handler ) + { + /* + * Sync up the CSN generator so that the new csn is greater + * than the entry's maxcsn and/or the parent's max dncsn. + */ + if ( e ) + { + basecsn = entry_get_maxcsn ( e ); + } + if ( parententry ) + { + parententry_dncsn = entry_get_dncsn ( parententry ); + if ( csn_compare ( parententry_dncsn, basecsn ) > 0 ) + { + basecsn = parententry_dncsn; + } + } + opcsn = op->o_csngen_handler ( pb, basecsn ); + + if (NULL != opcsn) + { + operation_set_csn (op, opcsn); + } + } + + return opcsn; +} + +/* + * Purge state information from the entry older than csnUpTo + * + * if csnUpTo is NULL, get rid of all the CSN related info. + * if csnUpTo is non-NULL, purge all info older than csnUpTo + */ +void +entry_purge_state_information(Slapi_Entry *e, const CSN *csnUpTo) +{ + Slapi_Attr *a=NULL; + + PR_ASSERT(e!=NULL); + + for(a = e->e_attrs; NULL != a; a = a->a_next) + { + /* + * we are passing in the entry so that we may be able to "optimize" + * the csn related information and roll it up higher to the level + * of entry + */ + attr_purge_state_information(e, a, csnUpTo); + } + for(a = e->e_deleted_attrs; NULL != a; a = a->a_next) + { + /* + * we are passing in the entry so that we may be able to "optimize" + * the csn related information and roll it up higher to the level + * of entry + */ + attr_purge_state_information(e, a, csnUpTo); + } +} + +/* + * Look for the attribute on the present and deleted attribute lists. + */ +int +entry_attr_find_wsi(Slapi_Entry *e, const char *type, Slapi_Attr **a) +{ + int retVal= ATTRIBUTE_NOTFOUND; + + PR_ASSERT(e!=NULL); + PR_ASSERT(type!=NULL); + PR_ASSERT(a!=NULL); + + /* Look on the present attribute list */ + *a= attrlist_find(e->e_attrs,type); + if(*a!=NULL) + { + /* The attribute is present */ + retVal= ATTRIBUTE_PRESENT; + } + else + { + /* Maybe the attribue was deleted... */ + *a= attrlist_find(e->e_deleted_attrs,type); + if(*a!=NULL) + { + /* The attribute is deleted */ + retVal= ATTRIBUTE_DELETED; + } + else + { + /* The attribute was not found */ + retVal= ATTRIBUTE_NOTFOUND; + } + } + return retVal; +} + +/* + * Add the attribute to the deleted attribute list. + * + * Consumes the attribute. + */ +int +entry_add_deleted_attribute_wsi(Slapi_Entry *e, Slapi_Attr *a) +{ + PR_ASSERT( e!=NULL ); + PR_ASSERT( a!=NULL ); + attrlist_add(&e->e_deleted_attrs, a); + return 0; +} + +/* + * Add the attribute to the present attribute list. + * + * Consumes the attribute. + */ +int +entry_add_present_attribute_wsi(Slapi_Entry *e, Slapi_Attr *a) +{ + PR_ASSERT( e!=NULL ); + PR_ASSERT( a!=NULL ); + attrlist_add(&e->e_attrs, a); + return 0; +} + +/* + * Add a list of values to the attribute, whilst maintaining state information. + * + * Preserves LDAP Information Model constraints, + * returning an LDAP result code. + */ +static int +entry_add_present_values_wsi(Slapi_Entry *e, const char *type, struct berval **bervals, const CSN *csn, int urp, long flags) +{ + int retVal= LDAP_SUCCESS; + Slapi_Value **valuestoadd = NULL; + valuearray_init_bervalarray(bervals,&valuestoadd); /* JCM SLOW FUNCTION */ + if(!valuearray_isempty(valuestoadd)) + { + Slapi_Attr *a= NULL; + long a_flags_orig; + int attr_state= entry_attr_find_wsi(e, type, &a); + if (ATTRIBUTE_NOTFOUND == attr_state) + { + /* Create a new attribute */ + a = slapi_attr_new(); + slapi_attr_init(a, type); + attrlist_add(&e->e_attrs, a); + } + a_flags_orig = a->a_flags; + a->a_flags |= flags; + if(urp) + { + /* + * Consolidate a->a_present_values and the pending values: + * Delete the pending values from a->a_present_values + * and transfer their csnsets to valuestoadd. + */ + valueset_remove_valuearray (&a->a_present_values, a, valuestoadd, + SLAPI_VALUE_FLAG_IGNOREERROR | + SLAPI_VALUE_FLAG_PRESERVECSNSET, NULL); + /* + * Consolidate a->a_deleted_values and the pending values + * similarly. + */ + valueset_remove_valuearray (&a->a_deleted_values, a, valuestoadd, + SLAPI_VALUE_FLAG_IGNOREERROR | + SLAPI_VALUE_FLAG_PRESERVECSNSET, NULL); + + /* Append the pending values to a->a_present_values */ + valuearray_update_csn (valuestoadd,CSN_TYPE_VALUE_UPDATED,csn); + valueset_add_valuearray_ext(&a->a_present_values, valuestoadd, SLAPI_VALUE_FLAG_PASSIN); + slapi_ch_free ( (void **)&valuestoadd ); + + /* + * Now delete non-RDN values from a->a_present_values; and + * restore possible RDN values from a->a_deleted_values + */ + resolve_attribute_state(e, a, attr_state); + retVal= LDAP_SUCCESS; + } + else + { + Slapi_Value **deletedvalues= NULL; + switch(attr_state) + { + case ATTRIBUTE_PRESENT: + /* The attribute is already on the present list */ + break; + case ATTRIBUTE_DELETED: + /* Move the deleted attribute onto the present list */ + entry_deleted_attribute_to_present_attribute(e, a); + break; + case ATTRIBUTE_NOTFOUND: + /* No-op - attribute was initialized & added to entry above */ + break; + } + /* Check if any of the values to be added are on the deleted list */ + valueset_remove_valuearray(&a->a_deleted_values, a, valuestoadd, SLAPI_VALUE_FLAG_IGNOREERROR,&deletedvalues); /* JCM Check return code */ + if(deletedvalues!=NULL && deletedvalues[0]!=NULL) + { + /* Some of the values to be added were on the deleted list */ + Slapi_Value **v= NULL; + Slapi_ValueSet vs; + /* Add each deleted value to the present list */ + valuearray_update_csn(deletedvalues,CSN_TYPE_VALUE_UPDATED,csn); + valueset_add_valuearray_ext(&a->a_present_values, deletedvalues, SLAPI_VALUE_FLAG_PASSIN); + /* Remove the deleted values from the values to add */ + valueset_set_valuearray_passin(&vs,valuestoadd); + valueset_remove_valuearray(&vs, a, deletedvalues, SLAPI_VALUE_FLAG_IGNOREERROR, &v); + valuestoadd= valueset_get_valuearray(&vs); + valuearray_free(&v); + slapi_ch_free((void **)&deletedvalues); + } + valuearray_update_csn(valuestoadd,CSN_TYPE_VALUE_UPDATED,csn); + retVal= attr_add_valuearray(a, valuestoadd, slapi_entry_get_dn_const(e)); + valuearray_free(&valuestoadd); + } + a->a_flags = a_flags_orig; + } + return(retVal); +} + +/* + * Delete a list of values from an attribute, whilst maintaining state information. + * + * Preserves LDAP Information Model constraints, + * returning an LDAP result code. + */ +static int +entry_delete_present_values_wsi(Slapi_Entry *e, const char *type, struct berval **vals, const CSN *csn, int urp, int mod_op) +{ + int retVal= LDAP_SUCCESS; + Slapi_Attr *a= NULL; + int attr_state= entry_attr_find_wsi(e, type, &a); + if(attr_state==ATTRIBUTE_PRESENT || (attr_state==ATTRIBUTE_DELETED && urp)) + { + /* The attribute is on the present list, or the deleted list and we're doing URP */ + if ( vals == NULL || vals[0] == NULL ) + { + /* delete the entire attribute */ + LDAPDebug( LDAP_DEBUG_ARGS, "removing entire attribute %s\n", type, 0, 0 ); + attr_set_deletion_csn(a,csn); + if(urp) + { + resolve_attribute_state(e, a, attr_state); /* ABSOLVED */ + } + else + { + if(!slapi_attr_flag_is_set(a,SLAPI_ATTR_FLAG_SINGLE)) + { + /* We don't maintain a deleted value list for single valued attributes */ + valueset_add_valueset(&a->a_deleted_values, &a->a_present_values); /* JCM Would be better to passin the valuestodelete */ + } + slapi_valueset_done(&a->a_present_values); + entry_present_attribute_to_deleted_attribute(e, a); + } + retVal= LDAP_SUCCESS; /* This Operation always succeeds when the attribute is Present */ + } + else + { + /* delete some specific values */ + Slapi_Value **valuestodelete= NULL; + valuearray_init_bervalarray(vals,&valuestodelete); /* JCM SLOW FUNCTION */ + if(urp) + { + Slapi_Value **valuesupdated= NULL; + valueset_update_csn_for_valuearray(&a->a_present_values, a, valuestodelete, CSN_TYPE_VALUE_DELETED, csn, &valuesupdated); + /* if we removed the last value, we need to mark the attribute as deleted + the resolve_attribute_state() code will "resurrect" the attribute if + there are present values with a later CSN - otherwise, even though + the value will be updated with a VDCSN which is later than the VUCSN, + the attribute will not be deleted */ + if(slapi_attr_flag_is_set(a,SLAPI_ATTR_FLAG_SINGLE) && valuesupdated && + *valuesupdated) + { + attr_set_deletion_csn(a,csn); + } + valuearray_free(&valuesupdated); + valueset_update_csn_for_valuearray(&a->a_deleted_values, a, valuestodelete, CSN_TYPE_VALUE_DELETED, csn, &valuesupdated); + valuearray_free(&valuesupdated); + valuearray_update_csn(valuestodelete,CSN_TYPE_VALUE_DELETED,csn); + valueset_add_valuearray_ext(&a->a_deleted_values, valuestodelete, SLAPI_VALUE_FLAG_PASSIN); + /* all the elements in valuestodelete are passed; + * should free valuestodelete only (don't call valuearray_free) + * [622023] */ + slapi_ch_free((void **)&valuestodelete); + resolve_attribute_state(e, a, attr_state); + retVal= LDAP_SUCCESS; + } + else + { + Slapi_Value **deletedvalues= NULL; + retVal= valueset_remove_valuearray(&a->a_present_values, a, valuestodelete, 0 /* Do Not Ignore Errors */,&deletedvalues); + if(retVal==LDAP_SUCCESS && deletedvalues != NULL) + { + if(!slapi_attr_flag_is_set(a,SLAPI_ATTR_FLAG_SINGLE)) + { + /* We don't maintain a deleted value list for single valued attributes */ + /* Add each deleted value to the deleted set */ + valuearray_update_csn(deletedvalues,CSN_TYPE_VALUE_DELETED,csn); + valueset_add_valuearray_ext(&a->a_deleted_values, deletedvalues, SLAPI_VALUE_FLAG_PASSIN); + slapi_ch_free((void **)&deletedvalues); + } + else { + valuearray_free(&deletedvalues); + } + if(valueset_isempty(&a->a_present_values)) + { + /* There are no present values, so move the + * attribute to the deleted attribute list. */ + entry_present_attribute_to_deleted_attribute(e, a); + } + } + else if (retVal != LDAP_SUCCESS) + { + /* Failed + * - Duplicate value + * - Value not found + * - Operations error + */ + if ( retVal==LDAP_OPERATIONS_ERROR ) + { + LDAPDebug( LDAP_DEBUG_ANY, "Possible existing duplicate " + "value for attribute type %s found in " + "entry %s\n", a->a_type, slapi_entry_get_dn_const(e), 0 ); + } + } + valuearray_free(&valuestodelete); + } + } + } + else if (attr_state==ATTRIBUTE_DELETED) + { + retVal= LDAP_NO_SUCH_ATTRIBUTE; + } + else if (attr_state==ATTRIBUTE_NOTFOUND) + { + if (!urp) + { + /* Only warn if not urping */ + LDAPDebug( LDAP_DEBUG_ARGS, "could not find attribute %s\n", type, 0, 0 ); + } + retVal= LDAP_NO_SUCH_ATTRIBUTE; + if (LDAP_MOD_REPLACE == mod_op) + { + /* Create a new attribute and set the adcsn */ + Slapi_Attr *a = slapi_attr_new(); + slapi_attr_init(a, type); + /* According to URP there should be an adcsn. + * According to Tests, there should not + * since the attribute never existed + * Tests 26 and 27 of MMRepl state. */ + if (urp) + attr_set_deletion_csn(a,csn); + attrlist_add(&e->e_attrs, a); + } + } + return( retVal ); +} + +/* + * Replace all the values of an attribute with a list of attribute values. + * + * Preserves LDAP Information Model constraints, + * returning an LDAP result code. + */ +static int +entry_replace_present_values_wsi(Slapi_Entry *e, const char *type, struct berval **vals, const CSN *csn, int urp) +{ + /* + * Remove all existing values. + */ + entry_delete_present_values_wsi(e, type, NULL /* Delete all values */, csn, urp, LDAP_MOD_REPLACE); + + /* + * Add the new values. If there are no new values, + * slapi_entry_add_values() returns LDAP_SUCCESS and so the + * attribute remains deleted (which is the correct outcome). + */ + return( entry_add_present_values_wsi( e, type, vals, csn, urp, SLAPI_ATTR_FLAG_CMP_BITBYBIT )); +} + +/* + * Applies the modification to the entry whilst + * maintaining state information. + */ +int +entry_apply_mod_wsi(Slapi_Entry *e, const LDAPMod *mod, const CSN *csn, int urp) +{ + int retVal= LDAP_SUCCESS; + int i; + + switch ( mod->mod_op & ~LDAP_MOD_BVALUES ) + { + case LDAP_MOD_ADD: + LDAPDebug( LDAP_DEBUG_ARGS, " add: %s\n", mod->mod_type, 0, 0 ); + retVal = entry_add_present_values_wsi( e, mod->mod_type, mod->mod_bvalues, csn, urp, 0 ); + break; + + case LDAP_MOD_DELETE: + LDAPDebug( LDAP_DEBUG_ARGS, " delete: %s\n", mod->mod_type, 0, 0 ); + retVal = entry_delete_present_values_wsi( e, mod->mod_type, mod->mod_bvalues, csn, urp, mod->mod_op ); + break; + + case LDAP_MOD_REPLACE: + LDAPDebug( LDAP_DEBUG_ARGS, " replace: %s\n", mod->mod_type, 0, 0 ); + retVal = entry_replace_present_values_wsi( e, mod->mod_type, mod->mod_bvalues, csn, urp ); + break; + } + for ( i = 0; mod->mod_bvalues != NULL && mod->mod_bvalues[i] != NULL; i++ ) + { + LDAPDebug( LDAP_DEBUG_ARGS, " %s: %s\n", mod->mod_type, mod->mod_bvalues[i]->bv_val, 0 ); + } + LDAPDebug( LDAP_DEBUG_ARGS, " -\n", 0, 0, 0 ); + + return retVal; +} + +/* + * Applies the set of modifications to the entry whilst + * maintaining state information. + */ +int +entry_apply_mods_wsi(Slapi_Entry *e, Slapi_Mods *smods, const CSN *csn, int urp) +{ + int retVal= LDAP_SUCCESS; + LDAPMod *mod; + + LDAPDebug( LDAP_DEBUG_TRACE, "=> entry_apply_mods_wsi\n", 0, 0, 0 ); + + mod = slapi_mods_get_first_mod(smods); + while(NULL!=mod && retVal==LDAP_SUCCESS) + { + if(csn!=NULL) + { + retVal= entry_apply_mod_wsi(e, mod, csn, urp); + } + else + { + retVal= entry_apply_mod(e, mod); + } + mod = slapi_mods_get_next_mod(smods); + } + + LDAPDebug( LDAP_DEBUG_TRACE, "<= entry_apply_mods_wsi %d\n", retVal, 0, 0 ); + + return retVal; +} + +/* + * This code implements a computed attribute called 'nscpEntryWSI'. + * By specifically asking for this attribute the client will recieve + * an LDIF dump of the entry with all its state information. + * + * JCM - Security... Only for the Directory Manager. + */ +static const char *nscpEntryWSI= "nscpEntryWSI"; +/* + */ +static int +entry_compute_nscpentrywsi(computed_attr_context *c,char* type,Slapi_Entry *e,slapi_compute_output_t outputfn) +{ + int rc = 0; + + if ( strcasecmp(type, nscpEntryWSI ) == 0) + { + /* If not, we return it as zero */ + char *es; + char *s; + char *p; + int len; + Slapi_Attr our_attr; + slapi_attr_init(&our_attr, nscpEntryWSI); + our_attr.a_flags = SLAPI_ATTR_FLAG_OPATTR; + es= slapi_entry2str_with_options(e, &len, SLAPI_DUMP_STATEINFO | SLAPI_DUMP_UNIQUEID | SLAPI_DUMP_NOWRAP); + s= es; + p= ldif_getline( &s ); + while(p!=NULL) + { + Slapi_Value *v; + char *t, *d; + /* Strip out the Continuation Markers (JCM - I think that NOWRAP means we don't need to do this any more)*/ + for ( t = p, d = p; *t; t++ ) + { + if ( *t != 0x01 ) + *d++ = *t; + } + *d = '\0'; + v= slapi_value_new_string(p); + slapi_attr_add_value(&our_attr,v); + slapi_value_free(&v); + p= ldif_getline( &s ); + } + slapi_ch_free((void**)&es); + rc = (*outputfn) (c, &our_attr, e); + attr_done(&our_attr); + return (rc); + } + + return -1; /* I see no ships */ +} + + +int +entry_computed_attr_init() +{ + slapi_compute_add_evaluator(entry_compute_nscpentrywsi); + return 0; +} + +static void +purge_attribute_state_multi_valued(const Slapi_Attr *a, Slapi_Value *v) +{ + const CSN *vdcsn= value_get_csn(v,CSN_TYPE_VALUE_DELETED); + const CSN *vucsn= value_get_csn(v,CSN_TYPE_VALUE_UPDATED); + if(vdcsn && csn_compare(vdcsn,vucsn)<0) + { + value_remove_csn(v,CSN_TYPE_VALUE_DELETED); + } +} + +/* + * utility function for value_distinguished_at_csn... + */ +static const CSN * +vdac_sniff_value(Slapi_ValueSet *vs, const Slapi_Value *v, const CSN *csn, const CSN *most_recent_mdcsn) +{ + const CSN *mdcsn= value_get_csn(v,CSN_TYPE_VALUE_DISTINGUISHED); + if(mdcsn!=NULL) + { + /* This value was/is distinguished... */ + if(csn_compare(csn,most_recent_mdcsn)<0) + { + /* ...and was distinguished before the point in time we're interested in... */ + int r= csn_compare(mdcsn,most_recent_mdcsn); + if(r>0) + { + /* ...and is the most recent MDCSN we've seen thus far. */ + slapi_valueset_done(vs); + slapi_valueset_add_value(vs,v); + most_recent_mdcsn= mdcsn; + } + else if(r==0) + { + /* ...and is as recent as the last most recent MDCSN we've seen thus far. */ + /* Must have been a multi-valued RDN */ + slapi_valueset_add_value(vs,v); + } + } + } + return most_recent_mdcsn; +} + +/* + * utility function for value_distinguished_at_csn... + */ +static const CSN * +vdac_sniff_attribute(Slapi_ValueSet *vs, Slapi_Attr *a, const CSN *csn, const CSN *most_recent_mdcsn) +{ + Slapi_Value *v; + int i= slapi_attr_first_value( a, &v ); + while(i!=-1) + { + most_recent_mdcsn= vdac_sniff_value( vs, v, csn, most_recent_mdcsn ); + i= slapi_attr_next_value( a, i, &v ); + } + i= attr_first_deleted_value( a, &v ); + while(i!=-1) + { + most_recent_mdcsn= vdac_sniff_value( vs, v, csn, most_recent_mdcsn ); + i= attr_next_deleted_value( a, i, &v ); + } + return most_recent_mdcsn; +} + +/* + * utility function for value_distinguished_at_csn... + * + * Return the set of values that made up the RDN at or before the csn point. + */ +static const CSN * +distinguished_values_at_csn(const Slapi_Entry *e, const CSN *csn, Slapi_ValueSet *vs) +{ + const CSN *most_recent_mdcsn= NULL; + Slapi_Attr *a; + int i= slapi_entry_first_attr( e, &a ); + while(i!=-1) + { + most_recent_mdcsn= vdac_sniff_attribute( vs, a, csn, most_recent_mdcsn); + i= slapi_entry_next_attr( e, a, &a ); + } + i= entry_first_deleted_attribute( e, &a ); + while(i!=-1) + { + most_recent_mdcsn= vdac_sniff_attribute( vs, a, csn, most_recent_mdcsn); + i= entry_next_deleted_attribute( e, &a ); + } + return most_recent_mdcsn; +} + +/* + * Work out if the value was distinguished at time csn. + */ +static int +value_distinguished_at_csn(const Slapi_Entry *e, const Slapi_Attr *original_attr, Slapi_Value *original_value, const CSN *csn) +{ + int r= 0; + const CSN *mdcsn= value_get_csn(original_value,CSN_TYPE_VALUE_DISTINGUISHED); + if(mdcsn!=NULL) + { + /* + * Oh bugger. This means that we have to work out what the RDN components + * were at this point in time. This is non-trivial since we must walk + * through all the present and deleted attributes and their present and + * deleted values. Slow :-( + */ + Slapi_ValueSet *vs= slapi_valueset_new(); + const CSN *most_recent_mdcsn= distinguished_values_at_csn(e, csn, vs); + /* + * We now know what the RDN components were at the point in time we're interested in. + * And the question we need to answer is :- + * 'Was the provided value one of those RDN components?' + */ + if(most_recent_mdcsn!=NULL) + { + Slapi_Value *v; + int i= slapi_valueset_first_value( vs, &v ); + while(i!=-1) + { + if(slapi_value_compare(original_attr, original_value, v)==0) + { + /* This value was distinguished at the time in question. */ + r= 1; + i= -1; + } + else + { + i= slapi_valueset_next_value( vs, i, &v ); + } + } + } + slapi_valueset_free(vs); + } + else + { + /* This value has never been distinguished */ + r= 0; + } + return r; +} + +static void +resolve_attribute_state_multi_valued(Slapi_Entry *e, Slapi_Attr *a, int attribute_state) +{ + int i; + const CSN *adcsn= attr_get_deletion_csn(a); + Slapi_ValueSet *vs= valueset_dup(&a->a_present_values); /* JCM This is slow... but otherwise we end up iterating through a changing array */ + Slapi_Value *v; + + /* Loop over the present attribute values */ + i= slapi_valueset_first_value( vs, &v ); + while(v!=NULL) + { + const CSN *vdcsn; + const CSN *vucsn; + const CSN *deletedcsn; + /* This call ensures that the value does not contain a deletion_csn + * which is before the presence_csn or distinguished_csn of the value. + */ + purge_attribute_state_multi_valued(a, v); + vdcsn= value_get_csn(v, CSN_TYPE_VALUE_DELETED); + vucsn= value_get_csn(v, CSN_TYPE_VALUE_UPDATED); + deletedcsn= csn_max(vdcsn, adcsn); + if(csn_compare(vucsn,deletedcsn)<0) /* check if the attribute or value was deleted after the value was last updated */ + { + if(!value_distinguished_at_csn(e, a, v, deletedcsn)) + { + entry_present_value_to_deleted_value(a,v); + } + } + i= slapi_valueset_next_value( vs, i, &v ); + } + slapi_valueset_free(vs); + + /* Loop over the deleted attribute values */ + vs= valueset_dup(&a->a_deleted_values); /* JCM This is slow... but otherwise we end up iterating through a changing array */ + i= slapi_valueset_first_value( vs, &v ); + while(v!=NULL) + { + const CSN *vdcsn; + const CSN *vucsn; + const CSN *deletedcsn; + /* This call ensures that the value does not contain a deletion_csn which is before the presence_csn or distinguished_csn of the value. */ + purge_attribute_state_multi_valued(a, v); + vdcsn= value_get_csn(v, CSN_TYPE_VALUE_DELETED); + vucsn= value_get_csn(v, CSN_TYPE_VALUE_UPDATED); + deletedcsn= csn_max(vdcsn, adcsn); + if((csn_compare(vucsn,deletedcsn)>=0) || /* check if the attribute or value was deleted after the value was last updated */ + value_distinguished_at_csn(e, a, v, deletedcsn)) + { + entry_deleted_value_to_present_value(a,v); + } + i= slapi_valueset_next_value( vs, i, &v ); + } + slapi_valueset_free(vs); + + if(valueset_isempty(&a->a_present_values)) + { + if(attribute_state==ATTRIBUTE_PRESENT) + { + entry_present_attribute_to_deleted_attribute(e, a); + } + } + else + { + if(attribute_state==ATTRIBUTE_DELETED) + { + entry_deleted_attribute_to_present_attribute(e, a); + } + } +} + +static void +resolve_attribute_state_single_valued(Slapi_Entry *e, Slapi_Attr *a, int attribute_state) +{ + Slapi_Value *current_value= NULL; + Slapi_Value *pending_value= NULL; + Slapi_Value *new_value= NULL; + const CSN *current_value_vucsn; + const CSN *pending_value_vucsn; + const CSN *adcsn; + int i; + + /* + * this call makes sure that the attribute does not have a pending_value + * or deletion_csn which is before the current_value. + */ + i= slapi_attr_first_value(a,¤t_value); + if(i!=-1) + { + slapi_attr_next_value(a,i,&new_value); + } + attr_first_deleted_value(a,&pending_value); + + /* purge_attribute_state_single_valued */ + adcsn= attr_get_deletion_csn(a); + current_value_vucsn= value_get_csn(current_value, CSN_TYPE_VALUE_UPDATED); + pending_value_vucsn= value_get_csn(pending_value, CSN_TYPE_VALUE_UPDATED); + if((pending_value!=NULL && (csn_compare(adcsn, pending_value_vucsn)<0)) || + (pending_value==NULL && (csn_compare(adcsn, current_value_vucsn)<0))) + { + attr_set_deletion_csn(a,NULL); + adcsn= NULL; + } + + if(new_value==NULL) + { + /* check if the pending value should become the current value */ + if(pending_value!=NULL) + { + if(!value_distinguished_at_csn(e,a,current_value,pending_value_vucsn)) + { + /* attribute.current_value = attribute.pending_value; */ + /* attribute.pending_value = NULL; */ + entry_present_value_to_zapped_value(a,current_value); + entry_deleted_value_to_present_value(a,pending_value); + current_value= pending_value; + pending_value= NULL; + current_value_vucsn= pending_value_vucsn; + pending_value_vucsn= NULL; + } + } + /* check if the current value should be deleted */ + if(current_value!=NULL) + { + if(csn_compare(adcsn,current_value_vucsn)>0) /* check if the attribute was deleted after the value was last updated */ + { + if(!value_distinguished_at_csn(e,a,current_value,current_value_vucsn)) + { + entry_present_value_to_zapped_value(a,current_value); + current_value= NULL; + current_value_vucsn= NULL; + } + } + } + } + else /* addition of a new value */ + { + const CSN *new_value_vucsn= value_get_csn(new_value,CSN_TYPE_VALUE_UPDATED); + if(csn_compare(new_value_vucsn,current_value_vucsn)<0) + { + /* + * if the new value was distinguished at the time the current value was added + * then the new value should become current + */ + if(value_distinguished_at_csn(e,a,new_value,current_value_vucsn)) + { + /* attribute.pending_value = attribute.current_value */ + /* attribute.current_value = new_value */ + if(pending_value==NULL) + { + entry_present_value_to_deleted_value(a,current_value); + } + else + { + entry_present_value_to_zapped_value(a,current_value); + } + pending_value= current_value; + current_value= new_value; + new_value= NULL; + pending_value_vucsn= current_value_vucsn; + current_value_vucsn= new_value_vucsn; + } + else + { + /* new_value= NULL */ + entry_present_value_to_zapped_value(a, new_value); + new_value= NULL; + } + } + else /* new value is after the current value */ + { + if(!value_distinguished_at_csn(e, a, current_value, new_value_vucsn)) + { + /* attribute.current_value = new_value */ + entry_present_value_to_zapped_value(a, current_value); + current_value= new_value; + new_value= NULL; + current_value_vucsn= new_value_vucsn; + } + else /* value is distinguished - check if we should replace the current pending value */ + { + if(csn_compare(new_value_vucsn, pending_value_vucsn)>0) + { + /* attribute.pending_value = new_value */ + entry_deleted_value_to_zapped_value(a,pending_value); + entry_present_value_to_deleted_value(a,new_value); + pending_value= new_value; + new_value= NULL; + pending_value_vucsn= new_value_vucsn; + } + } + } + } + + /* + * This call ensures that the attribute does not have a pending_value + * or a deletion_csn that is earlier than the current_value. + */ + /* purge_attribute_state_single_valued */ + if((pending_value!=NULL && (csn_compare(adcsn, pending_value_vucsn)<0)) || + (pending_value==NULL && (csn_compare(adcsn, current_value_vucsn)<0))) + { + attr_set_deletion_csn(a,NULL); + adcsn= NULL; + } + + /* set attribute state */ + if(current_value==NULL) + { + if(attribute_state==ATTRIBUTE_PRESENT) + { + entry_present_attribute_to_deleted_attribute(e, a); + } + } + else + { + if(attribute_state==ATTRIBUTE_DELETED) + { + entry_deleted_attribute_to_present_attribute(e, a); + } + } +} + +static void +resolve_attribute_state(Slapi_Entry *e, Slapi_Attr *a, int attribute_state) +{ + if(slapi_attr_flag_is_set(a,SLAPI_ATTR_FLAG_SINGLE)) + { + resolve_attribute_state_single_valued(e,a,attribute_state); + } + else + { + resolve_attribute_state_multi_valued(e,a,attribute_state); + } +} |