/* bind.c - ldap backend bind function */ /* $OpenLDAP$ */ /* * Copyright 1998-2003 The OpenLDAP Foundation, All Rights Reserved. * COPYING RESTRICTIONS APPLY, see COPYRIGHT file */ /* This is an altered version */ /* * Copyright 1999, Howard Chu, All rights reserved. * * Permission is granted to anyone to use this software for any purpose * on any computer system, and to alter it and redistribute it, subject * to the following restrictions: * * 1. The author is not responsible for the consequences of use of this * software, no matter how awful, even if they arise from flaws in it. * * 2. The origin of this software must not be misrepresented, either by * explicit claim or by omission. Since few users ever read sources, * credits should appear in the documentation. * * 3. Altered versions must be plainly marked as such, and must not be * misrepresented as being the original software. Since few users * ever read sources, credits should appear in the documentation. * * 4. This notice may not be removed or altered. * * * * Copyright 2000, Pierangelo Masarati, All rights reserved. * * This software is being modified by Pierangelo Masarati. * The previously reported conditions apply to the modified code as well. * Changes in the original code are highlighted where required. * Credits for the original code go to the author, Howard Chu. */ #include "portable.h" #include #include #include #define AVL_INTERNAL #include "slap.h" #include "back-ldap.h" #define PRINT_CONNTREE 0 static LDAP_REBIND_PROC ldap_back_rebind; int ldap_back_bind( Operation *op, SlapReply *rs ) { struct ldapinfo *li = (struct ldapinfo *) op->o_bd->be_private; struct ldapconn *lc; struct berval mdn = { 0, NULL }; int rc = 0; ber_int_t msgid; lc = ldap_back_getconn(op, rs); if ( !lc ) { return( -1 ); } /* * Rewrite the bind dn if needed */ #ifdef ENABLE_REWRITE switch ( rewrite_session( li->rwinfo, "bindDn", op->o_req_dn.bv_val, op->o_conn, &mdn.bv_val ) ) { case REWRITE_REGEXEC_OK: if ( mdn.bv_val == NULL ) { mdn = op->o_req_dn; } else { mdn.bv_len = strlen( mdn.bv_val ); } #ifdef NEW_LOGGING LDAP_LOG( BACK_LDAP, DETAIL1, "[rw] bindDn: \"%s\" -> \"%s\"\n", op->o_req_dn.bv_val, mdn.bv_val, 0 ); #else /* !NEW_LOGGING */ Debug( LDAP_DEBUG_ARGS, "rw> bindDn: \"%s\" -> \"%s\"\n", op->o_req_dn.bv_val, mdn.bv_val, 0 ); #endif /* !NEW_LOGGING */ break; case REWRITE_REGEXEC_UNWILLING: send_ldap_error( op, rs, LDAP_UNWILLING_TO_PERFORM, "Operation not allowed" ); return( -1 ); case REWRITE_REGEXEC_ERR: send_ldap_error( op, rs, LDAP_OTHER, "Rewrite error" ); return( -1 ); } #else /* !ENABLE_REWRITE */ ldap_back_dn_massage( li, &op->o_req_dn, &mdn, 0, 1 ); #endif /* !ENABLE_REWRITE */ if ( lc->bound_dn.bv_val ) { ch_free( lc->bound_dn.bv_val ); lc->bound_dn.bv_len = 0; lc->bound_dn.bv_val = NULL; } lc->bound = 0; /* method is always LDAP_AUTH_SIMPLE if we got here */ rs->sr_err = ldap_sasl_bind(lc->ld, mdn.bv_val, LDAP_SASL_SIMPLE, &op->oq_bind.rb_cred, op->o_ctrls, NULL, &msgid); rc = ldap_back_op_result( lc, op, rs, msgid, 1 ); if (rc == LDAP_SUCCESS) { lc->bound = 1; if ( mdn.bv_val != op->o_req_dn.bv_val ) { lc->bound_dn = mdn; } else { ber_dupbv( &lc->bound_dn, &op->o_req_dn ); } mdn.bv_val = NULL; if ( li->savecred ) { if ( lc->cred.bv_val ) ch_free( lc->cred.bv_val ); ber_dupbv( &lc->cred, &op->oq_bind.rb_cred ); ldap_set_rebind_proc( lc->ld, ldap_back_rebind, lc ); } } /* must re-insert if local DN changed as result of bind */ if ( lc->bound && !bvmatch(&op->o_req_ndn, &lc->local_dn ) ) { int lerr; ldap_pvt_thread_mutex_lock( &li->conn_mutex ); lc = avl_delete( &li->conntree, (caddr_t)lc, ldap_back_conn_cmp ); if ( lc->local_dn.bv_val ) ch_free( lc->local_dn.bv_val ); ber_dupbv( &lc->local_dn, &op->o_req_ndn ); lerr = avl_insert( &li->conntree, (caddr_t)lc, ldap_back_conn_cmp, ldap_back_conn_dup ); ldap_pvt_thread_mutex_unlock( &li->conn_mutex ); if ( lerr == -1 ) { ldap_back_conn_free( lc ); } } if ( mdn.bv_val && mdn.bv_val != op->o_req_dn.bv_val ) { free( mdn.bv_val ); } return( rc ); } /* * ldap_back_conn_cmp * * compares two struct ldapconn based on the value of the conn pointer; * used by avl stuff */ int ldap_back_conn_cmp( const void *c1, const void *c2 ) { const struct ldapconn *lc1 = (const struct ldapconn *)c1; const struct ldapconn *lc2 = (const struct ldapconn *)c2; int rc; /* If local DNs don't match, it is definitely not a match */ if ( ( rc = ber_bvcmp( &lc1->local_dn, &lc2->local_dn )) ) return rc; /* For shared sessions, conn is NULL. Only explicitly * bound sessions will have non-NULL conn. */ return lc1->conn - lc2->conn; } /* * ldap_back_conn_dup * * returns -1 in case a duplicate struct ldapconn has been inserted; * used by avl stuff */ int ldap_back_conn_dup( void *c1, void *c2 ) { struct ldapconn *lc1 = (struct ldapconn *)c1; struct ldapconn *lc2 = (struct ldapconn *)c2; /* Cannot have more than one shared session with same DN */ if ( dn_match( &lc1->local_dn, &lc2->local_dn ) && lc1->conn == lc2->conn ) return -1; return 0; } #if PRINT_CONNTREE > 0 static void ravl_print( Avlnode *root, int depth ) { int i; struct ldapconn *lc; if ( root == 0 ) return; ravl_print( root->avl_right, depth+1 ); for ( i = 0; i < depth; i++ ) printf( " " ); lc = root->avl_data; printf( "lc(%lx) local(%s) conn(%lx) %d\n", lc, lc->local_dn.bv_val, lc->conn, root->avl_bf ); ravl_print( root->avl_left, depth+1 ); } static void myprint( Avlnode *root ) { printf( "********\n" ); if ( root == 0 ) printf( "\tNULL\n" ); else ravl_print( root, 0 ); printf( "********\n" ); } #endif /* PRINT_CONNTREE */ struct ldapconn * ldap_back_getconn(Operation *op, SlapReply *rs) { struct ldapinfo *li = (struct ldapinfo *)op->o_bd->be_private; struct ldapconn *lc, lc_curr; LDAP *ld; int is_priv = 0; /* Searches for a ldapconn in the avl tree */ /* Explicit binds must not be shared */ if ( op->o_tag == LDAP_REQ_BIND || (op->o_conn && (op->o_bd == op->o_conn->c_authz_backend ))) { lc_curr.conn = op->o_conn; } else { lc_curr.conn = NULL; } /* Internal searches are privileged and shared. So is root. */ if ( op->o_do_not_cache || be_isroot( li->be, &op->o_ndn ) ) { lc_curr.local_dn = li->be->be_rootndn; lc_curr.conn = NULL; is_priv = 1; } else { lc_curr.local_dn = op->o_ndn; } ldap_pvt_thread_mutex_lock( &li->conn_mutex ); lc = (struct ldapconn *)avl_find( li->conntree, (caddr_t)&lc_curr, ldap_back_conn_cmp ); ldap_pvt_thread_mutex_unlock( &li->conn_mutex ); /* Looks like we didn't get a bind. Open a new session... */ if (!lc) { int vers = op->o_conn->c_protocol; rs->sr_err = ldap_initialize(&ld, li->url); if (rs->sr_err != LDAP_SUCCESS) { rs->sr_err = ldap_back_map_result(rs); if (rs->sr_text == NULL) { rs->sr_text = "ldap_initialize() failed"; } send_ldap_result( op, rs ); rs->sr_text = NULL; return( NULL ); } /* Set LDAP version. This will always succeed: If the client * bound with a particular version, then so can we. */ ldap_set_option(ld, LDAP_OPT_PROTOCOL_VERSION, (const void *)&vers); /* FIXME: configurable? */ ldap_set_option(ld, LDAP_OPT_REFERRALS, LDAP_OPT_ON); lc = (struct ldapconn *)ch_malloc(sizeof(struct ldapconn)); lc->conn = lc_curr.conn; lc->ld = ld; ber_dupbv( &lc->local_dn, &lc_curr.local_dn ); #ifdef ENABLE_REWRITE /* * Sets a cookie for the rewrite session * * FIXME: the o_conn might be no longer valid, * since we may have different entries * for the same connection */ ( void )rewrite_session_init( li->rwinfo, op->o_conn ); #endif /* ENABLE_REWRITE */ ldap_pvt_thread_mutex_init( &lc->lc_mutex ); if ( is_priv ) { ber_dupbv( &lc->cred, &li->bindpw ); ber_dupbv( &lc->bound_dn, &li->binddn ); } else { lc->cred.bv_len = 0; lc->cred.bv_val = NULL; if ( op->o_conn->c_dn.bv_len != 0 && ( op->o_bd == op->o_conn->c_authz_backend ) ) { /* * Rewrite the bind dn if needed */ #ifdef ENABLE_REWRITE lc->bound_dn.bv_val = NULL; lc->bound_dn.bv_len = 0; switch ( rewrite_session( li->rwinfo, "bindDn", op->o_conn->c_dn.bv_val, op->o_conn, &lc->bound_dn.bv_val ) ) { case REWRITE_REGEXEC_OK: if ( lc->bound_dn.bv_val == NULL ) { ber_dupbv( &lc->bound_dn, &op->o_conn->c_dn ); } else { lc->bound_dn.bv_len = strlen( lc->bound_dn.bv_val ); } #ifdef NEW_LOGGING LDAP_LOG( BACK_LDAP, DETAIL1, "[rw] bindDn: \"%s\" ->" " \"%s\"\n", op->o_conn->c_dn.bv_val, lc->bound_dn.bv_val, 0 ); #else /* !NEW_LOGGING */ Debug( LDAP_DEBUG_ARGS, "rw> bindDn: \"%s\" ->" " \"%s\"\n", op->o_conn->c_dn.bv_val, lc->bound_dn.bv_val, 0 ); #endif /* !NEW_LOGGING */ break; case REWRITE_REGEXEC_UNWILLING: send_ldap_error( op, rs, LDAP_UNWILLING_TO_PERFORM, "Operation not allowed" ); return( NULL ); case REWRITE_REGEXEC_ERR: send_ldap_error( op, rs, LDAP_OTHER, "Rewrite error" ); return( NULL ); } #else /* !ENABLE_REWRITE */ struct berval bv; ldap_back_dn_massage( li, &op->o_conn->c_dn, &bv, 0, 1 ); if ( bv.bv_val == op->o_conn->c_dn.bv_val ) { ber_dupbv( &lc->bound_dn, &bv ); } else { lc->bound_dn = bv; } #endif /* !ENABLE_REWRITE */ } else { lc->bound_dn.bv_val = NULL; lc->bound_dn.bv_len = 0; } } lc->bound = 0; /* Inserts the newly created ldapconn in the avl tree */ ldap_pvt_thread_mutex_lock( &li->conn_mutex ); rs->sr_err = avl_insert( &li->conntree, (caddr_t)lc, ldap_back_conn_cmp, ldap_back_conn_dup ); #if PRINT_CONNTREE > 0 myprint( li->conntree ); #endif /* PRINT_CONNTREE */ ldap_pvt_thread_mutex_unlock( &li->conn_mutex ); #ifdef NEW_LOGGING LDAP_LOG( BACK_LDAP, INFO, "ldap_back_getconn: conn %lx inserted\n", lc, 0, 0); #else /* !NEW_LOGGING */ Debug( LDAP_DEBUG_TRACE, "=>ldap_back_getconn: conn %lx inserted\n", lc, 0, 0 ); #endif /* !NEW_LOGGING */ /* Err could be -1 in case a duplicate ldapconn is inserted */ if ( rs->sr_err != 0 ) { ldap_back_conn_free( lc ); send_ldap_error( op, rs, LDAP_OTHER, "internal server error" ); return( NULL ); } } else { #ifdef NEW_LOGGING LDAP_LOG( BACK_LDAP, INFO, "ldap_back_getconn: conn %lx fetched\n", lc, 0, 0 ); #else /* !NEW_LOGGING */ Debug( LDAP_DEBUG_TRACE, "=>ldap_back_getconn: conn %lx fetched\n", lc, 0, 0 ); #endif /* !NEW_LOGGING */ } return( lc ); } /* * ldap_back_dobind * * Note: as the check for the value of lc->bound was already here, I removed * it from all the callers, and I made the function return the flag, so * it can be used to simplify the check. */ int ldap_back_dobind( struct ldapconn *lc, Operation *op, SlapReply *rs ) { int rc; ber_int_t msgid; ldap_pvt_thread_mutex_lock( &lc->lc_mutex ); if ( !lc->bound ) { rs->sr_err = ldap_sasl_bind(lc->ld, lc->bound_dn.bv_val, LDAP_SASL_SIMPLE, &lc->cred, NULL, NULL, &msgid); rc = ldap_back_op_result( lc, op, rs, msgid, 0 ); if (rc == LDAP_SUCCESS) { lc->bound = 1; } } rc = lc->bound; ldap_pvt_thread_mutex_unlock( &lc->lc_mutex ); return rc; } /* * ldap_back_rebind * * This is a callback used for chasing referrals using the same * credentials as the original user on this session. */ static int ldap_back_rebind( LDAP *ld, LDAP_CONST char *url, ber_tag_t request, ber_int_t msgid, void *params ) { struct ldapconn *lc = params; return ldap_bind_s( ld, lc->bound_dn.bv_val, lc->cred.bv_val, LDAP_AUTH_SIMPLE ); } /* Map API errors to protocol errors... */ int ldap_back_map_result(SlapReply *rs) { switch(rs->sr_err) { case LDAP_SERVER_DOWN: return LDAP_UNAVAILABLE; case LDAP_LOCAL_ERROR: return LDAP_OTHER; case LDAP_ENCODING_ERROR: case LDAP_DECODING_ERROR: return LDAP_PROTOCOL_ERROR; case LDAP_TIMEOUT: return LDAP_UNAVAILABLE; case LDAP_AUTH_UNKNOWN: return LDAP_AUTH_METHOD_NOT_SUPPORTED; case LDAP_FILTER_ERROR: rs->sr_text = "Filter error"; return LDAP_OTHER; case LDAP_USER_CANCELLED: rs->sr_text = "User cancelled"; return LDAP_OTHER; case LDAP_PARAM_ERROR: return LDAP_PROTOCOL_ERROR; case LDAP_NO_MEMORY: return LDAP_OTHER; case LDAP_CONNECT_ERROR: return LDAP_UNAVAILABLE; case LDAP_NOT_SUPPORTED: return LDAP_UNWILLING_TO_PERFORM; case LDAP_CONTROL_NOT_FOUND: return LDAP_PROTOCOL_ERROR; case LDAP_NO_RESULTS_RETURNED: return LDAP_NO_SUCH_OBJECT; case LDAP_MORE_RESULTS_TO_RETURN: rs->sr_text = "More results to return"; return LDAP_OTHER; case LDAP_CLIENT_LOOP: case LDAP_REFERRAL_LIMIT_EXCEEDED: return LDAP_LOOP_DETECT; default: if LDAP_API_ERROR(rs->sr_err) return LDAP_OTHER; else return rs->sr_err; } } int ldap_back_op_result(struct ldapconn *lc, Operation *op, SlapReply *rs, ber_int_t msgid, int sendok) { struct ldapinfo *li = (struct ldapinfo *)op->o_bd->be_private; char *match = NULL; LDAPMessage *res; char *text = NULL; rs->sr_text = NULL; rs->sr_matched = NULL; if (rs->sr_err == LDAP_SUCCESS) { if (ldap_result(lc->ld, msgid, 1, NULL, &res) == -1) { ldap_get_option(lc->ld, LDAP_OPT_ERROR_NUMBER, &rs->sr_err); } else { int rc = ldap_parse_result(lc->ld, res, &rs->sr_err, &match, &text, NULL, NULL, 1); rs->sr_text = text; if (rc != LDAP_SUCCESS) rs->sr_err = rc; } } if (rs->sr_err != LDAP_SUCCESS) { rs->sr_err = ldap_back_map_result(rs); /* internal ops must not reply to client */ if ( op->o_conn && !op->o_do_not_cache ) { #ifdef ENABLE_REWRITE if (match) { switch(rewrite_session(li->rwinfo, "matchedDn", match, op->o_conn, (char **)&rs->sr_matched)) { case REWRITE_REGEXEC_OK: if (!rs->sr_matched) rs->sr_matched = match; break; case REWRITE_REGEXEC_UNWILLING: case REWRITE_REGEXEC_ERR: break; } } #else struct berval dn, mdn; if (match) { ber_str2bv(match, 0, 0, &dn); ldap_back_dn_massage(li, &dn, &mdn, 0, 0); rs->sr_matched = mdn.bv_val; } #endif } } if (op->o_conn && (sendok || rs->sr_err != LDAP_SUCCESS)) { send_ldap_result( op, rs ); } if ( match ) { if ( rs->sr_matched != match ) { free( (char *)rs->sr_matched ); } rs->sr_matched = NULL; ldap_memfree( match ); } if ( text ) { ldap_memfree( text ); } rs->sr_text = NULL; return( (rs->sr_err == LDAP_SUCCESS) ? 0 : -1 ); }