/* attr.c - backend routines for dealing with attributes */ /* $OpenLDAP$ */ /* This work is part of OpenLDAP Software . * * Copyright 2000-2005 The OpenLDAP Foundation. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted only as authorized by the OpenLDAP * Public License. * * A copy of this license is available in the file LICENSE in the * top-level directory of the distribution or, alternatively, at * . */ #include "portable.h" #include #include #include #include "slap.h" #include "back-bdb.h" #include "lutil.h" static int ainfo_type_cmp( const void *v_desc, const void *v_a ) { const AttributeDescription *desc = v_desc; const AttrInfo *a = v_a; return SLAP_PTRCMP(desc, a->ai_desc); } static int ainfo_cmp( const void *v_a, const void *v_b ) { const AttrInfo *a = v_a, *b = v_b; return SLAP_PTRCMP(a->ai_desc, b->ai_desc); } AttrInfo * bdb_attr_mask( struct bdb_info *bdb, AttributeDescription *desc ) { return avl_find( bdb->bi_attrs, desc, ainfo_type_cmp ); } int bdb_attr_index_config( struct bdb_info *bdb, const char *fname, int lineno, int argc, char **argv ) { int rc; int i; slap_mask_t mask; char **attrs; char **indexes = NULL; attrs = ldap_str2charray( argv[0], "," ); if( attrs == NULL ) { fprintf( stderr, "%s: line %d: " "no attributes specified: %s\n", fname, lineno, argv[0] ); return LDAP_PARAM_ERROR; } if ( argc > 1 ) { indexes = ldap_str2charray( argv[1], "," ); if( indexes == NULL ) { fprintf( stderr, "%s: line %d: " "no indexes specified: %s\n", fname, lineno, argv[1] ); return LDAP_PARAM_ERROR; } } if( indexes == NULL ) { mask = bdb->bi_defaultmask; } else { mask = 0; for ( i = 0; indexes[i] != NULL; i++ ) { slap_mask_t index; rc = slap_str2index( indexes[i], &index ); if( rc != LDAP_SUCCESS ) { fprintf( stderr, "%s: line %d: " "index type \"%s\" undefined\n", fname, lineno, indexes[i] ); return LDAP_PARAM_ERROR; } mask |= index; } } if( !mask ) { fprintf( stderr, "%s: line %d: " "no indexes selected\n", fname, lineno ); return LDAP_PARAM_ERROR; } for ( i = 0; attrs[i] != NULL; i++ ) { AttrInfo *a; AttributeDescription *ad; const char *text; #ifdef LDAP_COMP_MATCH ComponentReference* cr = NULL; AttrInfo *a_cr = NULL; #endif if( strcasecmp( attrs[i], "default" ) == 0 ) { bdb->bi_defaultmask |= mask; continue; } #ifdef LDAP_COMP_MATCH if ( is_component_reference( attrs[i] ) ) { rc = extract_component_reference( attrs[i], &cr ); if ( rc != LDAP_SUCCESS ) { fprintf( stderr, "%s: line %d: " "index component reference\"%s\" undefined\n", fname, lineno, attrs[i] ); return rc; } cr->cr_indexmask = mask; /* * After extracting a component reference * only the name of a attribute will be remaining */ } else { cr = NULL; } #endif a = (AttrInfo *) ch_malloc( sizeof(AttrInfo) ); #ifdef LDAP_COMP_MATCH a->ai_cr = NULL; #endif ad = NULL; rc = slap_str2ad( attrs[i], &ad, &text ); if( rc != LDAP_SUCCESS ) { fprintf( stderr, "%s: line %d: " "index attribute \"%s\" undefined\n", fname, lineno, attrs[i] ); return rc; } if( slap_ad_is_binary( ad ) ) { fprintf( stderr, "%s: line %d: " "index of attribute \"%s\" disallowed\n", fname, lineno, attrs[i] ); return LDAP_UNWILLING_TO_PERFORM; } if( IS_SLAP_INDEX( mask, SLAP_INDEX_APPROX ) && !( ad->ad_type->sat_approx && ad->ad_type->sat_approx->smr_indexer && ad->ad_type->sat_approx->smr_filter ) ) { fprintf( stderr, "%s: line %d: " "approx index of attribute \"%s\" disallowed\n", fname, lineno, attrs[i] ); return LDAP_INAPPROPRIATE_MATCHING; } if( IS_SLAP_INDEX( mask, SLAP_INDEX_EQUALITY ) && !( ad->ad_type->sat_equality && ad->ad_type->sat_equality->smr_indexer && ad->ad_type->sat_equality->smr_filter ) ) { fprintf( stderr, "%s: line %d: " "equality index of attribute \"%s\" disallowed\n", fname, lineno, attrs[i] ); return LDAP_INAPPROPRIATE_MATCHING; } if( IS_SLAP_INDEX( mask, SLAP_INDEX_SUBSTR ) && !( ad->ad_type->sat_substr && ad->ad_type->sat_substr->smr_indexer && ad->ad_type->sat_substr->smr_filter ) ) { fprintf( stderr, "%s: line %d: " "substr index of attribute \"%s\" disallowed\n", fname, lineno, attrs[i] ); return LDAP_INAPPROPRIATE_MATCHING; } Debug( LDAP_DEBUG_CONFIG, "index %s 0x%04lx\n", ad->ad_cname.bv_val, mask, 0 ); a->ai_desc = ad; if ( bdb->bi_flags & BDB_IS_OPEN ) { a->ai_indexmask = 0; a->ai_newmask = mask; } else { a->ai_indexmask = mask; a->ai_newmask = 0; } #ifdef LDAP_COMP_MATCH if ( cr ) { a_cr = avl_find( bdb->bi_attrs, ad, ainfo_type_cmp ); if ( a_cr ) { /* * AttrInfo is already in AVL * just add the extracted component reference * in the AttrInfo */ rc = insert_component_reference( cr, &a_cr->ai_cr ); if ( rc != LDAP_SUCCESS) { fprintf( stderr, " error during inserting component reference in %s ", attrs[i]); return LDAP_PARAM_ERROR; } continue; } else { rc = insert_component_reference( cr, &a->ai_cr ); if ( rc != LDAP_SUCCESS) { fprintf( stderr, " error during inserting component reference in %s ", attrs[i]); return LDAP_PARAM_ERROR; } } } #endif rc = avl_insert( &bdb->bi_attrs, (caddr_t) a, ainfo_cmp, avl_dup_error ); if( rc ) { if ( bdb->bi_flags & BDB_IS_OPEN ) { AttrInfo *b = avl_find( bdb->bi_attrs, ad, ainfo_type_cmp ); /* If we were editing this attr, reset it */ b->ai_indexmask &= ~BDB_INDEX_DELETING; /* If this is leftover from a previous add, commit it */ if ( b->ai_newmask ) b->ai_indexmask = b->ai_newmask; b->ai_newmask = a->ai_newmask; ch_free( a ); continue; } fprintf( stderr, "%s: line %d: duplicate index definition " "for attr \"%s\"" SLAPD_CONF_UNKNOWN_IGNORED ".\n", fname, lineno, attrs[i] ); return LDAP_PARAM_ERROR; } } ldap_charray_free( attrs ); if ( indexes != NULL ) ldap_charray_free( indexes ); return LDAP_SUCCESS; } static int bdb_attr_index_unparser( void *v1, void *v2 ) { AttrInfo *ai = v1; BerVarray *bva = v2; struct berval bv; char *ptr; slap_index2bvlen( ai->ai_indexmask, &bv ); if ( bv.bv_len ) { bv.bv_len += ai->ai_desc->ad_cname.bv_len + 1; ptr = ch_malloc( bv.bv_len+1 ); bv.bv_val = lutil_strcopy( ptr, ai->ai_desc->ad_cname.bv_val ); *bv.bv_val++ = ' '; slap_index2bv( ai->ai_indexmask, &bv ); bv.bv_val = ptr; ber_bvarray_add( bva, &bv ); } } static AttributeDescription addef = { NULL, NULL, BER_BVC("default") }; static AttrInfo aidef = { &addef }; void bdb_attr_index_unparse( struct bdb_info *bdb, BerVarray *bva ) { if ( bdb->bi_defaultmask ) { aidef.ai_indexmask = bdb->bi_defaultmask; bdb_attr_index_unparser( &aidef, bva ); } avl_apply( bdb->bi_attrs, bdb_attr_index_unparser, bva, -1, AVL_INORDER ); } static void bdb_attrinfo_free( void *v ) { AttrInfo *ai = v; #ifdef LDAP_COMP_MATCH free( ai->ai_cr ); #endif free( ai ); } void bdb_attr_index_destroy( Avlnode *tree ) { avl_free( tree, bdb_attrinfo_free ); } void bdb_attr_index_free( struct bdb_info *bdb, AttributeDescription *ad ) { AttrInfo *ai; ai = avl_delete( &bdb->bi_attrs, ad, ainfo_type_cmp ); if ( ai ) bdb_attrinfo_free( ai ); } /* Get a list of AttrInfo's to delete */ typedef struct Alist { struct Alist *next; AttrInfo *ptr; } Alist; static int bdb_attrinfo_flush( void *v1, void *arg ) { AttrInfo *ai = v1; if ( ai->ai_indexmask & BDB_INDEX_DELETING ) { Alist **al = arg; Alist *a = ch_malloc( sizeof( Alist )); a->ptr = ai; a->next = *al; *al = a; } return 0; } void bdb_attr_flush( struct bdb_info *bdb ) { Alist *al = NULL, *a2; avl_apply( bdb->bi_attrs, bdb_attrinfo_flush, &al, -1, AVL_INORDER ); while (( a2 = al )) { al = al->next; avl_delete( &bdb->bi_attrs, a2->ptr, ainfo_cmp ); bdb_attrinfo_free( a2->ptr ); ch_free( a2 ); } }