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