1*549b59edSchristos /* $NetBSD: passwd.c,v 1.3 2021/08/14 16:14:56 christos Exp $ */
24e6df137Slukem
3d11b170bStron /* $OpenLDAP$ */
42de962bdSlukem /* This work is part of OpenLDAP Software <http://www.openldap.org/>.
52de962bdSlukem *
6*549b59edSchristos * Copyright 1998-2021 The OpenLDAP Foundation.
72de962bdSlukem * All rights reserved.
82de962bdSlukem *
92de962bdSlukem * Redistribution and use in source and binary forms, with or without
102de962bdSlukem * modification, are permitted only as authorized by the OpenLDAP
112de962bdSlukem * Public License.
122de962bdSlukem *
132de962bdSlukem * A copy of this license is available in the file LICENSE in the
142de962bdSlukem * top-level directory of the distribution or, alternatively, at
152de962bdSlukem * <http://www.OpenLDAP.org/license.html>.
162de962bdSlukem */
172de962bdSlukem /* ACKNOWLEDGEMENTS:
18*549b59edSchristos * This program was originally developed by Kurt D. Zeilenga for inclusion in
192de962bdSlukem * OpenLDAP Software.
202de962bdSlukem */
212de962bdSlukem
22376af7d7Schristos #include <sys/cdefs.h>
23*549b59edSchristos __RCSID("$NetBSD: passwd.c,v 1.3 2021/08/14 16:14:56 christos Exp $");
24376af7d7Schristos
252de962bdSlukem #include "portable.h"
262de962bdSlukem
272de962bdSlukem #include <stdio.h>
282de962bdSlukem #include <ac/stdlib.h>
292de962bdSlukem #include <ac/string.h>
302de962bdSlukem #include <ac/time.h>
312de962bdSlukem
322de962bdSlukem #include "ldap-int.h"
332de962bdSlukem
342de962bdSlukem /*
352de962bdSlukem * LDAP Password Modify (Extended) Operation (RFC 3062)
362de962bdSlukem */
372de962bdSlukem
ldap_parse_passwd(LDAP * ld,LDAPMessage * res,struct berval * newpasswd)382de962bdSlukem int ldap_parse_passwd(
392de962bdSlukem LDAP *ld,
402de962bdSlukem LDAPMessage *res,
412de962bdSlukem struct berval *newpasswd )
422de962bdSlukem {
432de962bdSlukem int rc;
442de962bdSlukem struct berval *retdata = NULL;
452de962bdSlukem
462de962bdSlukem assert( ld != NULL );
472de962bdSlukem assert( LDAP_VALID( ld ) );
482de962bdSlukem assert( res != NULL );
492de962bdSlukem assert( newpasswd != NULL );
502de962bdSlukem
512de962bdSlukem newpasswd->bv_val = NULL;
522de962bdSlukem newpasswd->bv_len = 0;
532de962bdSlukem
542de962bdSlukem rc = ldap_parse_extended_result( ld, res, NULL, &retdata, 0 );
552de962bdSlukem if ( rc != LDAP_SUCCESS ) {
562de962bdSlukem return rc;
572de962bdSlukem }
582de962bdSlukem
592de962bdSlukem if ( retdata != NULL ) {
602de962bdSlukem ber_tag_t tag;
612de962bdSlukem BerElement *ber = ber_init( retdata );
622de962bdSlukem
632de962bdSlukem if ( ber == NULL ) {
642de962bdSlukem rc = ld->ld_errno = LDAP_NO_MEMORY;
652de962bdSlukem goto done;
662de962bdSlukem }
672de962bdSlukem
682de962bdSlukem /* we should check the tag */
692de962bdSlukem tag = ber_scanf( ber, "{o}", newpasswd );
702de962bdSlukem ber_free( ber, 1 );
712de962bdSlukem
722de962bdSlukem if ( tag == LBER_ERROR ) {
732de962bdSlukem rc = ld->ld_errno = LDAP_DECODING_ERROR;
742de962bdSlukem }
752de962bdSlukem }
762de962bdSlukem
772de962bdSlukem done:;
782de962bdSlukem ber_bvfree( retdata );
792de962bdSlukem
802de962bdSlukem return rc;
812de962bdSlukem }
822de962bdSlukem
832de962bdSlukem int
ldap_passwd(LDAP * ld,struct berval * user,struct berval * oldpw,struct berval * newpw,LDAPControl ** sctrls,LDAPControl ** cctrls,int * msgidp)842de962bdSlukem ldap_passwd( LDAP *ld,
852de962bdSlukem struct berval *user,
862de962bdSlukem struct berval *oldpw,
872de962bdSlukem struct berval *newpw,
882de962bdSlukem LDAPControl **sctrls,
892de962bdSlukem LDAPControl **cctrls,
902de962bdSlukem int *msgidp )
912de962bdSlukem {
922de962bdSlukem int rc;
932de962bdSlukem struct berval bv = BER_BVNULL;
942de962bdSlukem BerElement *ber = NULL;
952de962bdSlukem
962de962bdSlukem assert( ld != NULL );
972de962bdSlukem assert( LDAP_VALID( ld ) );
982de962bdSlukem assert( msgidp != NULL );
992de962bdSlukem
1002de962bdSlukem if( user != NULL || oldpw != NULL || newpw != NULL ) {
1012de962bdSlukem /* build change password control */
1022de962bdSlukem ber = ber_alloc_t( LBER_USE_DER );
1032de962bdSlukem
1042de962bdSlukem if( ber == NULL ) {
1052de962bdSlukem ld->ld_errno = LDAP_NO_MEMORY;
1062de962bdSlukem return ld->ld_errno;
1072de962bdSlukem }
1082de962bdSlukem
1092de962bdSlukem ber_printf( ber, "{" /*}*/ );
1102de962bdSlukem
1112de962bdSlukem if( user != NULL ) {
1122de962bdSlukem ber_printf( ber, "tO",
1132de962bdSlukem LDAP_TAG_EXOP_MODIFY_PASSWD_ID, user );
1142de962bdSlukem }
1152de962bdSlukem
1162de962bdSlukem if( oldpw != NULL ) {
1172de962bdSlukem ber_printf( ber, "tO",
1182de962bdSlukem LDAP_TAG_EXOP_MODIFY_PASSWD_OLD, oldpw );
1192de962bdSlukem }
1202de962bdSlukem
1212de962bdSlukem if( newpw != NULL ) {
1222de962bdSlukem ber_printf( ber, "tO",
1232de962bdSlukem LDAP_TAG_EXOP_MODIFY_PASSWD_NEW, newpw );
1242de962bdSlukem }
1252de962bdSlukem
1262de962bdSlukem ber_printf( ber, /*{*/ "N}" );
1272de962bdSlukem
1282de962bdSlukem rc = ber_flatten2( ber, &bv, 0 );
1292de962bdSlukem
1302de962bdSlukem if( rc < 0 ) {
1312de962bdSlukem ld->ld_errno = LDAP_ENCODING_ERROR;
1322de962bdSlukem return ld->ld_errno;
1332de962bdSlukem }
1342de962bdSlukem
1352de962bdSlukem }
1362de962bdSlukem
1372de962bdSlukem rc = ldap_extended_operation( ld, LDAP_EXOP_MODIFY_PASSWD,
1382de962bdSlukem bv.bv_val ? &bv : NULL, sctrls, cctrls, msgidp );
1392de962bdSlukem
1402de962bdSlukem ber_free( ber, 1 );
1412de962bdSlukem
1422de962bdSlukem return rc;
1432de962bdSlukem }
1442de962bdSlukem
1452de962bdSlukem int
ldap_passwd_s(LDAP * ld,struct berval * user,struct berval * oldpw,struct berval * newpw,struct berval * newpasswd,LDAPControl ** sctrls,LDAPControl ** cctrls)1462de962bdSlukem ldap_passwd_s(
1472de962bdSlukem LDAP *ld,
1482de962bdSlukem struct berval *user,
1492de962bdSlukem struct berval *oldpw,
1502de962bdSlukem struct berval *newpw,
1512de962bdSlukem struct berval *newpasswd,
1522de962bdSlukem LDAPControl **sctrls,
1532de962bdSlukem LDAPControl **cctrls )
1542de962bdSlukem {
1552de962bdSlukem int rc;
1562de962bdSlukem int msgid;
1572de962bdSlukem LDAPMessage *res;
1582de962bdSlukem
1592de962bdSlukem rc = ldap_passwd( ld, user, oldpw, newpw, sctrls, cctrls, &msgid );
1602de962bdSlukem if ( rc != LDAP_SUCCESS ) {
1612de962bdSlukem return rc;
1622de962bdSlukem }
1632de962bdSlukem
1642de962bdSlukem if ( ldap_result( ld, msgid, LDAP_MSG_ALL, (struct timeval *) NULL, &res ) == -1 || !res ) {
1652de962bdSlukem return ld->ld_errno;
1662de962bdSlukem }
1672de962bdSlukem
1682de962bdSlukem rc = ldap_parse_passwd( ld, res, newpasswd );
1692de962bdSlukem if( rc != LDAP_SUCCESS ) {
1702de962bdSlukem ldap_msgfree( res );
1712de962bdSlukem return rc;
1722de962bdSlukem }
1732de962bdSlukem
1742de962bdSlukem return( ldap_result2error( ld, res, 1 ) );
1752de962bdSlukem }
176