#include <ac/string.h>
#include <ac/time.h>
+#include "ldap_pvt.h"
#include "slap.h"
static void modlist_free(LDAPModList *ml);
Operation *op
)
{
- char *ndn;
+ char *dn, *ndn;
char *last;
ber_tag_t tag;
ber_len_t len;
- LDAPModList *modlist;
- LDAPModList **modtail;
+ LDAPModList *modlist = NULL;
+ LDAPModList **modtail = &modlist;
#ifdef LDAP_DEBUG
LDAPModList *tmp;
#endif
* }
*/
- if ( ber_scanf( op->o_ber, "{a" /*}*/, &ndn ) == LBER_ERROR ) {
+ if ( ber_scanf( op->o_ber, "{a" /*}*/, &dn ) == LBER_ERROR ) {
Debug( LDAP_DEBUG_ANY, "do_modify: ber_scanf failed\n", 0, 0, 0 );
send_ldap_disconnect( conn, op,
LDAP_PROTOCOL_ERROR, "decoding error" );
return -1;
}
- Debug( LDAP_DEBUG_ARGS, "do_modify: dn (%s)\n", ndn, 0, 0 );
+ Debug( LDAP_DEBUG_ARGS, "do_modify: dn (%s)\n", dn, 0, 0 );
- if( dn_normalize_case( ndn ) == NULL ) {
- Debug( LDAP_DEBUG_ANY, "do_modify: invalid dn (%s)\n", ndn, 0, 0 );
+ ndn = ch_strdup( dn );
+
+ if( dn_normalize( ndn ) == NULL ) {
+ Debug( LDAP_DEBUG_ANY, "do_modify: invalid dn (%s)\n", dn, 0, 0 );
send_ldap_result( conn, op, rc = LDAP_INVALID_DN_SYNTAX, NULL,
"invalid DN", NULL, NULL );
- free( ndn );
- return rc;
+ goto cleanup;
}
/* collect modifications & save for later */
- modlist = NULL;
- modtail = &modlist;
for ( tag = ber_first_element( op->o_ber, &len, &last );
tag != LBER_DEFAULT;
{
send_ldap_disconnect( conn, op,
LDAP_PROTOCOL_ERROR, "decoding modlist error" );
- free( ndn );
- free( *modtail );
- *modtail = NULL;
- modlist_free( modlist );
- return -1;
+ rc = -1;
+ goto cleanup;
}
(*modtail)->ml_op = mop;
(*modtail)->ml_op != LDAP_MOD_DELETE &&
(*modtail)->ml_op != LDAP_MOD_REPLACE )
{
+ Debug( LDAP_DEBUG_ANY,
+ "do_modify: invalid modify operation (%ld)\n",
+ (long) (*modtail)->ml_op, 0, 0 );
send_ldap_result( conn, op, LDAP_PROTOCOL_ERROR,
NULL, "unrecognized modify operation", NULL, NULL );
- free( ndn );
- modlist_free( modlist );
- return LDAP_PROTOCOL_ERROR;
+ rc = LDAP_PROTOCOL_ERROR;
+ goto cleanup;
}
- if ( (*modtail)->ml_bvalues == NULL
- && (*modtail)->ml_op != LDAP_MOD_DELETE )
+ if ( (*modtail)->ml_bvalues == NULL && (
+ (*modtail)->ml_op != LDAP_MOD_REPLACE &&
+ (*modtail)->ml_op != LDAP_MOD_DELETE ) )
{
+ Debug( LDAP_DEBUG_ANY,
+ "do_modify: invalid modify operation (%ld) without values\n",
+ (long) (*modtail)->ml_op, 0, 0 );
send_ldap_result( conn, op, LDAP_PROTOCOL_ERROR,
- NULL, "unrecognized modify operation", NULL, NULL );
- free( ndn );
- modlist_free( modlist );
- return LDAP_PROTOCOL_ERROR;
+ NULL, "unrecognized modify operation without values",
+ NULL, NULL );
+ rc = LDAP_PROTOCOL_ERROR;
+ goto cleanup;
}
attr_normalize( (*modtail)->ml_type );
#endif
if( (rc = get_ctrls( conn, op, 1 )) != LDAP_SUCCESS ) {
- free( ndn );
- modlist_free( modlist );
Debug( LDAP_DEBUG_ANY, "do_modify: get_ctrls failed\n", 0, 0, 0 );
- return rc;
+ goto cleanup;
}
Statslog( LDAP_DEBUG_STATS, "conn=%ld op=%d MOD dn=\"%s\"\n",
- op->o_connid, op->o_opid, ndn, 0, 0 );
+ op->o_connid, op->o_opid, dn, 0, 0 );
/*
* We could be serving multiple database backends. Select the
* if we don't hold it.
*/
if ( (be = select_backend( ndn )) == NULL ) {
- free( ndn );
- modlist_free( modlist );
send_ldap_result( conn, op, rc = LDAP_REFERRAL,
NULL, NULL, default_referral, NULL );
- return rc;
+ goto cleanup;
+ }
+
+ /* make sure this backend recongizes critical controls */
+ rc = backend_check_controls( be, conn, op ) ;
+
+ if( rc != LDAP_SUCCESS ) {
+ send_ldap_result( conn, op, rc,
+ NULL, NULL, NULL, NULL );
}
if ( global_readonly || be->be_readonly ) {
0, 0, 0 );
send_ldap_result( conn, op, rc = LDAP_UNWILLING_TO_PERFORM,
NULL, "database is read-only", NULL, NULL );
- goto done;
+ goto cleanup;
}
/* deref suffix alias if appropriate */
rc = add_modified_attrs( op, &modlist );
if( rc != LDAP_SUCCESS ) {
- free( ndn );
- modlist_free( modlist );
send_ldap_result( conn, op, rc,
NULL, "no-user-modification attribute type",
NULL, NULL );
- return rc;
+ goto cleanup;
}
}
- if ( (*be->be_modify)( be, conn, op, ndn, modlist ) == 0
+ if ( (*be->be_modify)( be, conn, op, dn, ndn, modlist ) == 0
#ifdef SLAPD_MULTIMASTER
&& ( be->be_update_ndn == NULL ||
strcmp( be->be_update_ndn, op->o_ndn ) != 0 )
#endif
) {
/* but we log only the ones not from a replicator user */
- replog( be, op, ndn, modlist );
+ replog( be, op, dn, modlist );
}
#ifndef SLAPD_MULTIMASTER
NULL, "Function not implemented", NULL, NULL );
}
-done:
+cleanup:
+ free( dn );
free( ndn );
- modlist_free( modlist );
+ if ( modlist != NULL )
+ modlist_free( modlist );
return rc;
}
currenttime = slap_get_time();
ldap_pvt_thread_mutex_lock( &gmtime_mutex );
-#ifndef LDAP_LOCALTIME
ltm = gmtime( ¤ttime );
strftime( buf, sizeof(buf), "%Y%m%d%H%M%SZ", ltm );
-#else
- ltm = localtime( ¤ttime );
- strftime( buf, sizeof(buf), "%y%m%d%H%M%SZ", ltm );
-#endif
ldap_pvt_thread_mutex_unlock( &gmtime_mutex );
bv.bv_val = buf;
for ( ; ml != NULL; ml = next ) {
next = ml->ml_next;
- free( ml->ml_type );
+ if (ml->ml_type)
+ free( ml->ml_type );
+
if ( ml->ml_bvalues != NULL )
ber_bvecfree( ml->ml_bvalues );