xref: /netbsd-src/external/bsd/openldap/dist/libraries/libldap/references.c (revision 549b59ed3ccf0d36d3097190a0db27b770f3a839)
1*549b59edSchristos /*	$NetBSD: references.c,v 1.3 2021/08/14 16:14:56 christos Exp $	*/
24e6df137Slukem 
32de962bdSlukem /* references.c */
4d11b170bStron /* $OpenLDAP$ */
52de962bdSlukem /* This work is part of OpenLDAP Software <http://www.openldap.org/>.
62de962bdSlukem  *
7*549b59edSchristos  * Copyright 1998-2021 The OpenLDAP Foundation.
82de962bdSlukem  * All rights reserved.
92de962bdSlukem  *
102de962bdSlukem  * Redistribution and use in source and binary forms, with or without
112de962bdSlukem  * modification, are permitted only as authorized by the OpenLDAP
122de962bdSlukem  * Public License.
132de962bdSlukem  *
142de962bdSlukem  * A copy of this license is available in the file LICENSE in the
152de962bdSlukem  * top-level directory of the distribution or, alternatively, at
162de962bdSlukem  * <http://www.OpenLDAP.org/license.html>.
172de962bdSlukem  */
182de962bdSlukem 
19376af7d7Schristos #include <sys/cdefs.h>
20*549b59edSchristos __RCSID("$NetBSD: references.c,v 1.3 2021/08/14 16:14:56 christos Exp $");
21376af7d7Schristos 
222de962bdSlukem #include "portable.h"
232de962bdSlukem 
242de962bdSlukem #include <stdio.h>
252de962bdSlukem 
262de962bdSlukem #include <ac/stdlib.h>
272de962bdSlukem 
282de962bdSlukem #include <ac/socket.h>
292de962bdSlukem #include <ac/string.h>
302de962bdSlukem #include <ac/time.h>
312de962bdSlukem 
322de962bdSlukem #include "ldap-int.h"
332de962bdSlukem 
342de962bdSlukem LDAPMessage *
ldap_first_reference(LDAP * ld,LDAPMessage * chain)352de962bdSlukem ldap_first_reference( LDAP *ld, LDAPMessage *chain )
362de962bdSlukem {
372de962bdSlukem 	assert( ld != NULL );
382de962bdSlukem 	assert( LDAP_VALID( ld ) );
392de962bdSlukem 	assert( chain != NULL );
402de962bdSlukem 
412de962bdSlukem 	return chain->lm_msgtype == LDAP_RES_SEARCH_REFERENCE
422de962bdSlukem 		? chain
432de962bdSlukem 		: ldap_next_reference( ld, chain );
442de962bdSlukem }
452de962bdSlukem 
462de962bdSlukem LDAPMessage *
ldap_next_reference(LDAP * ld,LDAPMessage * ref)472de962bdSlukem ldap_next_reference( LDAP *ld, LDAPMessage *ref )
482de962bdSlukem {
492de962bdSlukem 	assert( ld != NULL );
502de962bdSlukem 	assert( LDAP_VALID( ld ) );
512de962bdSlukem 	assert( ref != NULL );
522de962bdSlukem 
532de962bdSlukem 	for (
542de962bdSlukem 		ref = ref->lm_chain;
552de962bdSlukem 		ref != NULL;
562de962bdSlukem 		ref = ref->lm_chain )
572de962bdSlukem 	{
582de962bdSlukem 		if( ref->lm_msgtype == LDAP_RES_SEARCH_REFERENCE ) {
592de962bdSlukem 			return( ref );
602de962bdSlukem 		}
612de962bdSlukem 	}
622de962bdSlukem 
632de962bdSlukem 	return( NULL );
642de962bdSlukem }
652de962bdSlukem 
662de962bdSlukem int
ldap_count_references(LDAP * ld,LDAPMessage * chain)672de962bdSlukem ldap_count_references( LDAP *ld, LDAPMessage *chain )
682de962bdSlukem {
692de962bdSlukem 	int	i;
702de962bdSlukem 
712de962bdSlukem 	assert( ld != NULL );
722de962bdSlukem 	assert( LDAP_VALID( ld ) );
732de962bdSlukem 
742de962bdSlukem 	for ( i = 0; chain != NULL; chain = chain->lm_chain ) {
752de962bdSlukem 		if( chain->lm_msgtype == LDAP_RES_SEARCH_REFERENCE ) {
762de962bdSlukem 			i++;
772de962bdSlukem 		}
782de962bdSlukem 	}
792de962bdSlukem 
802de962bdSlukem 	return( i );
812de962bdSlukem }
822de962bdSlukem 
832de962bdSlukem int
ldap_parse_reference(LDAP * ld,LDAPMessage * ref,char *** referralsp,LDAPControl *** serverctrls,int freeit)842de962bdSlukem ldap_parse_reference(
852de962bdSlukem 	LDAP            *ld,
862de962bdSlukem 	LDAPMessage     *ref,
872de962bdSlukem 	char            ***referralsp,
882de962bdSlukem 	LDAPControl     ***serverctrls,
892de962bdSlukem 	int             freeit)
902de962bdSlukem {
912de962bdSlukem 	BerElement be;
922de962bdSlukem 	char **refs = NULL;
932de962bdSlukem 	int rc;
942de962bdSlukem 
952de962bdSlukem 	assert( ld != NULL );
962de962bdSlukem 	assert( LDAP_VALID( ld ) );
972de962bdSlukem 	assert( ref !=  NULL );
982de962bdSlukem 
992de962bdSlukem 	if( ref->lm_msgtype != LDAP_RES_SEARCH_REFERENCE ) {
1002de962bdSlukem 		return LDAP_PARAM_ERROR;
1012de962bdSlukem 	}
1022de962bdSlukem 
1032de962bdSlukem 	/* make a private copy of BerElement */
1042de962bdSlukem 	AC_MEMCPY(&be, ref->lm_ber, sizeof(be));
1052de962bdSlukem 
1062de962bdSlukem 	if ( ber_scanf( &be, "{v" /*}*/, &refs ) == LBER_ERROR ) {
1072de962bdSlukem 		rc = LDAP_DECODING_ERROR;
1082de962bdSlukem 		goto free_and_return;
1092de962bdSlukem 	}
1102de962bdSlukem 
1112de962bdSlukem 	if ( serverctrls == NULL ) {
1122de962bdSlukem 		rc = LDAP_SUCCESS;
1132de962bdSlukem 		goto free_and_return;
1142de962bdSlukem 	}
1152de962bdSlukem 
1162de962bdSlukem 	if ( ber_scanf( &be, /*{*/ "}" ) == LBER_ERROR ) {
1172de962bdSlukem 		rc = LDAP_DECODING_ERROR;
1182de962bdSlukem 		goto free_and_return;
1192de962bdSlukem 	}
1202de962bdSlukem 
1212de962bdSlukem 	rc = ldap_pvt_get_controls( &be, serverctrls );
1222de962bdSlukem 
1232de962bdSlukem free_and_return:
1242de962bdSlukem 
1252de962bdSlukem 	if( referralsp != NULL ) {
126*549b59edSchristos 		/* provide references regardless of return code */
1272de962bdSlukem 		*referralsp = refs;
1282de962bdSlukem 
1292de962bdSlukem 	} else {
1302de962bdSlukem 		LDAP_VFREE( refs );
1312de962bdSlukem 	}
1322de962bdSlukem 
1332de962bdSlukem 	if( freeit ) {
1342de962bdSlukem 		ldap_msgfree( ref );
1352de962bdSlukem 	}
1362de962bdSlukem 
1372de962bdSlukem 	if( rc != LDAP_SUCCESS ) {
1382de962bdSlukem 		ld->ld_errno = rc;
1392de962bdSlukem 
1402de962bdSlukem 		if( ld->ld_matched != NULL ) {
1412de962bdSlukem 			LDAP_FREE( ld->ld_matched );
1422de962bdSlukem 			ld->ld_matched = NULL;
1432de962bdSlukem 		}
1442de962bdSlukem 
1452de962bdSlukem 		if( ld->ld_error != NULL ) {
1462de962bdSlukem 			LDAP_FREE( ld->ld_error );
1472de962bdSlukem 			ld->ld_error = NULL;
1482de962bdSlukem 		}
1492de962bdSlukem 	}
1502de962bdSlukem 
1512de962bdSlukem 	return rc;
1522de962bdSlukem }
153