mirror of
https://git.openldap.org/openldap/openldap.git
synced 2024-12-21 03:10:25 +08:00
6f81235386
from IBM India Research (Apurva Kumar) Code review by Pierangelo, Kurt, and Jong.
333 lines
9.1 KiB
C
333 lines
9.1 KiB
C
/* Copyright (c) 2003 by International Business Machines, Inc.
|
|
*
|
|
* International Business Machines, Inc. (hereinafter called IBM) grants
|
|
* permission under its copyrights to use, copy, modify, and distribute this
|
|
* Software with or without fee, provided that the above copyright notice and
|
|
* all paragraphs of this notice appear in all copies, and that the name of IBM
|
|
* not be used in connection with the marketing of any product incorporating
|
|
* the Software or modifications thereof, without specific, written prior
|
|
* permission.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", AND IBM DISCLAIMS ALL WARRANTIES,
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
* PARTICULAR PURPOSE. IN NO EVENT SHALL IBM BE LIABLE FOR ANY SPECIAL,
|
|
* DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER ARISING
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE, EVEN
|
|
* IF IBM IS APPRISED OF THE POSSIBILITY OF SUCH DAMAGES.
|
|
*/
|
|
|
|
#include "portable.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <ac/string.h>
|
|
#include <ac/socket.h>
|
|
|
|
#include "slap.h"
|
|
#include "../back-ldap/back-ldap.h"
|
|
#include "back-meta.h"
|
|
|
|
#ifdef LDAP_CACHING
|
|
#define MAX_ATTR_SETS 500
|
|
static void find_supersets( struct attr_set* attr_sets, int numsets );
|
|
static int compare_sets( struct attr_set* setA, int, int );
|
|
|
|
int
|
|
meta_back_cache_config(
|
|
BackendDB *be,
|
|
const char *fname,
|
|
int lineno,
|
|
int argc,
|
|
char **argv
|
|
)
|
|
{
|
|
struct metainfo *li = ( struct metainfo * )be->be_private;
|
|
|
|
cache_manager* cm = li->cm;
|
|
query_manager* qm = cm->qm;
|
|
QueryTemplate* temp;
|
|
AttributeName* attr_name;
|
|
AttributeName* attrs;
|
|
AttributeName* attrarray;
|
|
const char* text=NULL;
|
|
|
|
int index, i;
|
|
int num;
|
|
|
|
if ( li == NULL ) {
|
|
fprintf( stderr, "%s: line %d: meta backend info is null!\n",
|
|
fname, lineno );
|
|
return 1;
|
|
}
|
|
|
|
if ( strcasecmp( argv[0], "cacheparams" ) == 0 ) {
|
|
struct berval cache_suffix;
|
|
|
|
if ( argc < 6 ) {
|
|
fprintf( stderr, "%s: line %d: missing arguments in \"cacheparams"
|
|
" <thresh_lo> <thresh_hi> <numattrsets> <entry limit> "
|
|
"<cycle_time>\" \n", fname, lineno );
|
|
return( 1 );
|
|
}
|
|
|
|
cm->caching = 1;
|
|
cm->thresh_lo = atoi( argv[1] );
|
|
cm->thresh_hi = atoi( argv[2] );
|
|
if ( cm->thresh_hi <= cm->thresh_lo ) {
|
|
fprintf( stderr, "%s: line %d: <thresh_lo> must be < <thresh_hi> "
|
|
"in \"cacheparams <thresh_lo> <thresh_hi> <numattrsets> "
|
|
"<entry limit> <cycle_time>\" \n", fname, lineno );
|
|
return( 1 );
|
|
}
|
|
|
|
cm->numattrsets = atoi( argv[3] );
|
|
if ( cm->numattrsets > MAX_ATTR_SETS ) {
|
|
fprintf( stderr, "%s: line %d: <numattrsets> must be <= %d in "
|
|
"\"cacheparams <thresh_lo> <thresh_hi> <numattrsets> "
|
|
"<entry limit> <cycle_time>\" \n",
|
|
fname, lineno, MAX_ATTR_SETS );
|
|
return( 1 );
|
|
}
|
|
|
|
cm->num_entries_limit = atoi( argv[4] );
|
|
cm->consistency_cycle_time = atoi( argv[5] );
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1,
|
|
"Total # of attribute sets to be cached = %d\n",
|
|
cm->numattrsets, 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY,
|
|
"Total # of attribute sets to be cached = %d\n",
|
|
cm->numattrsets, 0, 0 );
|
|
#endif
|
|
qm->attr_sets = ( struct attr_set * )malloc( cm->numattrsets *
|
|
sizeof( struct attr_set ));
|
|
for ( i = 0; i < cm->numattrsets; i++ ) {
|
|
qm->attr_sets[i].attrs = NULL;
|
|
}
|
|
rewrite_session( li->rwinfo, "cacheBase", be->be_nsuffix[0].bv_val,
|
|
0, &cache_suffix.bv_val );
|
|
if ( cache_suffix.bv_val != NULL ) {
|
|
cache_suffix.bv_len = strlen( cache_suffix.bv_val );
|
|
} else {
|
|
cache_suffix = be->be_nsuffix[0];
|
|
}
|
|
li->glue_be = select_backend( &cache_suffix, 0, 1 );
|
|
if ( cache_suffix.bv_val != be->be_nsuffix[0].bv_val ) {
|
|
ch_free( cache_suffix.bv_val );
|
|
}
|
|
|
|
} else if ( strcasecmp( argv[0], "attrset" ) == 0 ) {
|
|
if ( argc < 3 ) {
|
|
fprintf( stderr, "%s: line %d: missing arguments in \"attr-set "
|
|
"<index> <attributes>\" line\n", fname, lineno );
|
|
return( 1 );
|
|
}
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1, "Attribute Set # %d\n",
|
|
atoi( argv[1] ), 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY, "Attribute Set # %d\n",
|
|
atoi( argv[1] ), 0, 0 );
|
|
#endif
|
|
if (atoi(argv[1]) >= cm->numattrsets) {
|
|
fprintf( stderr, "%s; line %d index out of bounds \n",
|
|
fname, lineno );
|
|
return 1;
|
|
}
|
|
index = atoi( argv[1] );
|
|
if ( argv[2] && ( strcmp( argv[2], "*" ) != 0 )) {
|
|
for ( i = 2; argv[i] != NULL; i++ ) {
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1, "\t %s\n",
|
|
argv[i], 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY, "\t %s\n",
|
|
argv[i], 0, 0 );
|
|
#endif
|
|
attrs = qm->attr_sets[index].attrs;
|
|
qm->attr_sets[index].attrs = (AttributeName*)realloc(
|
|
attrs, i * sizeof( AttributeName ));
|
|
attr_name = qm->attr_sets[index].attrs + ( i - 2 );
|
|
ber_str2bv( argv[i], strlen(argv[i]), 1,
|
|
&attr_name->an_name);
|
|
attr_name->an_desc = NULL;
|
|
slap_bv2ad( &attr_name->an_name,
|
|
&attr_name->an_desc, &text );
|
|
attr_name++;
|
|
attr_name->an_name.bv_val = NULL;
|
|
attr_name->an_name.bv_len = 0;
|
|
}
|
|
qm->attr_sets[index].count = i - 2;
|
|
}
|
|
} else if ( strcasecmp( argv[0], "addtemplate" ) == 0 ) {
|
|
if ( argc != 4 ) {
|
|
fprintf( stderr, "%s: line %d: missing argument(s) in "
|
|
"\"addtemplate <filter> <proj attr set> <TTL>\" line\n",
|
|
fname, lineno );
|
|
return( 1 );
|
|
}
|
|
if (( i = atoi( argv[2] )) >= cm->numattrsets ) {
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1,
|
|
"%s: line %d, template index invalid\n",
|
|
fname, lineno, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY,
|
|
"%s: line %d, template index invalid\n",
|
|
fname, lineno, 0 );
|
|
#endif
|
|
return 1;
|
|
}
|
|
num = cm->numtemplates;
|
|
if ( num == 0 )
|
|
find_supersets( qm->attr_sets, cm->numattrsets );
|
|
qm->templates = ( QueryTemplate* )realloc( qm->templates,
|
|
( num + 2 ) * sizeof( QueryTemplate ));
|
|
temp = qm->templates + num;
|
|
ldap_pvt_thread_rdwr_init( &temp->t_rwlock );
|
|
temp->query = temp->query_last = NULL;
|
|
temp->ttl = atoi( argv[3] );
|
|
temp->no_of_queries = 0;
|
|
if ( argv[1] == NULL ) {
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1,
|
|
"Templates string not specified "
|
|
"for template %d\n", num, 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY,
|
|
"Templates string not specified "
|
|
"for template %d\n", num, 0, 0 );
|
|
#endif
|
|
return 1;
|
|
}
|
|
temp->querystr = ch_strdup( argv[1] );
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1, "Template:\n", 0, 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY, "Template:\n", 0, 0, 0 );
|
|
#endif
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1, " query template: %s\n",
|
|
temp->querystr, 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY, " query template: %s\n",
|
|
temp->querystr, 0, 0 );
|
|
#endif
|
|
temp->attr_set_index = i;
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1, " attributes: \n", 0, 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY, " attributes: \n", 0, 0, 0 );
|
|
#endif
|
|
if ( attrarray = qm->attr_sets[i].attrs ) {
|
|
for ( i=0; attrarray[i].an_name.bv_val; i++ )
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG( BACK_META, DETAIL1, "\t%s\n",
|
|
attrarray[i].an_name.bv_val, 0, 0 );
|
|
#else
|
|
Debug( LDAP_DEBUG_ANY, "\t%s\n",
|
|
attrarray[i].an_name.bv_val, 0, 0 );
|
|
#endif
|
|
}
|
|
temp++;
|
|
temp->querystr = NULL;
|
|
cm->numtemplates++;
|
|
}
|
|
/* anything else */
|
|
else {
|
|
fprintf( stderr, "%s: line %d: unknown directive \"%s\" in meta "
|
|
"database definition (ignored)\n",
|
|
fname, lineno, argv[0] );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
find_supersets ( struct attr_set* attr_sets, int numsets )
|
|
{
|
|
int num[MAX_ATTR_SETS];
|
|
int i, j, res;
|
|
int* id_array;
|
|
for ( i = 0; i < MAX_ATTR_SETS; i++ )
|
|
num[i] = 0;
|
|
|
|
for ( i = 0; i < numsets; i++ ) {
|
|
attr_sets[i].ID_array = (int*) malloc( sizeof( int ) );
|
|
attr_sets[i].ID_array[0] = -1;
|
|
}
|
|
|
|
for ( i = 0; i < numsets; i++ ) {
|
|
for ( j=i+1; j < numsets; j++ ) {
|
|
res = compare_sets( attr_sets, i, j );
|
|
switch ( res ) {
|
|
case 0:
|
|
break;
|
|
case 3:
|
|
case 1:
|
|
id_array = attr_sets[i].ID_array;
|
|
attr_sets[i].ID_array = (int *) realloc( id_array,
|
|
( num[i] + 2 ) * sizeof( int ));
|
|
attr_sets[i].ID_array[num[i]] = j;
|
|
attr_sets[i].ID_array[num[i]+1] = -1;
|
|
num[i]++;
|
|
if (res == 1)
|
|
break;
|
|
case 2:
|
|
id_array = attr_sets[j].ID_array;
|
|
attr_sets[j].ID_array = (int *) realloc( id_array,
|
|
( num[j] + 2 ) * sizeof( int ));
|
|
attr_sets[j].ID_array[num[j]] = i;
|
|
attr_sets[j].ID_array[num[j]+1] = -1;
|
|
num[j]++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* compares two sets of attributes (indices i and j)
|
|
* returns 0: if neither set is contained in the other set
|
|
* 1: if set i is contained in set j
|
|
* 2: if set j is contained in set i
|
|
* 3: the sets are equivalent
|
|
*/
|
|
|
|
int
|
|
compare_sets(struct attr_set* set, int i, int j)
|
|
{
|
|
int k,l,numI,numJ;
|
|
int common=0;
|
|
int result=0;
|
|
|
|
if (( set[i].attrs == NULL ) && ( set[j].attrs == NULL ))
|
|
return 3;
|
|
|
|
if ( set[i].attrs == NULL )
|
|
return 2;
|
|
|
|
if ( set[j].attrs == NULL )
|
|
return 1;
|
|
|
|
numI = set[i].count;
|
|
numJ = set[j].count;
|
|
|
|
for ( l=0; l < numI; l++ ) {
|
|
for ( k = 0; k < numJ; k++ ) {
|
|
if ( strcmp( set[i].attrs[l].an_name.bv_val,
|
|
set[j].attrs[k].an_name.bv_val ) == 0 )
|
|
common++;
|
|
}
|
|
}
|
|
|
|
if ( common == numI )
|
|
result = 1;
|
|
|
|
if ( common == numJ )
|
|
result += 2;
|
|
|
|
return result;
|
|
}
|
|
#endif
|