2009-12-30 12:04:17 +03:00
/*
Unix SMB / CIFS implementation .
handle removal of deleted objects
Copyright ( C ) 2009 Andrew Tridgell
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program 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 General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "lib/events/events.h"
# include "dsdb/samdb/samdb.h"
# include "auth/auth.h"
# include "smbd/service.h"
# include "lib/messaging/irpc.h"
# include "dsdb/kcc/kcc_connection.h"
# include "dsdb/kcc/kcc_service.h"
# include "lib/ldb/include/ldb_errors.h"
# include "../lib/util/dlinklist.h"
# include "librpc/gen_ndr/ndr_misc.h"
# include "librpc/gen_ndr/ndr_drsuapi.h"
# include "librpc/gen_ndr/ndr_drsblobs.h"
# include "param/param.h"
2009-12-30 13:40:17 +03:00
/*
onelevel search with SHOW_DELETED control
*/
static int search_onelevel_with_deleted ( struct ldb_context * ldb ,
TALLOC_CTX * mem_ctx ,
struct ldb_result * * _res ,
struct ldb_dn * basedn ,
const char * const * attrs )
{
struct ldb_request * req ;
TALLOC_CTX * tmp_ctx ;
struct ldb_result * res ;
int ret ;
tmp_ctx = talloc_new ( mem_ctx ) ;
res = talloc_zero ( tmp_ctx , struct ldb_result ) ;
if ( ! res ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
ret = ldb_build_search_req ( & req , ldb , tmp_ctx ,
basedn ,
LDB_SCOPE_ONELEVEL ,
NULL ,
attrs ,
NULL ,
res ,
ldb_search_default_callback ,
NULL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( tmp_ctx ) ;
return ret ;
}
ret = ldb_request_add_control ( req , LDB_CONTROL_SHOW_DELETED_OID , true , NULL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( tmp_ctx ) ;
return ret ;
}
ret = ldb_request ( ldb , req ) ;
if ( ret = = LDB_SUCCESS ) {
ret = ldb_wait ( req - > handle , LDB_WAIT_ALL ) ;
}
talloc_free ( req ) ;
* _res = talloc_steal ( mem_ctx , res ) ;
return ret ;
}
2009-12-30 12:04:17 +03:00
/*
check to see if any deleted objects need scavenging
*/
NTSTATUS kccsrv_check_deleted ( struct kccsrv_service * s , TALLOC_CTX * mem_ctx )
{
struct kccsrv_partition * part ;
int ret ;
2009-12-30 13:40:17 +03:00
uint32_t tombstoneLifetime ;
2009-12-30 12:04:17 +03:00
time_t t = time ( NULL ) ;
2009-12-30 13:40:17 +03:00
if ( t - s - > last_deleted_check < lp_parm_int ( s - > task - > lp_ctx , NULL , " kccsrv " ,
2009-12-30 12:04:17 +03:00
" check_deleted_interval " , 600 ) ) {
return NT_STATUS_OK ;
}
s - > last_deleted_check = t ;
2009-12-30 13:40:17 +03:00
ret = dsdb_tombstone_lifetime ( s - > samdb , & tombstoneLifetime ) ;
if ( ret ! = LDB_SUCCESS ) {
DEBUG ( 1 , ( __location__ " : Failed to get tombstone lifetime \n " ) ) ;
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2009-12-30 12:04:17 +03:00
for ( part = s - > partitions ; part ; part = part - > next ) {
struct ldb_dn * do_dn ;
struct ldb_result * res ;
2009-12-30 13:40:17 +03:00
const char * attrs [ ] = { " whenChanged " , NULL } ;
int i ;
2009-12-30 12:04:17 +03:00
ret = dsdb_get_deleted_objects_dn ( s - > samdb , mem_ctx , part - > dn , & do_dn ) ;
2009-12-30 13:40:17 +03:00
if ( ret ! = LDB_SUCCESS ) {
/* some partitions have no Deleted Objects
container */
continue ;
}
ret = search_onelevel_with_deleted ( s - > samdb , do_dn , & res , do_dn , attrs ) ;
if ( ret ! = LDB_SUCCESS ) {
DEBUG ( 1 , ( __location__ " : Failed to search for deleted objects in %s \n " ,
ldb_dn_get_linearized ( do_dn ) ) ) ;
talloc_free ( do_dn ) ;
continue ;
}
for ( i = 0 ; i < res - > count ; i + + ) {
const char * tstring ;
time_t whenChanged = 0 ;
2009-12-30 12:04:17 +03:00
2009-12-30 13:40:17 +03:00
tstring = samdb_result_string ( res - > msgs [ i ] , " whenChanged " , NULL ) ;
if ( tstring ) {
whenChanged = ldb_string_to_time ( tstring ) ;
}
if ( t - whenChanged > tombstoneLifetime * 60 * 60 * 24 ) {
ret = ldb_delete ( s - > samdb , res - > msgs [ i ] - > dn ) ;
if ( ret ! = LDB_SUCCESS ) {
DEBUG ( 1 , ( __location__ " : Failed to remove deleted object %s \n " ,
ldb_dn_get_linearized ( res - > msgs [ i ] - > dn ) ) ) ;
} else {
DEBUG ( 4 , ( " Removed deleted object %s \n " ,
ldb_dn_get_linearized ( res - > msgs [ i ] - > dn ) ) ) ;
}
}
}
talloc_free ( do_dn ) ;
}
2009-12-30 12:04:17 +03:00
2009-12-30 13:40:17 +03:00
return NT_STATUS_OK ;
2009-12-30 12:04:17 +03:00
}