2004-04-03 16:29:21 +04:00
/*
ldb database library
Copyright ( C ) Andrew Tridgell 2004
* * NOTE ! The following LGPL license applies to the ldb
* * library . This does NOT imply that all of Samba is released
* * under the LGPL
This library is free software ; you can redistribute it and / or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation ; either
2007-07-10 06:46:15 +04:00
version 3 of the License , or ( at your option ) any later version .
2004-04-03 16:29:21 +04:00
This library is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
Lesser General Public License for more details .
You should have received a copy of the GNU Lesser General Public
2007-07-10 07:42:26 +04:00
License along with this library ; if not , see < http : //www.gnu.org/licenses/>.
2004-04-03 16:29:21 +04:00
*/
/*
* Name : ldb
*
* Component : ldbmodify
*
* Description : utility to modify records - modelled on ldapmodify
*
* Author : Andrew Tridgell
*/
2012-06-11 00:59:00 +04:00
# include "replace.h"
2009-01-30 02:39:30 +03:00
# include "ldb.h"
2007-05-05 22:50:56 +04:00
# include "tools/cmdline.h"
2009-12-22 20:44:19 +03:00
# include "ldbutil.h"
2012-04-04 09:17:32 +04:00
# include "include/ldb_private.h"
2004-04-03 16:29:21 +04:00
2009-11-20 03:34:24 +03:00
static struct ldb_cmdline * options ;
2004-04-03 16:29:21 +04:00
2010-11-01 10:45:25 +03:00
static void usage ( struct ldb_context * ldb )
2004-04-11 00:18:22 +04:00
{
printf ( " Usage: ldbmodify <options> <ldif...> \n " ) ;
printf ( " Modifies a ldb based upon ldif change records \n \n " ) ;
2010-11-01 10:45:25 +03:00
ldb_cmdline_help ( ldb , " ldbmodify " , stdout ) ;
2011-02-01 22:34:44 +03:00
exit ( LDB_ERR_OPERATIONS_ERROR ) ;
2004-04-11 00:18:22 +04:00
}
2004-04-03 16:29:21 +04:00
2004-04-11 00:18:22 +04:00
/*
process modifies for one file
*/
2012-04-04 06:51:00 +04:00
static int process_file ( struct ldb_context * ldb , FILE * f , unsigned int * count )
2004-04-11 00:18:22 +04:00
{
struct ldb_ldif * ldif ;
2011-04-04 14:59:30 +04:00
int fun_ret = LDB_SUCCESS , ret ;
2009-12-22 20:44:19 +03:00
struct ldb_control * * req_ctrls = ldb_parse_control_strings ( ldb , ldb , ( const char * * ) options - > controls ) ;
2012-04-04 09:17:32 +04:00
struct ldif_read_file_state state = {
. f = f
} ;
2011-02-01 22:34:44 +03:00
2009-12-22 20:44:19 +03:00
if ( options - > controls ! = NULL & & req_ctrls = = NULL ) {
printf ( " parsing controls failed: %s \n " , ldb_errstring ( ldb ) ) ;
2011-02-01 22:34:44 +03:00
exit ( LDB_ERR_OPERATIONS_ERROR ) ;
2009-12-22 20:44:19 +03:00
}
2012-04-04 09:17:32 +04:00
fun_ret = ldb_transaction_start ( ldb ) ;
if ( fun_ret ! = LDB_SUCCESS ) {
fprintf ( stderr , " ERR: (%s) on transaction start \n " ,
ldb_errstring ( ldb ) ) ;
return fun_ret ;
}
while ( ( ldif = ldb_ldif_read_file_state ( ldb , & state ) ) ) {
2011-04-05 16:42:06 +04:00
struct ldb_dn * olddn ;
bool deleteoldrdn = false ;
struct ldb_dn * newdn ;
const char * errstr = NULL ;
2004-04-03 16:29:21 +04:00
switch ( ldif - > changetype ) {
case LDB_CHANGETYPE_NONE :
case LDB_CHANGETYPE_ADD :
2012-04-04 06:51:00 +04:00
ret = ldb_add_ctrl ( ldb , ldif - > msg , req_ctrls ) ;
2004-04-03 16:29:21 +04:00
break ;
case LDB_CHANGETYPE_DELETE :
2012-04-04 06:51:00 +04:00
ret = ldb_delete_ctrl ( ldb , ldif - > msg - > dn , req_ctrls ) ;
2004-04-03 16:29:21 +04:00
break ;
case LDB_CHANGETYPE_MODIFY :
2012-04-04 06:51:00 +04:00
ret = ldb_modify_ctrl ( ldb , ldif - > msg , req_ctrls ) ;
2004-04-03 16:29:21 +04:00
break ;
2011-04-05 16:42:06 +04:00
case LDB_CHANGETYPE_MODRDN :
ret = ldb_ldif_parse_modrdn ( ldb , ldif , ldif , & olddn ,
NULL , & deleteoldrdn ,
NULL , & newdn ) ;
if ( ret = = LDB_SUCCESS ) {
if ( deleteoldrdn ) {
ret = ldb_rename ( ldb , olddn , newdn ) ;
} else {
errstr = " modrdn: deleteoldrdn=0 "
" not supported. " ;
ret = LDB_ERR_CONSTRAINT_VIOLATION ;
}
}
break ;
2004-04-03 16:29:21 +04:00
}
2006-10-14 12:26:10 +04:00
if ( ret ! = LDB_SUCCESS ) {
2011-04-05 16:42:06 +04:00
if ( errstr = = NULL ) {
errstr = ldb_errstring ( ldb ) ;
}
2020-06-15 13:32:36 +03:00
fprintf ( stderr ,
" ERR: (%s) \" %s \" on DN %s at block before "
" line %zu \n " ,
2009-12-29 03:39:29 +03:00
ldb_strerror ( ret ) ,
2020-06-15 13:32:36 +03:00
errstr ,
ldb_dn_get_linearized ( ldif - > msg - > dn ) ,
state . line_no ) ;
2011-04-04 14:59:30 +04:00
fun_ret = ret ;
2004-04-03 16:29:21 +04:00
} else {
2006-10-14 12:26:10 +04:00
( * count ) + + ;
2009-11-20 03:34:24 +03:00
if ( options - > verbose ) {
printf ( " Modified %s \n " , ldb_dn_get_linearized ( ldif - > msg - > dn ) ) ;
}
2004-04-03 16:29:21 +04:00
}
2004-05-20 17:25:06 +04:00
ldb_ldif_read_free ( ldb , ldif ) ;
2012-04-04 09:17:32 +04:00
if ( ret ) {
break ;
}
2004-04-03 16:29:21 +04:00
}
2012-04-04 09:17:32 +04:00
if ( fun_ret = = LDB_SUCCESS & & ! feof ( f ) ) {
2011-03-29 09:31:17 +04:00
fprintf ( stderr , " Failed to parse ldif \n " ) ;
2011-04-04 14:59:30 +04:00
fun_ret = LDB_ERR_OPERATIONS_ERROR ;
2011-03-29 09:31:17 +04:00
}
2012-04-04 09:17:32 +04:00
if ( fun_ret = = LDB_SUCCESS ) {
fun_ret = ldb_transaction_commit ( ldb ) ;
if ( fun_ret ! = LDB_SUCCESS ) {
fprintf ( stderr , " ERR: (%s) on transaction commit \n " ,
ldb_errstring ( ldb ) ) ;
}
} else {
ldb_transaction_cancel ( ldb ) ;
}
2011-04-04 14:59:30 +04:00
return fun_ret ;
2004-04-11 00:18:22 +04:00
}
2006-05-01 05:34:04 +04:00
int main ( int argc , const char * * argv )
2004-04-11 00:18:22 +04:00
{
struct ldb_context * ldb ;
2011-01-14 14:37:32 +03:00
unsigned int i , count = 0 ;
int ret = LDB_SUCCESS ;
2010-05-02 17:53:14 +04:00
TALLOC_CTX * mem_ctx = talloc_new ( NULL ) ;
2004-04-11 00:18:22 +04:00
2010-05-02 17:53:14 +04:00
ldb = ldb_init ( mem_ctx , NULL ) ;
2011-02-01 22:26:12 +03:00
if ( ldb = = NULL ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
2004-11-15 14:40:27 +03:00
2008-12-18 06:31:52 +03:00
options = ldb_cmdline_process ( ldb , argc , argv , usage ) ;
2005-06-18 11:42:21 +04:00
if ( options - > argc = = 0 ) {
2012-04-04 06:51:00 +04:00
ret = process_file ( ldb , stdin , & count ) ;
2005-06-19 08:20:54 +04:00
} else {
for ( i = 0 ; i < options - > argc ; i + + ) {
const char * fname = options - > argv [ i ] ;
FILE * f ;
2005-06-18 11:42:21 +04:00
f = fopen ( fname , " r " ) ;
2005-06-19 08:20:54 +04:00
if ( ! f ) {
perror ( fname ) ;
2011-02-01 22:34:44 +03:00
return LDB_ERR_OPERATIONS_ERROR ;
2005-06-19 08:20:54 +04:00
}
2012-04-04 06:51:00 +04:00
ret = process_file ( ldb , f , & count ) ;
2009-06-12 15:31:13 +04:00
fclose ( f ) ;
2004-04-11 00:18:22 +04:00
}
}
2010-05-02 17:53:14 +04:00
talloc_free ( mem_ctx ) ;
2004-04-03 16:29:21 +04:00
2012-04-04 09:17:32 +04:00
if ( ret ) {
printf ( " Modify failed after processing %u records \n " , count ) ;
} else {
printf ( " Modified %u records successfully \n " , count ) ;
}
2006-10-14 12:26:10 +04:00
return ret ;
2004-04-03 16:29:21 +04:00
}