openldap/servers/slapd/saslauthz.c

658 lines
16 KiB
C
Raw Normal View History

2000-09-22 01:32:54 +08:00
/*
* Copyright (c) 2000, Mark Adamson, Carnegie Mellon. All rights reserved.
* This software is not subject to any license of Carnegie Mellon University.
*
* Redistribution and use in source and binary forms are permitted without
* restriction or fee of any kind as long as this notice is preserved.
*
* The name "Carnegie Mellon" must not be used to endorse or promote
* products derived from this software without prior written permission.
*
*/
#include "portable.h"
#include <stdio.h>
#include <ac/stdlib.h>
#include <ac/string.h>
2000-09-22 01:32:54 +08:00
2001-09-25 08:03:24 +08:00
#include "slap.h"
2000-09-22 01:32:54 +08:00
#ifdef HAVE_CYRUS_SASL
#include <limits.h>
#include <sasl.h>
#include <ldap_pvt.h>
#endif
/* URI format: ldap://<host>/<base>[?[<attrs>][?[<scope>][?[<filter>]]]] */
2000-09-22 01:32:54 +08:00
static int slap_parseURI( char *uri,
struct berval *searchbase, int *scope, Filter **filter )
2000-09-22 01:32:54 +08:00
{
char *start, *end;
struct berval bv;
2001-12-26 23:29:17 +08:00
int rc;
2000-09-22 01:32:54 +08:00
assert( uri != NULL );
2001-12-26 23:29:17 +08:00
searchbase->bv_val = NULL;
searchbase->bv_len = 0;
2000-09-22 01:32:54 +08:00
*scope = -1;
*filter = NULL;
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_parseURI: parsing %s\n", uri ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE, "slap_parseURI: parsing %s\n", uri, 0, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
/* If it does not look like a URI, assume it is a DN */
if( !strncasecmp( uri, "dn:", sizeof("dn:")-1 ) ) {
uri += sizeof("dn:")-1;
uri += strspn( uri, " " );
2001-12-26 23:29:17 +08:00
bv.bv_val = uri;
/* FIXME: if dnNormalize actually uses input bv_len we
* will have to make this right.
*/
is_dn: bv.bv_len = 1;
rc = dnNormalize2( NULL, &bv, searchbase );
2001-12-26 23:29:17 +08:00
if (rc == LDAP_SUCCESS) {
*scope = LDAP_SCOPE_BASE;
}
return( rc );
}
/* FIXME: should use ldap_url_parse() */
if( strncasecmp( uri, "ldap://", sizeof("ldap://")-1 ) ) {
2001-12-26 23:29:17 +08:00
bv.bv_val = uri;
goto is_dn;
2000-09-22 01:32:54 +08:00
}
end = strchr( uri + (sizeof("ldap://")-1), '/' );
2000-09-22 01:32:54 +08:00
if ( end == NULL )
return( LDAP_PROTOCOL_ERROR );
/* could check the hostname here */
/* Grab the searchbase */
start = end+1;
2000-09-22 09:40:57 +08:00
end = strchr( start, '?' );
2001-12-26 23:29:17 +08:00
bv.bv_val = start;
2000-09-22 01:32:54 +08:00
if( end == NULL ) {
2001-12-26 23:29:17 +08:00
bv.bv_len = 1;
return dnNormalize2( NULL, &bv, searchbase );
2000-09-22 01:32:54 +08:00
}
*end = '\0';
2001-12-26 23:29:17 +08:00
bv.bv_len = end - start;
rc = dnNormalize2( NULL, &bv, searchbase );
2000-09-22 01:32:54 +08:00
*end = '?';
2001-12-26 23:29:17 +08:00
if (rc != LDAP_SUCCESS)
return( rc );
2000-09-22 01:32:54 +08:00
/* Skip the attrs */
start = end+1;
2000-09-22 09:40:57 +08:00
end = strchr( start, '?' );
2000-09-22 01:32:54 +08:00
if( end == NULL ) {
return( LDAP_SUCCESS );
}
/* Grab the scope */
start = end+1;
if( !strncasecmp( start, "base?", sizeof("base?")-1 )) {
2000-09-22 01:32:54 +08:00
*scope = LDAP_SCOPE_BASE;
start += sizeof("base?")-1;
2000-09-22 01:32:54 +08:00
}
else if( !strncasecmp( start, "one?", sizeof("one?")-1 )) {
2000-09-22 01:32:54 +08:00
*scope = LDAP_SCOPE_ONELEVEL;
start += sizeof("one?")-1;
2000-09-22 01:32:54 +08:00
}
else if( !strncasecmp( start, "sub?", sizeof("sub?")-1 )) {
2000-09-22 01:32:54 +08:00
*scope = LDAP_SCOPE_SUBTREE;
start += sizeof("sub?")-1;
2000-09-22 01:32:54 +08:00
}
else {
2001-12-26 23:29:17 +08:00
free( searchbase->bv_val );
searchbase->bv_val = NULL;
2000-09-22 01:32:54 +08:00
return( LDAP_PROTOCOL_ERROR );
}
/* Grab the filter */
*filter = str2filter( start );
return( LDAP_SUCCESS );
}
int slap_sasl_regexp_config( const char *match, const char *replace )
{
#ifdef HAVE_CYRUS_SASL
const char *c;
int rc, n;
SaslRegexp_t *reg;
SaslRegexp = (SaslRegexp_t *) ch_realloc( (char *) SaslRegexp,
(nSaslRegexp + 1) * sizeof(SaslRegexp_t) );
reg = &( SaslRegexp[nSaslRegexp] );
reg->match = ch_strdup( match );
reg->replace = ch_strdup( replace );
dn_normalize( reg->match );
dn_normalize( reg->replace );
/* Precompile matching pattern */
rc = regcomp( &reg->workspace, reg->match, REG_EXTENDED|REG_ICASE );
if ( rc ) {
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ERR,
2001-02-02 21:49:26 +08:00
"slap_sasl_regexp_config: \"%s\" could not be compiled.\n",
2001-01-18 00:35:53 +08:00
reg->match ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_ANY,
"SASL match pattern %s could not be compiled by regexp engine\n",
reg->match, 0, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
return( LDAP_OPERATIONS_ERROR );
}
/* Precompile replace pattern. Find the $<n> placeholders */
reg->offset[0] = -2;
n = 1;
2001-01-18 00:35:53 +08:00
for ( c = reg->replace; *c; c++ ) {
2000-09-22 01:32:54 +08:00
if ( *c == '\\' ) {
c++;
continue;
}
if ( *c == '$' ) {
if ( n == SASLREGEX_REPLACE ) {
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ERR,
"slap_sasl_regexp_config: \"%s\" has too many $n placeholders (max %d)\n",
reg->replace, SASLREGEX_REPLACE ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_ANY,
"SASL replace pattern %s has too many $n placeholders (max %d)\n",
reg->replace, SASLREGEX_REPLACE, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
return( LDAP_OPERATIONS_ERROR );
}
reg->offset[n] = c - reg->replace;
n++;
}
}
/* Final placeholder, after the last $n */
reg->offset[n] = c - reg->replace;
n++;
reg->offset[n] = -1;
nSaslRegexp++;
#endif
return( LDAP_SUCCESS );
}
#ifdef HAVE_CYRUS_SASL
/* Take the passed in SASL name and attempt to convert it into an
LDAP URI to find the matching LDAP entry, using the pattern matching
strings given in the saslregexp config file directive(s) */
static
char *slap_sasl_regexp( char *saslname )
{
char *uri=NULL;
int i, n, len, insert;
SaslRegexp_t *reg;
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_regexp: converting SASL name %s\n", saslname ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE, "slap_sasl_regexp: converting SASL name %s\n",
saslname, 0, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
if (( saslname == NULL ) || ( nSaslRegexp == 0 ))
return( NULL );
/* Match the normalized SASL name to the saslregexp patterns */
for( reg = SaslRegexp,i=0; i<nSaslRegexp; i++,reg++ ) {
if ( regexec( &reg->workspace, saslname, SASLREGEX_REPLACE,
reg->strings, 0) == 0 )
2000-09-22 01:32:54 +08:00
break;
}
if( i >= nSaslRegexp )
return( NULL );
/*
* The match pattern may have been of the form "a(b.*)c(d.*)e" and the
* replace pattern of the form "x$1y$2z". The returned string needs
* to replace the $1,$2 with the strings that matched (b.*) and (d.*)
*/
/* Get the total length of the final URI */
n=1;
len = 0;
while( reg->offset[n] >= 0 ) {
/* Len of next section from replacement string (x,y,z above) */
len += reg->offset[n] - reg->offset[n-1] - 2;
if( reg->offset[n+1] < 0)
break;
/* Len of string from saslname that matched next $i (b,d above) */
2001-01-18 00:35:53 +08:00
i = reg->replace[ reg->offset[n] + 1 ] - '0';
2000-09-22 01:32:54 +08:00
len += reg->strings[i].rm_eo - reg->strings[i].rm_so;
n++;
}
uri = ch_malloc( len + 1 );
/* Fill in URI with replace string, replacing $i as we go */
n=1;
insert = 0;
while( reg->offset[n] >= 0) {
/* Paste in next section from replacement string (x,y,z above) */
len = reg->offset[n] - reg->offset[n-1] - 2;
strncpy( uri+insert, reg->replace + reg->offset[n-1] + 2, len);
insert += len;
if( reg->offset[n+1] < 0)
break;
/* Paste in string from saslname that matched next $i (b,d above) */
2001-01-18 00:35:53 +08:00
i = reg->replace[ reg->offset[n] + 1 ] - '0';
2000-09-22 01:32:54 +08:00
len = reg->strings[i].rm_eo - reg->strings[i].rm_so;
strncpy( uri+insert, saslname + reg->strings[i].rm_so, len );
insert += len;
n++;
}
uri[insert] = '\0';
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_regexp: converted SASL name to %s\n", uri ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"slap_sasl_regexp: converted SASL name to %s\n", uri, 0, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
return( uri );
}
/*
* Given a SASL name (e.g. "UID=name,cn=REALM,cn=MECH,cn=AUTH")
2000-09-22 01:32:54 +08:00
* return the LDAP DN to which it matches. The SASL regexp rules in the config
* file turn the SASL name into an LDAP URI. If the URI is just a DN (or a
* search with scope=base), just return the URI (or its searchbase). Otherwise
* an internal search must be done, and if that search returns exactly one
* entry, return the DN of that one entry.
*/
char *slap_sasl2dn( char *saslname )
{
char *uri=NULL;
struct berval searchbase = {0, NULL};
struct berval dn = {0, NULL};
struct berval ndn;
2000-09-22 01:32:54 +08:00
int rc, scope;
Backend *be;
Filter *filter=NULL;
Connection *conn=NULL;
LDAP *client=NULL;
LDAPMessage *res=NULL, *msg;
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl2dn: converting SASL name %s to DN.\n", saslname ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"==>slap_sasl2dn: Converting SASL name %s to a DN\n", saslname, 0,0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
/* Convert the SASL name into an LDAP URI */
uri = slap_sasl_regexp( saslname );
if( uri == NULL )
goto FINISHED;
2001-12-26 23:29:17 +08:00
rc = slap_parseURI( uri, &searchbase, &scope, &filter );
if( rc ) {
2000-09-22 01:32:54 +08:00
goto FINISHED;
}
2000-09-22 01:32:54 +08:00
/* Massive shortcut: search scope == base */
if( scope == LDAP_SCOPE_BASE ) {
dn = searchbase;
2001-12-26 23:29:17 +08:00
searchbase.bv_len = 0;
searchbase.bv_val = NULL;
2000-09-22 01:32:54 +08:00
goto FINISHED;
}
/* Must do an internal search */
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_DETAIL1,
2001-05-12 04:19:16 +08:00
"slap_sasl2dn: performing internal search (base=%s, scope=%d)\n",
searchbase.bv_val, scope ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"slap_sasl2dn: performing internal search (base=%s, scope=%d)\n",
searchbase.bv_val, scope, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
be = select_backend( &searchbase, 0, 1 );
2000-09-22 01:32:54 +08:00
if(( be == NULL ) || ( be->be_search == NULL))
goto FINISHED;
suffix_alias( be, &searchbase );
2000-09-22 01:32:54 +08:00
rc = connection_internal_open( &conn, &client, saslname );
if( rc != LDAP_SUCCESS )
goto FINISHED;
(*be->be_search)( be, conn, LDAP_STAILQ_FIRST(&conn->c_ops), /*base*/NULL, &searchbase,
2000-09-22 01:32:54 +08:00
scope, /*deref=*/1, /*sizelimit=*/1, /*time=*/0, filter, /*fstr=*/NULL,
/*attrs=*/NULL, /*attrsonly=*/0 );
/* Read the client side of the internal search */
rc = ldap_result( client, LDAP_RES_ANY, LDAP_MSG_ALL, NULL, &res );
if( rc == -1 )
goto FINISHED;
/* Make sure exactly one entry was returned */
rc = ldap_count_entries( client, res );
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_DETAIL1,
"slap_sasl2dn: search DN returned %d entries\n", rc ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"slap_sasl2dn: search DN returned %d entries\n", rc,0,0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
if( rc != 1 )
goto FINISHED;
msg = ldap_first_entry( client, res );
dn.bv_val = ldap_get_dn( client, msg );
dn.bv_len = dn.bv_val ? strlen( dn.bv_val ) : 0;
if( dn.bv_val ) {
rc = dnNormalize2( NULL, &dn, &ndn );
ldap_memfree( dn.bv_val );
if( rc != LDAP_SUCCESS ) {
dn.bv_val = NULL;
dn.bv_len = 0;
goto FINISHED;
}
dn = ndn;
}
2000-09-22 01:32:54 +08:00
FINISHED:
if( searchbase.bv_len ) ch_free( searchbase.bv_val );
2000-09-22 01:32:54 +08:00
if( filter ) filter_free( filter );
if( uri ) ch_free( uri );
if( conn ) connection_internal_close( conn );
if( res ) ldap_msgfree( res );
if( client ) ldap_unbind( client );
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl2dn: Converted SASL name to %s\n",
dn.bv_len ? dn.bv_val : "<nothing>" ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE, "<==slap_sasl2dn: Converted SASL name to %s\n",
dn.bv_len ? dn.bv_val : "<nothing>", 0, 0 );
2001-01-16 03:17:29 +08:00
#endif
return( dn.bv_val );
2000-09-22 01:32:54 +08:00
}
/*
* Map a SASL regexp rule to a DN. If the rule is just a DN or a scope=base
* URI, just strcmp the rule (or its searchbase) to the *assertDN. Otherwise,
* the rule must be used as an internal search for entries. If that search
* returns the *assertDN entry, the match is successful.
*
* The assertDN should not have the dn: prefix
2000-09-22 01:32:54 +08:00
*/
static
int slap_sasl_match( char *rule, char *assertDN, char *authc )
{
struct berval searchbase = {0, NULL};
2000-09-22 01:32:54 +08:00
int rc, scope;
Backend *be;
Filter *filter=NULL;
Connection *conn=NULL;
LDAP *client=NULL;
LDAPMessage *res=NULL, *msg;
regex_t reg;
2000-09-22 01:32:54 +08:00
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_match: comparing DN %s to rule %s\n", assertDN, rule ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"===>slap_sasl_match: comparing DN %s to rule %s\n", assertDN, rule, 0 );
2001-01-16 03:17:29 +08:00
#endif
2001-12-26 23:29:17 +08:00
rc = slap_parseURI( rule, &searchbase, &scope, &filter );
2000-09-22 01:32:54 +08:00
if( rc != LDAP_SUCCESS )
goto CONCLUDED;
/* Massive shortcut: search scope == base */
if( scope == LDAP_SCOPE_BASE ) {
rc = regcomp(&reg, searchbase.bv_val,
REG_EXTENDED|REG_ICASE|REG_NOSUB);
if ( rc == 0 ) {
rc = regexec(&reg, assertDN, 0, NULL, 0);
regfree( &reg );
}
if ( rc == 0 )
2000-09-22 01:32:54 +08:00
rc = LDAP_SUCCESS;
else
rc = LDAP_INAPPROPRIATE_AUTH;
goto CONCLUDED;
}
/* Must run an internal search. */
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_DETAIL1,
"slap_sasl_match: performing internal search (base=%s, scope=%d)\n",
searchbase.bv_val, scope ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"slap_sasl_match: performing internal search (base=%s, scope=%d)\n",
searchbase.bv_val, scope, 0 );
2001-01-16 03:17:29 +08:00
#endif
be = select_backend( &searchbase, 0, 1 );
2000-09-22 01:32:54 +08:00
if(( be == NULL ) || ( be->be_search == NULL)) {
rc = LDAP_INAPPROPRIATE_AUTH;
goto CONCLUDED;
}
suffix_alias( be, &searchbase );
2000-09-22 01:32:54 +08:00
/* Make an internal connection on which to run the search */
rc = connection_internal_open( &conn, &client, authc );
if( rc != LDAP_SUCCESS )
goto CONCLUDED;
(*be->be_search)( be, conn, LDAP_STAILQ_FIRST(&conn->c_ops), /*base=*/NULL, &searchbase,
2000-09-22 01:32:54 +08:00
scope, /*deref=*/1, /*sizelimit=*/0, /*time=*/0, filter, /*fstr=*/NULL,
/*attrs=*/NULL, /*attrsonly=*/0 );
/* On the client side of the internal search, read the results. Check
if the assertDN matches any of the DN's returned by the search */
rc = ldap_result( client, LDAP_RES_ANY, LDAP_MSG_ALL, NULL, &res );
if( rc == -1 )
goto CONCLUDED;
for( msg=ldap_first_entry( client, res );
msg;
msg=ldap_next_entry( client, msg ) )
{
struct berval dn;
dn.bv_val = ldap_get_dn( client, msg );
if( dn.bv_val ) {
struct berval ndn;
dn.bv_len = strlen( dn.bv_val );
rc = dnNormalize2( NULL, &dn, &ndn );
ldap_memfree( dn.bv_val );
if( rc != LDAP_SUCCESS ) {
goto CONCLUDED;
}
rc = strcmp( ndn.bv_val, assertDN );
free( ndn.bv_val );
if( rc == 0 ) {
rc = LDAP_SUCCESS;
goto CONCLUDED;
}
2000-09-22 01:32:54 +08:00
}
}
rc = LDAP_INAPPROPRIATE_AUTH;
CONCLUDED:
if( searchbase.bv_len ) ch_free( searchbase.bv_val );
2000-09-22 01:32:54 +08:00
if( filter ) filter_free( filter );
if( conn ) connection_internal_close( conn );
if( res ) ldap_msgfree( res );
if( client ) ldap_unbind( client );
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_match: comparison returned %d\n", rc ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"<===slap_sasl_match: comparison returned %d\n", rc, 0, 0);
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
return( rc );
}
/*
* This function answers the question, "Can this ID authorize to that ID?",
* based on authorization rules. The rules are stored in the *searchDN, in the
* attribute named by *attr. If any of those rules map to the *assertDN, the
* authorization is approved.
*
* DN's passed in should have a dn: prefix
2000-09-22 01:32:54 +08:00
*/
static int
slap_sasl_check_authz(char *searchDN, char *assertDN, char *attr, char *authc)
{
const char *errmsg;
int i, rc;
2002-01-02 19:00:36 +08:00
BVarray vals=NULL;
2000-09-22 01:32:54 +08:00
AttributeDescription *ad=NULL;
struct berval bv;
2000-09-22 01:32:54 +08:00
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_check_authz: does %s match %s rule in %s?\n",
assertDN, attr, searchDN ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"==>slap_sasl_check_authz: does %s match %s rule in %s?\n",
assertDN, attr, searchDN);
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
rc = slap_str2ad( attr, &ad, &errmsg );
if( rc != LDAP_SUCCESS )
goto COMPLETE;
bv.bv_val = searchDN+3;
bv.bv_len = strlen(bv.bv_val);
rc = backend_attribute( NULL, NULL, NULL, NULL, &bv, ad, &vals );
2000-09-22 01:32:54 +08:00
if( rc != LDAP_SUCCESS )
goto COMPLETE;
/* Check if the *assertDN matches any **vals */
2002-01-02 19:00:36 +08:00
for( i=0; vals[i].bv_val != NULL; i++ ) {
rc = slap_sasl_match( vals[i].bv_val, assertDN+3, authc );
2000-09-22 01:32:54 +08:00
if ( rc == LDAP_SUCCESS )
goto COMPLETE;
}
rc = LDAP_INAPPROPRIATE_AUTH;
COMPLETE:
2002-01-02 19:00:36 +08:00
if( vals ) bvarray_free( vals );
2000-09-22 01:32:54 +08:00
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_check_authz: %s check returning %s\n", attr, rc ));
2001-01-16 03:17:29 +08:00
#else
2000-09-22 01:32:54 +08:00
Debug( LDAP_DEBUG_TRACE,
"<==slap_sasl_check_authz: %s check returning %d\n", attr, rc, 0);
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
return( rc );
}
2001-01-18 00:35:53 +08:00
#endif /* HAVE_CYRUS_SASL */
2000-09-22 01:32:54 +08:00
/* Check if a bind can SASL authorize to another identity.
Accepts authorization DN's with "dn:" prefix */
2000-09-22 01:32:54 +08:00
int slap_sasl_authorized( char *authcDN, char *authzDN )
2000-09-22 01:32:54 +08:00
{
2001-12-09 10:34:45 +08:00
int rc = LDAP_INAPPROPRIATE_AUTH;
2000-09-22 01:32:54 +08:00
#ifdef HAVE_CYRUS_SASL
/* User binding as anonymous */
if ( authzDN == NULL ) {
rc = LDAP_SUCCESS;
2000-09-22 01:32:54 +08:00
goto DONE;
}
2000-09-22 01:32:54 +08:00
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_authorized: can %s become %s?\n", authcDN, authzDN ));
2001-01-16 03:17:29 +08:00
#else
Debug( LDAP_DEBUG_TRACE,
"==>slap_sasl_authorized: can %s become %s?\n", authcDN, authzDN, 0 );
2001-01-16 03:17:29 +08:00
#endif
/* If person is authorizing to self, succeed */
if ( !strcmp( authcDN, authzDN ) ) {
rc = LDAP_SUCCESS;
goto DONE;
}
2000-09-22 01:32:54 +08:00
/* Check source rules */
rc = slap_sasl_check_authz( authcDN, authzDN, SASL_AUTHZ_SOURCE_ATTR,
authcDN );
2001-12-09 10:34:45 +08:00
if( rc == LDAP_SUCCESS ) {
2000-09-22 01:32:54 +08:00
goto DONE;
2001-12-09 10:34:45 +08:00
}
2000-09-22 01:32:54 +08:00
/* Check destination rules */
rc = slap_sasl_check_authz( authzDN, authcDN, SASL_AUTHZ_DEST_ATTR,
authcDN );
2001-12-09 10:34:45 +08:00
if( rc == LDAP_SUCCESS ) {
2000-09-22 01:32:54 +08:00
goto DONE;
2001-12-09 10:34:45 +08:00
}
2000-09-22 01:32:54 +08:00
rc = LDAP_INAPPROPRIATE_AUTH;
DONE:
2001-12-09 10:34:45 +08:00
#endif
2001-01-16 03:17:29 +08:00
#ifdef NEW_LOGGING
2001-01-18 00:35:53 +08:00
LDAP_LOG(( "sasl", LDAP_LEVEL_ENTRY,
"slap_sasl_authorized: return %d\n", rc ));
2001-01-16 03:17:29 +08:00
#else
Debug( LDAP_DEBUG_TRACE,
"<== slap_sasl_authorized: return %d\n", rc, 0, 0 );
2001-01-16 03:17:29 +08:00
#endif
2000-09-22 01:32:54 +08:00
return( rc );
}