mirror of
https://git.openldap.org/openldap/openldap.git
synced 2025-01-12 10:54:48 +08:00
82fad7d0c8
Includes rewriting of URLs where the DN of the referral object and the DN of the ref attribute attribute are not the same. Also, always returns explicit DN and scope. Currently, back-ldbm only. Needs to be ported to back-bdb.
124 lines
2.8 KiB
C
124 lines
2.8 KiB
C
/* referral.c - LDBM backend referral handler */
|
|
/* $OpenLDAP$ */
|
|
/*
|
|
* Copyright 2000 The OpenLDAP Foundation, All Rights Reserved.
|
|
* COPYING RESTRICTIONS APPLY, see COPYRIGHT file
|
|
*/
|
|
|
|
#include "portable.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <ac/string.h>
|
|
#include <ac/socket.h>
|
|
|
|
#include "slap.h"
|
|
#include "back-ldbm.h"
|
|
|
|
int
|
|
ldbm_back_referrals(
|
|
Backend *be,
|
|
Connection *conn,
|
|
Operation *op,
|
|
const char *dn,
|
|
const char *ndn,
|
|
const char **text )
|
|
{
|
|
struct ldbminfo *li = (struct ldbminfo *) be->be_private;
|
|
int rc = LDAP_SUCCESS;
|
|
Entry *e, *matched;
|
|
|
|
if( op->o_tag == LDAP_REQ_SEARCH ) {
|
|
/* let search take care of itself */
|
|
return rc;
|
|
}
|
|
|
|
if( get_manageDSAit( op ) ) {
|
|
/* let op take care of DSA management */
|
|
return rc;
|
|
}
|
|
|
|
/* get entry with reader lock */
|
|
e = dn2entry_r( be, ndn, &matched );
|
|
if ( e == NULL ) {
|
|
char *matched_dn = NULL;
|
|
struct berval **refs = NULL;
|
|
|
|
if ( matched != NULL ) {
|
|
matched_dn = ch_strdup( matched->e_dn );
|
|
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG(( "backend", LDAP_LEVEL_DETAIL1,
|
|
"ldbm_back_referrals: op=%ld target=\"%s\" matched=\"%s\"\n",
|
|
op->o_tag, dn, matched_dn ));
|
|
#else
|
|
Debug( LDAP_DEBUG_TRACE,
|
|
"ldbm_referrals: op=%ld target=\"%s\" matched=\"%s\"\n",
|
|
op->o_tag, dn, matched_dn );
|
|
#endif
|
|
|
|
if( is_entry_referral( matched ) ) {
|
|
rc = LDAP_OTHER;
|
|
refs = get_entry_referrals( be, conn, op, matched,
|
|
dn, LDAP_SCOPE_DEFAULT );
|
|
}
|
|
|
|
cache_return_entry_r( &li->li_cache, matched );
|
|
|
|
} else if ( default_referral != NULL ) {
|
|
rc = LDAP_OTHER;
|
|
refs = referral_rewrite( default_referral,
|
|
NULL, dn, LDAP_SCOPE_DEFAULT );
|
|
}
|
|
|
|
if( refs != NULL ) {
|
|
/* send referrals */
|
|
send_ldap_result( conn, op, rc = LDAP_REFERRAL,
|
|
matched_dn, NULL, refs, NULL );
|
|
ber_bvecfree( refs );
|
|
|
|
} else if ( rc != LDAP_SUCCESS ) {
|
|
send_ldap_result( conn, op, rc, matched_dn,
|
|
matched_dn ? "bad referral object" : "bad default referral",
|
|
NULL, NULL );
|
|
}
|
|
|
|
free( matched_dn );
|
|
return rc;
|
|
}
|
|
|
|
if ( is_entry_referral( e ) ) {
|
|
/* entry is a referral */
|
|
struct berval **refs = get_entry_referrals( be,
|
|
conn, op, e, dn, LDAP_SCOPE_DEFAULT );
|
|
struct berval **rrefs = referral_rewrite(
|
|
refs, e->e_dn, dn, LDAP_SCOPE_DEFAULT );
|
|
|
|
#ifdef NEW_LOGGING
|
|
LDAP_LOG(( "backend", LDAP_LEVEL_DETAIL1,
|
|
"ldbm_referrals: op=%ld target=\"%s\" matched=\"%s\"\n",
|
|
op->o_tag, dn, e->e_dn ));
|
|
#else
|
|
Debug( LDAP_DEBUG_TRACE,
|
|
"ldbm_referrals: op=%ld target=\"%s\" matched=\"%s\"\n",
|
|
op->o_tag, dn, e->e_dn );
|
|
#endif
|
|
|
|
if( rrefs != NULL ) {
|
|
send_ldap_result( conn, op, rc = LDAP_REFERRAL,
|
|
e->e_dn, NULL, rrefs, NULL );
|
|
|
|
ber_bvecfree( rrefs );
|
|
|
|
} else {
|
|
send_ldap_result( conn, op, rc = LDAP_OTHER, e->e_dn,
|
|
"bad referral object", NULL, NULL );
|
|
}
|
|
|
|
if( refs != NULL ) ber_bvecfree( refs );
|
|
}
|
|
|
|
cache_return_entry_r( &li->li_cache, e );
|
|
return rc;
|
|
}
|