Patrick Welche <prlw1@cam.ac.uk>
[netbsd-mini2440.git] / external / bsd / openldap / dist / servers / slapd / back-meta / candidates.c
blob64d4359cf6a8ae034cf0e5c0449bf2f90e542b50
1 /* $OpenLDAP: pkg/ldap/servers/slapd/back-meta/candidates.c,v 1.28.2.5 2008/02/11 23:26:46 kurt Exp $ */
2 /* This work is part of OpenLDAP Software <http://www.openldap.org/>.
4 * Copyright 1999-2008 The OpenLDAP Foundation.
5 * Portions Copyright 2001-2003 Pierangelo Masarati.
6 * Portions Copyright 1999-2003 Howard Chu.
7 * All rights reserved.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted only as authorized by the OpenLDAP
11 * Public License.
13 * A copy of this license is available in the file LICENSE in the
14 * top-level directory of the distribution or, alternatively, at
15 * <http://www.OpenLDAP.org/license.html>.
17 /* ACKNOWLEDGEMENTS:
18 * This work was initially developed by the Howard Chu for inclusion
19 * in OpenLDAP Software and subsequently enhanced by Pierangelo
20 * Masarati.
23 #include "portable.h"
25 #include <stdio.h>
26 #include "ac/string.h"
28 #include "slap.h"
29 #include "../back-ldap/back-ldap.h"
30 #include "back-meta.h"
33 * The meta-directory has one suffix, called <suffix>.
34 * It handles a pool of target servers, each with a branch suffix
35 * of the form <branch X>,<suffix>
37 * When the meta-directory receives a request with a dn that belongs
38 * to a branch, the corresponding target is invoked. When the dn
39 * does not belong to a specific branch, all the targets that
40 * are compatible with the dn are selected as candidates, and
41 * the request is spawned to all the candidate targets
43 * A request is characterized by a dn. The following cases are handled:
44 * - the dn is the suffix: <dn> == <suffix>,
45 * all the targets are candidates (search ...)
46 * - the dn is a branch suffix: <dn> == <branch X>,<suffix>, or
47 * - the dn is a subtree of a branch suffix:
48 * <dn> == <rdn>,<branch X>,<suffix>,
49 * the target is the only candidate.
51 * A possible extension will include the handling of multiple suffixes
56 * returns 1 if suffix is candidate for dn, otherwise 0
58 * Note: this function should never be called if dn is the <suffix>.
60 int
61 meta_back_is_candidate(
62 metatarget_t *mt,
63 struct berval *ndn,
64 int scope )
66 if ( dnIsSuffix( ndn, &mt->mt_nsuffix ) ) {
67 if ( mt->mt_subtree_exclude ) {
68 int i;
70 for ( i = 0; !BER_BVISNULL( &mt->mt_subtree_exclude[ i ] ); i++ ) {
71 if ( dnIsSuffix( ndn, &mt->mt_subtree_exclude[ i ] ) ) {
72 return META_NOT_CANDIDATE;
77 switch ( mt->mt_scope ) {
78 case LDAP_SCOPE_SUBTREE:
79 default:
80 return META_CANDIDATE;
82 case LDAP_SCOPE_SUBORDINATE:
83 if ( ndn->bv_len > mt->mt_nsuffix.bv_len ) {
84 return META_CANDIDATE;
86 break;
88 /* nearly useless; not allowed by config */
89 case LDAP_SCOPE_ONELEVEL:
90 if ( ndn->bv_len > mt->mt_nsuffix.bv_len ) {
91 struct berval rdn = *ndn;
93 rdn.bv_len -= mt->mt_nsuffix.bv_len
94 + STRLENOF( "," );
95 if ( dnIsOneLevelRDN( &rdn ) ) {
96 return META_CANDIDATE;
99 break;
101 /* nearly useless; not allowed by config */
102 case LDAP_SCOPE_BASE:
103 if ( ndn->bv_len == mt->mt_nsuffix.bv_len ) {
104 return META_CANDIDATE;
106 break;
109 return META_NOT_CANDIDATE;
112 if ( scope == LDAP_SCOPE_SUBTREE && dnIsSuffix( &mt->mt_nsuffix, ndn ) ) {
114 * suffix longer than dn, but common part matches
116 return META_CANDIDATE;
119 return META_NOT_CANDIDATE;
123 * meta_back_select_unique_candidate
125 * returns the index of the candidate in case it is unique, otherwise
126 * META_TARGET_NONE if none matches, or
127 * META_TARGET_MULTIPLE if more than one matches
128 * Note: ndn MUST be normalized.
131 meta_back_select_unique_candidate(
132 metainfo_t *mi,
133 struct berval *ndn )
135 int i, candidate = META_TARGET_NONE;
137 for ( i = 0; i < mi->mi_ntargets; i++ ) {
138 metatarget_t *mt = mi->mi_targets[ i ];
140 if ( meta_back_is_candidate( mt, ndn, LDAP_SCOPE_BASE ) ) {
141 if ( candidate == META_TARGET_NONE ) {
142 candidate = i;
144 } else {
145 return META_TARGET_MULTIPLE;
150 return candidate;
154 * meta_clear_unused_candidates
156 * clears all candidates except candidate
159 meta_clear_unused_candidates(
160 Operation *op,
161 int candidate )
163 metainfo_t *mi = ( metainfo_t * )op->o_bd->be_private;
164 int i;
165 SlapReply *candidates = meta_back_candidates_get( op );
167 for ( i = 0; i < mi->mi_ntargets; ++i ) {
168 if ( i == candidate ) {
169 continue;
171 META_CANDIDATE_RESET( &candidates[ i ] );
174 return 0;
178 * meta_clear_one_candidate
180 * clears the selected candidate
183 meta_clear_one_candidate(
184 Operation *op,
185 metaconn_t *mc,
186 int candidate )
188 metasingleconn_t *msc = &mc->mc_conns[ candidate ];
190 if ( msc->msc_ld != NULL ) {
192 #ifdef DEBUG_205
193 char buf[ BUFSIZ ];
195 snprintf( buf, sizeof( buf ), "meta_clear_one_candidate ldap_unbind_ext[%d] mc=%p ld=%p",
196 candidate, (void *)mc, (void *)msc->msc_ld );
197 Debug( LDAP_DEBUG_ANY, "### %s %s\n",
198 op ? op->o_log_prefix : "", buf, 0 );
199 #endif /* DEBUG_205 */
201 ldap_unbind_ext( msc->msc_ld, NULL, NULL );
202 msc->msc_ld = NULL;
205 if ( !BER_BVISNULL( &msc->msc_bound_ndn ) ) {
206 ber_memfree_x( msc->msc_bound_ndn.bv_val, NULL );
207 BER_BVZERO( &msc->msc_bound_ndn );
210 if ( !BER_BVISNULL( &msc->msc_cred ) ) {
211 memset( msc->msc_cred.bv_val, 0, msc->msc_cred.bv_len );
212 ber_memfree_x( msc->msc_cred.bv_val, NULL );
213 BER_BVZERO( &msc->msc_cred );
216 msc->msc_mscflags = 0;
218 return 0;