r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
/*
Partitions ldb module
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2006
2007-01-07 19:11:27 +00:00
Copyright ( C ) Stefan Metzmacher < metze @ samba . org > 2007
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
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
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
( 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
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
*/
/*
* Name : ldb
*
* Component : ldb partitions module
*
* Description : Implement LDAP partitions
*
* Author : Andrew Bartlett
2007-01-07 19:11:27 +00:00
* Author : Stefan Metzmacher
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
*/
2009-10-02 10:28:29 +10:00
# include "dsdb/samdb/ldb_modules/partition.h"
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
2008-09-11 18:36:28 -04:00
struct part_request {
struct ldb_module * module ;
struct ldb_request * req ;
} ;
2006-07-22 16:56:33 +00:00
struct partition_context {
2006-07-17 22:36:04 +00:00
struct ldb_module * module ;
2008-09-11 18:36:28 -04:00
struct ldb_request * req ;
2006-07-17 22:36:04 +00:00
2008-09-11 18:36:28 -04:00
struct part_request * part_req ;
2006-07-29 01:13:53 +00:00
int num_requests ;
int finished_requests ;
2006-07-17 22:36:04 +00:00
} ;
2008-09-11 18:36:28 -04:00
static struct partition_context * partition_init_ctx ( struct ldb_module * module , struct ldb_request * req )
2006-07-17 22:36:04 +00:00
{
2006-07-22 16:56:33 +00:00
struct partition_context * ac ;
2006-07-17 22:36:04 +00:00
2008-09-11 18:36:28 -04:00
ac = talloc_zero ( req , struct partition_context ) ;
2006-07-17 22:36:04 +00:00
if ( ac = = NULL ) {
2009-06-30 15:12:29 +10:00
ldb_set_errstring ( ldb_module_get_ctx ( module ) , " Out of Memory " ) ;
2006-07-17 22:36:04 +00:00
return NULL ;
}
ac - > module = module ;
2008-09-11 18:36:28 -04:00
ac - > req = req ;
2007-01-11 09:38:21 +00:00
return ac ;
2006-07-17 22:36:04 +00:00
}
2008-09-11 18:36:28 -04:00
/*
2010-02-20 20:31:41 +01:00
* helper functions to call the next module in chain
*/
2009-10-15 10:49:25 +11:00
int partition_request ( struct ldb_module * module , struct ldb_request * request )
2006-07-10 11:24:46 +00:00
{
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) { \
const struct dsdb_control_current_partition * partition = NULL ;
struct ldb_control * partition_ctrl = ldb_request_get_control ( request , DSDB_CONTROL_CURRENT_PARTITION_OID ) ;
if ( partition_ctrl ) {
partition = talloc_get_type ( partition_ctrl - > data ,
struct dsdb_control_current_partition ) ;
}
if ( partition ! = NULL ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_request() -> %s " ,
ldb_dn_get_linearized ( partition - > dn ) ) ;
} else {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_request() -> (metadata partition) " ) ;
}
2008-09-11 18:36:28 -04:00
}
2009-10-27 10:54:16 +11:00
return ldb_next_request ( module , request ) ;
2006-07-10 11:24:46 +00:00
}
2009-09-02 13:36:54 +10:00
static struct dsdb_partition * find_partition ( struct partition_private_data * data ,
struct ldb_dn * dn ,
struct ldb_request * req )
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
{
int i ;
2009-09-02 13:36:54 +10:00
struct ldb_control * partition_ctrl ;
/* see if the request has the partition DN specified in a
* control . The repl_meta_data module can specify this to
* ensure that replication happens to the right partition
*/
partition_ctrl = ldb_request_get_control ( req , DSDB_CONTROL_CURRENT_PARTITION_OID ) ;
if ( partition_ctrl ) {
const struct dsdb_control_current_partition * partition ;
partition = talloc_get_type ( partition_ctrl - > data ,
struct dsdb_control_current_partition ) ;
if ( partition ! = NULL ) {
dn = partition - > dn ;
}
}
2007-01-17 13:46:16 +00:00
2009-09-13 18:12:05 +10:00
if ( dn = = NULL ) {
return NULL ;
}
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
/* Look at base DN */
/* Figure out which partition it is under */
2009-06-27 14:16:19 +02:00
/* Skip the lot if 'data' isn't here yet (initialisation) */
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2009-09-02 13:36:54 +10:00
if ( ldb_dn_compare_base ( data - > partitions [ i ] - > ctrl - > dn , dn ) = = 0 ) {
2007-01-17 13:46:16 +00:00
return data - > partitions [ i ] ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
}
2007-01-17 13:46:16 +00:00
return NULL ;
2009-04-03 19:27:16 +02:00
}
2007-01-17 13:46:16 +00:00
2007-12-17 04:22:44 +01:00
/**
* fire the caller ' s callback for every entry , but only send ' done ' once .
*/
2008-09-11 18:36:28 -04:00
static int partition_req_callback ( struct ldb_request * req ,
struct ldb_reply * ares )
2006-07-29 01:13:53 +00:00
{
struct partition_context * ac ;
2008-09-11 18:36:28 -04:00
struct ldb_module * module ;
struct ldb_request * nreq ;
2009-10-16 16:25:49 +11:00
int ret ;
2009-09-28 12:28:59 +10:00
struct partition_private_data * data ;
2009-10-15 10:49:25 +11:00
struct ldb_control * partition_ctrl ;
2006-07-29 01:13:53 +00:00
2008-09-11 18:36:28 -04:00
ac = talloc_get_type ( req - > context , struct partition_context ) ;
2009-09-28 12:28:59 +10:00
data = talloc_get_type ( ac - > module - > private_data , struct partition_private_data ) ;
2006-07-29 01:13:53 +00:00
2008-09-11 18:36:28 -04:00
if ( ! ares ) {
return ldb_module_done ( ac - > req , NULL , NULL ,
LDB_ERR_OPERATIONS_ERROR ) ;
}
2008-10-03 12:21:53 -07:00
2009-10-15 10:49:25 +11:00
partition_ctrl = ldb_request_get_control ( req , DSDB_CONTROL_CURRENT_PARTITION_OID ) ;
if ( partition_ctrl & & ( ac - > num_requests = = 1 | | ares - > type = = LDB_REPLY_ENTRY ) ) {
/* If we didn't fan this request out to mulitple partitions,
* or this is an individual search result , we can
* deterministily tell the caller what partition this was
* written to ( repl_meta_data likes to know ) */
2010-02-20 20:31:41 +01:00
ret = ldb_reply_add_control ( ares ,
DSDB_CONTROL_CURRENT_PARTITION_OID ,
false , partition_ctrl - > data ) ;
if ( ret ! = LDB_SUCCESS ) {
return ldb_module_done ( ac - > req , NULL , NULL ,
ret ) ;
}
2009-10-15 10:49:25 +11:00
}
2010-01-07 10:22:26 +11:00
if ( ares - > error ! = LDB_SUCCESS ) {
2008-09-11 18:36:28 -04:00
return ldb_module_done ( ac - > req , ares - > controls ,
ares - > response , ares - > error ) ;
2006-07-29 01:13:53 +00:00
}
2008-09-11 18:36:28 -04:00
switch ( ares - > type ) {
case LDB_REPLY_REFERRAL :
/* ignore referrals for now */
break ;
2006-07-29 01:13:53 +00:00
2008-09-11 18:36:28 -04:00
case LDB_REPLY_ENTRY :
if ( ac - > req - > operation ! = LDB_SEARCH ) {
2009-06-30 15:12:29 +10:00
ldb_set_errstring ( ldb_module_get_ctx ( ac - > module ) ,
2008-09-11 18:36:28 -04:00
" partition_req_callback: "
" Unsupported reply type for this request " ) ;
return ldb_module_done ( ac - > req , NULL , NULL ,
LDB_ERR_OPERATIONS_ERROR ) ;
}
2009-09-28 12:28:59 +10:00
2008-12-16 08:59:05 +01:00
return ldb_module_send_entry ( ac - > req , ares - > message , ares - > controls ) ;
2008-09-11 18:36:28 -04:00
case LDB_REPLY_DONE :
if ( ac - > req - > operation = = LDB_EXTENDED ) {
/* FIXME: check for ares->response, replmd does not fill it ! */
if ( ares - > response ) {
if ( strcmp ( ares - > response - > oid , LDB_EXTENDED_START_TLS_OID ) ! = 0 ) {
2009-06-30 15:12:29 +10:00
ldb_set_errstring ( ldb_module_get_ctx ( ac - > module ) ,
2008-09-11 18:36:28 -04:00
" partition_req_callback: "
" Unknown extended reply, "
" only supports START_TLS " ) ;
talloc_free ( ares ) ;
return ldb_module_done ( ac - > req , NULL , NULL ,
LDB_ERR_OPERATIONS_ERROR ) ;
}
}
}
2006-07-29 02:00:33 +00:00
2006-07-29 01:13:53 +00:00
ac - > finished_requests + + ;
if ( ac - > finished_requests = = ac - > num_requests ) {
2008-09-11 18:36:28 -04:00
/* this was the last one, call callback */
return ldb_module_done ( ac - > req , ares - > controls ,
2008-10-03 12:21:53 -07:00
ares - > response ,
2010-01-07 10:22:26 +11:00
ares - > error ) ;
2006-07-29 01:13:53 +00:00
}
2008-09-11 18:36:28 -04:00
/* not the last, now call the next one */
module = ac - > part_req [ ac - > finished_requests ] . module ;
nreq = ac - > part_req [ ac - > finished_requests ] . req ;
ret = partition_request ( module , nreq ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( ares ) ;
return ldb_module_done ( ac - > req , NULL , NULL , ret ) ;
}
break ;
2006-07-29 01:13:53 +00:00
}
2008-09-11 18:36:28 -04:00
2006-07-29 01:13:53 +00:00
talloc_free ( ares ) ;
2008-09-11 18:36:28 -04:00
return LDB_SUCCESS ;
2006-07-29 01:13:53 +00:00
}
2008-09-11 18:36:28 -04:00
static int partition_prep_request ( struct partition_context * ac ,
2009-09-02 13:36:54 +10:00
struct dsdb_partition * partition )
2006-07-17 22:36:04 +00:00
{
int ret ;
2006-09-14 03:15:30 +00:00
struct ldb_request * req ;
2007-01-17 16:55:37 +00:00
2008-09-11 18:36:28 -04:00
ac - > part_req = talloc_realloc ( ac , ac - > part_req ,
struct part_request ,
ac - > num_requests + 1 ) ;
if ( ac - > part_req = = NULL ) {
2009-06-30 15:12:29 +10:00
ldb_oom ( ldb_module_get_ctx ( ac - > module ) ) ;
2006-07-17 22:36:04 +00:00
return LDB_ERR_OPERATIONS_ERROR ;
}
2008-09-11 18:36:28 -04:00
switch ( ac - > req - > operation ) {
case LDB_SEARCH :
2009-06-30 15:12:29 +10:00
ret = ldb_build_search_req_ex ( & req , ldb_module_get_ctx ( ac - > module ) ,
2008-09-11 18:36:28 -04:00
ac - > part_req ,
ac - > req - > op . search . base ,
ac - > req - > op . search . scope ,
ac - > req - > op . search . tree ,
ac - > req - > op . search . attrs ,
ac - > req - > controls ,
ac , partition_req_callback ,
ac - > req ) ;
break ;
case LDB_ADD :
2009-06-30 15:12:29 +10:00
ret = ldb_build_add_req ( & req , ldb_module_get_ctx ( ac - > module ) , ac - > part_req ,
2008-09-11 18:36:28 -04:00
ac - > req - > op . add . message ,
ac - > req - > controls ,
ac , partition_req_callback ,
ac - > req ) ;
break ;
case LDB_MODIFY :
2009-06-30 15:12:29 +10:00
ret = ldb_build_mod_req ( & req , ldb_module_get_ctx ( ac - > module ) , ac - > part_req ,
2008-09-11 18:36:28 -04:00
ac - > req - > op . mod . message ,
ac - > req - > controls ,
ac , partition_req_callback ,
ac - > req ) ;
break ;
case LDB_DELETE :
2009-06-30 15:12:29 +10:00
ret = ldb_build_del_req ( & req , ldb_module_get_ctx ( ac - > module ) , ac - > part_req ,
2008-09-11 18:36:28 -04:00
ac - > req - > op . del . dn ,
ac - > req - > controls ,
ac , partition_req_callback ,
ac - > req ) ;
break ;
case LDB_RENAME :
2009-06-30 15:12:29 +10:00
ret = ldb_build_rename_req ( & req , ldb_module_get_ctx ( ac - > module ) , ac - > part_req ,
2008-09-11 18:36:28 -04:00
ac - > req - > op . rename . olddn ,
ac - > req - > op . rename . newdn ,
ac - > req - > controls ,
ac , partition_req_callback ,
ac - > req ) ;
break ;
case LDB_EXTENDED :
2009-06-30 15:12:29 +10:00
ret = ldb_build_extended_req ( & req , ldb_module_get_ctx ( ac - > module ) ,
2008-09-11 18:36:28 -04:00
ac - > part_req ,
ac - > req - > op . extended . oid ,
ac - > req - > op . extended . data ,
ac - > req - > controls ,
ac , partition_req_callback ,
ac - > req ) ;
break ;
default :
2009-06-30 15:12:29 +10:00
ldb_set_errstring ( ldb_module_get_ctx ( ac - > module ) ,
2008-09-11 18:36:28 -04:00
" Unsupported request type! " ) ;
ret = LDB_ERR_UNWILLING_TO_PERFORM ;
2006-07-17 22:36:04 +00:00
}
2007-02-22 01:54:40 +00:00
2008-09-11 18:36:28 -04:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
ac - > part_req [ ac - > num_requests ] . req = req ;
if ( ac - > req - > controls ) {
req - > controls = talloc_memdup ( req , ac - > req - > controls ,
talloc_get_size ( ac - > req - > controls ) ) ;
2007-02-22 01:54:40 +00:00
if ( req - > controls = = NULL ) {
2009-06-30 15:12:29 +10:00
ldb_oom ( ldb_module_get_ctx ( ac - > module ) ) ;
2007-02-22 01:54:40 +00:00
return LDB_ERR_OPERATIONS_ERROR ;
}
}
2006-09-14 03:15:30 +00:00
2007-01-17 16:55:37 +00:00
if ( partition ) {
2008-09-11 18:36:28 -04:00
ac - > part_req [ ac - > num_requests ] . module = partition - > module ;
2009-09-02 13:36:54 +10:00
if ( ! ldb_request_get_control ( req , DSDB_CONTROL_CURRENT_PARTITION_OID ) ) {
ret = ldb_request_add_control ( req ,
DSDB_CONTROL_CURRENT_PARTITION_OID ,
false , partition - > ctrl ) ;
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2007-01-17 16:55:37 +00:00
}
2008-09-11 18:36:28 -04:00
if ( req - > operation = = LDB_SEARCH ) {
/* If the search is for 'more' than this partition,
* then change the basedn , so a remote LDAP server
* doesn ' t object */
2009-09-02 13:36:54 +10:00
if ( ldb_dn_compare_base ( partition - > ctrl - > dn ,
2008-09-11 18:36:28 -04:00
req - > op . search . base ) ! = 0 ) {
2009-09-02 13:36:54 +10:00
req - > op . search . base = partition - > ctrl - > dn ;
2008-09-11 18:36:28 -04:00
}
}
} else {
2009-10-27 10:54:16 +11:00
/* make sure you put the module here, or
* or ldb_next_request ( ) will skip a module */
ac - > part_req [ ac - > num_requests ] . module = ac - > module ;
2006-07-17 22:36:04 +00:00
}
2007-01-17 16:55:37 +00:00
2006-07-29 01:13:53 +00:00
ac - > num_requests + + ;
2008-09-11 18:36:28 -04:00
2006-07-17 22:36:04 +00:00
return LDB_SUCCESS ;
}
2008-09-11 18:36:28 -04:00
static int partition_call_first ( struct partition_context * ac )
{
return partition_request ( ac - > part_req [ 0 ] . module , ac - > part_req [ 0 ] . req ) ;
}
2007-12-17 04:22:44 +01:00
/**
* Send a request down to all the partitions
*/
2006-07-29 01:13:53 +00:00
static int partition_send_all ( struct ldb_module * module ,
2007-03-13 00:59:06 +00:00
struct partition_context * ac ,
struct ldb_request * req )
2006-07-29 01:13:53 +00:00
{
int i ;
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2008-09-11 18:36:28 -04:00
int ret = partition_prep_request ( ac , NULL ) ;
2006-07-29 01:13:53 +00:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2008-09-11 18:36:28 -04:00
ret = partition_prep_request ( ac , data - > partitions [ i ] ) ;
2006-07-29 01:13:53 +00:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
}
2008-09-11 18:36:28 -04:00
/* fire the first one */
return partition_call_first ( ac ) ;
2006-07-29 01:13:53 +00:00
}
2007-12-17 04:22:44 +01:00
/**
* Figure out which backend a request needs to be aimed at . Some
* requests must be replicated to all backends
*/
2006-11-22 00:59:34 +00:00
static int partition_replicate ( struct ldb_module * module , struct ldb_request * req , struct ldb_dn * dn )
2006-07-29 01:13:53 +00:00
{
2008-09-11 18:36:28 -04:00
struct partition_context * ac ;
2007-01-17 16:11:46 +00:00
unsigned i ;
int ret ;
2009-09-02 13:36:54 +10:00
struct dsdb_partition * partition ;
2006-07-29 01:13:53 +00:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2008-09-11 18:36:28 -04:00
if ( ! data | | ! data - > partitions ) {
return ldb_next_request ( module , req ) ;
}
2009-10-14 12:57:03 +11:00
2009-10-27 10:54:16 +11:00
if ( req - > operation ! = LDB_SEARCH & & ldb_dn_is_special ( dn ) ) {
2007-03-13 00:59:06 +00:00
/* Is this a special DN, we need to replicate to every backend? */
for ( i = 0 ; data - > replicate & & data - > replicate [ i ] ; i + + ) {
if ( ldb_dn_compare ( data - > replicate [ i ] ,
dn ) = = 0 ) {
2008-09-11 18:36:28 -04:00
ac = partition_init_ctx ( module , req ) ;
2007-03-13 00:59:06 +00:00
if ( ! ac ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
2008-01-16 08:49:54 +11:00
return partition_send_all ( module , ac , req ) ;
2006-07-29 01:13:53 +00:00
}
}
}
2007-01-17 14:19:06 +00:00
/* Otherwise, we need to find the partition to fire it to */
/* Find partition */
2009-09-02 13:36:54 +10:00
partition = find_partition ( data , dn , req ) ;
2007-01-17 14:19:06 +00:00
if ( ! partition ) {
/*
* if we haven ' t found a matching partition
* pass the request to the main ldb
*
* TODO : we should maybe return an error here
* if it ' s not a special dn
*/
2008-01-16 08:49:54 +11:00
2007-01-17 14:19:06 +00:00
return ldb_next_request ( module , req ) ;
}
2008-09-11 18:36:28 -04:00
ac = partition_init_ctx ( module , req ) ;
if ( ! ac ) {
2007-01-17 14:19:06 +00:00
return LDB_ERR_OPERATIONS_ERROR ;
}
2006-07-29 01:13:53 +00:00
2008-09-11 18:36:28 -04:00
/* we need to add a control but we never touch the original request */
ret = partition_prep_request ( ac , partition ) ;
2007-01-17 16:11:46 +00:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2008-09-11 18:36:28 -04:00
/* fire the first one */
return partition_call_first ( ac ) ;
2006-07-29 01:13:53 +00:00
}
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
/* search */
static int partition_search ( struct ldb_module * module , struct ldb_request * req )
{
2009-10-14 12:57:03 +11:00
int ret ;
2008-01-16 08:49:54 +11:00
struct ldb_control * * saved_controls ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
/* Find backend */
2006-07-17 22:36:04 +00:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2009-09-13 18:12:05 +10:00
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
/* issue request */
/* (later) consider if we should be searching multiple
* partitions ( for ' invisible ' partition behaviour */
2008-09-11 18:36:28 -04:00
2007-02-22 01:54:40 +00:00
struct ldb_control * search_control = ldb_request_get_control ( req , LDB_CONTROL_SEARCH_OPTIONS_OID ) ;
2008-01-16 08:49:54 +11:00
struct ldb_control * domain_scope_control = ldb_request_get_control ( req , LDB_CONTROL_DOMAIN_SCOPE_OID ) ;
2007-02-22 01:54:40 +00:00
struct ldb_search_options_control * search_options = NULL ;
2009-09-13 18:12:05 +10:00
struct dsdb_partition * p ;
2010-02-22 11:19:10 +01:00
bool domain_scope = false , phantom_root = false ;
2009-09-13 18:12:05 +10:00
2009-10-14 12:57:03 +11:00
ret = partition_reload_if_required ( module , data ) ;
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2009-09-13 18:12:05 +10:00
p = find_partition ( data , NULL , req ) ;
if ( p ! = NULL ) {
/* the caller specified what partition they want the
* search - just pass it on
*/
return ldb_next_request ( p - > module , req ) ;
}
2010-02-22 11:19:10 +01:00
/* Get back the search options from the search control, and mark it as
* non - critical ( to make backends and also dcpromo happy ) .
*/
2007-02-22 01:54:40 +00:00
if ( search_control ) {
search_options = talloc_get_type ( search_control - > data , struct ldb_search_options_control ) ;
2010-02-22 11:19:10 +01:00
search_control - > critical = 0 ;
2007-02-22 01:54:40 +00:00
}
2010-02-22 11:19:10 +01:00
/* Remove the "domain_scope" control, so we don't confuse a backend
* server */
2008-01-16 08:49:54 +11:00
if ( domain_scope_control & & ! save_controls ( domain_scope_control , req , & saved_controls ) ) {
2009-06-30 15:12:29 +10:00
ldb_oom ( ldb_module_get_ctx ( module ) ) ;
2008-01-16 08:49:54 +11:00
return LDB_ERR_OPERATIONS_ERROR ;
}
2010-02-22 11:19:10 +01:00
/* Locate the options */
domain_scope = ( search_options
& & ( search_options - > search_options & LDB_SEARCH_OPTION_DOMAIN_SCOPE ) )
| | domain_scope_control ;
phantom_root = search_options
& & ( search_options - > search_options & LDB_SEARCH_OPTION_PHANTOM_ROOT ) ;
/* Remove handled options from the search control flag */
if ( search_options ) {
search_options - > search_options = search_options - > search_options
& ~ LDB_SEARCH_OPTION_DOMAIN_SCOPE
& ~ LDB_SEARCH_OPTION_PHANTOM_ROOT ;
2008-10-02 11:11:14 +02:00
}
2010-02-22 11:19:10 +01:00
if ( ( ! domain_scope ) | | phantom_root ) {
2009-10-14 12:57:03 +11:00
int i ;
2006-07-22 16:56:33 +00:00
struct partition_context * ac ;
2010-02-20 20:31:41 +01:00
2008-09-11 18:36:28 -04:00
ac = partition_init_ctx ( module , req ) ;
2007-01-11 09:38:21 +00:00
if ( ! ac ) {
2006-07-17 22:36:04 +00:00
return LDB_ERR_OPERATIONS_ERROR ;
}
2007-01-11 09:38:21 +00:00
2006-09-14 03:15:30 +00:00
/* Search from the base DN */
2006-11-22 00:59:34 +00:00
if ( ! req - > op . search . base | | ldb_dn_is_null ( req - > op . search . base ) ) {
2008-01-16 08:49:54 +11:00
return partition_send_all ( module , ac , req ) ;
2006-09-14 03:15:30 +00:00
}
2006-07-17 22:36:04 +00:00
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2008-10-03 12:21:53 -07:00
bool match = false , stop = false ;
/* Find all partitions under the search base
we match if :
1 ) the DN we are looking for exactly matches the partition
or
2 ) the DN we are looking for is a parent of the partition and it isn ' t
a scope base search
or
3 ) the DN we are looking for is a child of the partition
*/
2009-09-02 13:36:54 +10:00
if ( ldb_dn_compare ( data - > partitions [ i ] - > ctrl - > dn , req - > op . search . base ) = = 0 ) {
2008-10-03 12:21:53 -07:00
match = true ;
2010-01-08 09:00:15 +11:00
stop = true ;
2008-10-03 12:21:53 -07:00
}
if ( ! match & &
2009-09-02 13:36:54 +10:00
( ldb_dn_compare_base ( req - > op . search . base , data - > partitions [ i ] - > ctrl - > dn ) = = 0 & &
2008-10-03 12:21:53 -07:00
req - > op . search . scope ! = LDB_SCOPE_BASE ) ) {
match = true ;
}
if ( ! match & &
2009-09-02 13:36:54 +10:00
ldb_dn_compare_base ( data - > partitions [ i ] - > ctrl - > dn , req - > op . search . base ) = = 0 ) {
2008-10-03 12:21:53 -07:00
match = true ;
stop = true ; /* note that this relies on partition ordering */
}
if ( match ) {
2008-09-11 18:36:28 -04:00
ret = partition_prep_request ( ac , data - > partitions [ i ] ) ;
2006-07-17 22:36:04 +00:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
}
2008-10-03 12:21:53 -07:00
if ( stop ) break ;
2006-07-17 22:36:04 +00:00
}
2006-09-14 03:15:30 +00:00
/* Perhaps we didn't match any partitions. Try the main partition, only */
2006-07-29 01:13:53 +00:00
if ( ac - > num_requests = = 0 ) {
2007-01-11 09:38:21 +00:00
talloc_free ( ac ) ;
2006-09-14 03:15:30 +00:00
return ldb_next_request ( module , req ) ;
2006-07-17 22:36:04 +00:00
}
2008-09-11 18:36:28 -04:00
/* fire the first one */
return partition_call_first ( ac ) ;
2006-07-17 22:36:04 +00:00
} else {
2007-03-13 00:59:06 +00:00
return partition_replicate ( module , req , req - > op . search . base ) ;
2006-07-17 22:36:04 +00:00
}
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
/* add */
static int partition_add ( struct ldb_module * module , struct ldb_request * req )
{
2006-07-29 01:13:53 +00:00
return partition_replicate ( module , req , req - > op . add . message - > dn ) ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
/* modify */
static int partition_modify ( struct ldb_module * module , struct ldb_request * req )
{
2006-07-29 01:13:53 +00:00
return partition_replicate ( module , req , req - > op . mod . message - > dn ) ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
/* delete */
static int partition_delete ( struct ldb_module * module , struct ldb_request * req )
{
2006-07-29 01:13:53 +00:00
return partition_replicate ( module , req , req - > op . del . dn ) ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
/* rename */
static int partition_rename ( struct ldb_module * module , struct ldb_request * req )
{
/* Find backend */
2009-09-02 13:36:54 +10:00
struct dsdb_partition * backend , * backend2 ;
2007-07-09 12:31:35 +00:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2009-06-27 14:16:19 +02:00
/* Skip the lot if 'data' isn't here yet (initialisation) */
2007-07-09 12:31:35 +00:00
if ( ! data ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
2009-09-02 13:36:54 +10:00
backend = find_partition ( data , req - > op . rename . olddn , req ) ;
backend2 = find_partition ( data , req - > op . rename . newdn , req ) ;
2007-07-09 12:31:35 +00:00
if ( ( backend & & ! backend2 ) | | ( ! backend & & backend2 ) ) {
return LDB_ERR_AFFECTS_MULTIPLE_DSAS ;
}
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
2007-07-09 12:31:35 +00:00
if ( backend ! = backend2 ) {
2009-06-30 15:12:29 +10:00
ldb_asprintf_errstring ( ldb_module_get_ctx ( module ) ,
2007-07-09 12:31:35 +00:00
" Cannot rename from %s in %s to %s in %s: %s " ,
ldb_dn_get_linearized ( req - > op . rename . olddn ) ,
2009-09-02 13:36:54 +10:00
ldb_dn_get_linearized ( backend - > ctrl - > dn ) ,
2007-07-09 12:31:35 +00:00
ldb_dn_get_linearized ( req - > op . rename . newdn ) ,
2009-09-02 13:36:54 +10:00
ldb_dn_get_linearized ( backend2 - > ctrl - > dn ) ,
2007-07-09 12:31:35 +00:00
ldb_strerror ( LDB_ERR_AFFECTS_MULTIPLE_DSAS ) ) ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
return LDB_ERR_AFFECTS_MULTIPLE_DSAS ;
}
2006-07-29 01:13:53 +00:00
return partition_replicate ( module , req , req - > op . rename . olddn ) ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
/* start a transaction */
static int partition_start_trans ( struct ldb_module * module )
{
2006-07-10 11:24:46 +00:00
int i , ret ;
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
/* Look at base DN */
/* Figure out which partition it is under */
2008-09-11 18:36:28 -04:00
/* Skip the lot if 'data' isn't here yet (initialization) */
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_start_trans() -> (metadata partition) " ) ;
}
2006-07-10 11:24:46 +00:00
ret = ldb_next_start_trans ( module ) ;
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2009-10-14 14:13:52 +11:00
ret = partition_reload_if_required ( module , data ) ;
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2006-07-10 11:24:46 +00:00
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_start_trans() -> %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ) ;
}
ret = ldb_next_start_trans ( data - > partitions [ i ] - > module ) ;
2006-07-10 11:24:46 +00:00
if ( ret ! = LDB_SUCCESS ) {
/* Back it out, if it fails on one */
for ( i - - ; i > = 0 ; i - - ) {
2009-10-27 10:54:16 +11:00
ldb_next_del_trans ( data - > partitions [ i ] - > module ) ;
2006-07-10 11:24:46 +00:00
}
2008-09-11 18:36:28 -04:00
ldb_next_del_trans ( module ) ;
2006-07-10 11:24:46 +00:00
return ret ;
}
}
2009-10-23 22:46:09 +11:00
data - > in_transaction + + ;
2006-07-10 11:24:46 +00:00
return LDB_SUCCESS ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
2009-09-03 18:32:42 +10:00
/* prepare for a commit */
static int partition_prepare_commit ( struct ldb_module * module )
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
{
2009-09-03 18:32:42 +10:00
int i ;
2006-07-10 11:24:46 +00:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2009-03-31 15:08:36 +11:00
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2009-09-03 18:32:42 +10:00
int ret ;
2009-03-31 15:08:36 +11:00
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_prepare_commit() -> %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ) ;
2009-03-31 15:08:36 +11:00
}
2009-10-27 10:54:16 +11:00
ret = ldb_next_prepare_commit ( data - > partitions [ i ] - > module ) ;
2009-03-31 15:08:36 +11:00
if ( ret ! = LDB_SUCCESS ) {
2010-02-20 20:31:41 +01:00
ldb_asprintf_errstring ( module - > ldb , " prepare_commit error on %s: %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ,
ldb_errstring ( module - > ldb ) ) ;
2009-03-31 15:08:36 +11:00
return ret ;
}
}
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_prepare_commit() -> (metadata partition) " ) ;
}
2009-09-03 18:32:42 +10:00
return ldb_next_prepare_commit ( module ) ;
}
2009-03-31 15:08:36 +11:00
2009-09-03 18:32:42 +10:00
/* end a transaction */
static int partition_end_trans ( struct ldb_module * module )
{
2009-10-27 12:13:46 +11:00
int i , ret , ret2 ;
2009-09-03 18:32:42 +10:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2008-09-11 18:36:28 -04:00
2009-10-27 12:13:46 +11:00
ret = LDB_SUCCESS ;
if ( data - > in_transaction = = 0 ) {
DEBUG ( 0 , ( " partition end transaction mismatch \n " ) ) ;
ret = LDB_ERR_OPERATIONS_ERROR ;
} else {
data - > in_transaction - - ;
}
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_end_trans() -> %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ) ;
}
2009-10-27 12:13:46 +11:00
ret2 = ldb_next_end_trans ( data - > partitions [ i ] - > module ) ;
if ( ret2 ! = LDB_SUCCESS ) {
2010-02-20 20:31:41 +01:00
ldb_asprintf_errstring ( module - > ldb , " end_trans error on %s: %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ,
ldb_errstring ( module - > ldb ) ) ;
2009-10-27 12:13:46 +11:00
ret = ret2 ;
2006-07-10 11:24:46 +00:00
}
}
2008-09-11 18:36:28 -04:00
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_end_trans() -> (metadata partition) " ) ;
}
2009-10-27 12:13:46 +11:00
ret2 = ldb_next_end_trans ( module ) ;
if ( ret2 ! = LDB_SUCCESS ) {
ret = ret2 ;
}
return ret ;
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
}
/* delete a transaction */
static int partition_del_trans ( struct ldb_module * module )
{
2009-09-03 18:32:42 +10:00
int i , ret , final_ret = LDB_SUCCESS ;
2006-07-10 11:24:46 +00:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_del_trans() -> %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ) ;
}
ret = ldb_next_del_trans ( data - > partitions [ i ] - > module ) ;
2006-07-10 11:24:46 +00:00
if ( ret ! = LDB_SUCCESS ) {
2010-02-20 20:31:41 +01:00
ldb_asprintf_errstring ( module - > ldb , " del_trans error on %s: %s " ,
ldb_dn_get_linearized ( data - > partitions [ i ] - > ctrl - > dn ) ,
ldb_errstring ( module - > ldb ) ) ;
2009-09-03 18:32:42 +10:00
final_ret = ret ;
2006-07-10 11:24:46 +00:00
}
2009-09-03 18:32:42 +10:00
}
2009-10-23 22:46:09 +11:00
if ( data - > in_transaction = = 0 ) {
DEBUG ( 0 , ( " partition del transaction mismatch \n " ) ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
data - > in_transaction - - ;
2009-10-27 10:54:16 +11:00
if ( ( module & & module - > ldb - > flags & LDB_FLG_ENABLE_TRACING ) ) {
ldb_debug ( module - > ldb , LDB_DEBUG_TRACE , " partition_del_trans() -> (metadata partition) " ) ;
}
2009-09-03 18:32:42 +10:00
ret = ldb_next_del_trans ( module ) ;
if ( ret ! = LDB_SUCCESS ) {
final_ret = ret ;
2006-07-10 11:24:46 +00:00
}
2009-09-03 18:32:42 +10:00
return final_ret ;
2006-07-10 11:24:46 +00:00
}
2009-10-14 12:57:03 +11:00
int partition_primary_sequence_number ( struct ldb_module * module , TALLOC_CTX * mem_ctx ,
enum ldb_sequence_type type , uint64_t * seq_number )
{
int ret ;
struct ldb_result * res ;
struct ldb_seqnum_request * tseq ;
struct ldb_request * treq ;
struct ldb_seqnum_result * seqr ;
res = talloc_zero ( mem_ctx , struct ldb_result ) ;
if ( res = = NULL ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq = talloc_zero ( res , struct ldb_seqnum_request ) ;
if ( tseq = = NULL ) {
talloc_free ( res ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq - > type = type ;
ret = ldb_build_extended_req ( & treq , ldb_module_get_ctx ( module ) , res ,
LDB_EXTENDED_SEQUENCE_NUMBER ,
tseq ,
NULL ,
res ,
ldb_extended_default_callback ,
NULL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
ret = ldb_next_request ( module , treq ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
ret = ldb_wait ( treq - > handle , LDB_WAIT_ALL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
seqr = talloc_get_type ( res - > extended - > data ,
struct ldb_seqnum_result ) ;
if ( seqr - > flags & LDB_SEQ_TIMESTAMP_SEQUENCE ) {
ret = LDB_ERR_OPERATIONS_ERROR ;
ldb_set_errstring ( ldb_module_get_ctx ( module ) , " Primary backend in partitions module returned a timestamp based seq number (must return a normal number) " ) ;
talloc_free ( res ) ;
return ret ;
} else {
* seq_number = seqr - > seq_num ;
}
talloc_free ( res ) ;
return LDB_SUCCESS ;
}
2008-10-15 14:03:20 -04:00
/* FIXME: This function is still semi-async */
2006-07-10 11:24:46 +00:00
static int partition_sequence_number ( struct ldb_module * module , struct ldb_request * req )
{
int i , ret ;
uint64_t seq_number = 0 ;
2006-09-21 06:44:12 +00:00
uint64_t timestamp_sequence = 0 ;
uint64_t timestamp = 0 ;
2006-07-10 11:24:46 +00:00
struct partition_private_data * data = talloc_get_type ( module - > private_data ,
struct partition_private_data ) ;
2008-10-15 14:03:20 -04:00
struct ldb_seqnum_request * seq ;
struct ldb_seqnum_result * seqr ;
struct ldb_request * treq ;
struct ldb_seqnum_request * tseq ;
struct ldb_seqnum_result * tseqr ;
struct ldb_extended * ext ;
struct ldb_result * res ;
2009-09-12 15:24:31 +10:00
struct dsdb_partition * p ;
2009-09-13 18:12:05 +10:00
p = find_partition ( data , NULL , req ) ;
2009-09-12 15:24:31 +10:00
if ( p ! = NULL ) {
/* the caller specified what partition they want the
* sequence number operation on - just pass it on
*/
return ldb_next_request ( p - > module , req ) ;
}
2006-07-11 02:04:43 +00:00
2008-10-15 14:03:20 -04:00
seq = talloc_get_type ( req - > op . extended . data , struct ldb_seqnum_request ) ;
switch ( seq - > type ) {
2006-09-21 06:44:12 +00:00
case LDB_SEQ_NEXT :
case LDB_SEQ_HIGHEST_SEQ :
2008-10-15 14:03:20 -04:00
2009-10-14 12:57:03 +11:00
ret = partition_primary_sequence_number ( module , req , seq - > type , & seq_number ) ;
2009-10-06 19:27:17 +02:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
2006-09-21 06:44:12 +00:00
}
2009-10-06 19:27:17 +02:00
2009-06-27 14:16:19 +02:00
/* Skip the lot if 'data' isn't here yet (initialisation) */
2006-09-21 06:44:12 +00:00
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2008-09-11 18:36:28 -04:00
2008-10-15 14:03:20 -04:00
res = talloc_zero ( req , struct ldb_result ) ;
if ( res = = NULL ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq = talloc_zero ( res , struct ldb_seqnum_request ) ;
if ( tseq = = NULL ) {
talloc_free ( res ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq - > type = seq - > type ;
2009-06-30 15:12:29 +10:00
ret = ldb_build_extended_req ( & treq , ldb_module_get_ctx ( module ) , res ,
2008-10-15 14:03:20 -04:00
LDB_EXTENDED_SEQUENCE_NUMBER ,
tseq ,
NULL ,
res ,
ldb_extended_default_callback ,
NULL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
2008-07-15 22:10:42 +10:00
}
2009-09-02 13:36:54 +10:00
if ( ! ldb_request_get_control ( treq , DSDB_CONTROL_CURRENT_PARTITION_OID ) ) {
ret = ldb_request_add_control ( treq ,
DSDB_CONTROL_CURRENT_PARTITION_OID ,
false , data - > partitions [ i ] - > ctrl ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
ret = partition_request ( data - > partitions [ i ] - > module , treq ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
ret = ldb_wait ( treq - > handle , LDB_WAIT_ALL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
tseqr = talloc_get_type ( res - > extended - > data ,
struct ldb_seqnum_result ) ;
if ( tseqr - > flags & LDB_SEQ_TIMESTAMP_SEQUENCE ) {
timestamp_sequence = MAX ( timestamp_sequence ,
tseqr - > seq_num ) ;
2006-09-21 06:44:12 +00:00
} else {
2008-10-15 14:03:20 -04:00
seq_number + = tseqr - > seq_num ;
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
talloc_free ( res ) ;
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
/* fall through */
2006-09-21 06:44:12 +00:00
case LDB_SEQ_HIGHEST_TIMESTAMP :
2008-10-15 14:03:20 -04:00
res = talloc_zero ( req , struct ldb_result ) ;
if ( res = = NULL ) {
2006-09-21 06:44:12 +00:00
return LDB_ERR_OPERATIONS_ERROR ;
}
2008-10-15 14:03:20 -04:00
tseq = talloc_zero ( res , struct ldb_seqnum_request ) ;
if ( tseq = = NULL ) {
talloc_free ( res ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq - > type = LDB_SEQ_HIGHEST_TIMESTAMP ;
2009-06-30 15:12:29 +10:00
ret = ldb_build_extended_req ( & treq , ldb_module_get_ctx ( module ) , res ,
2008-10-15 14:03:20 -04:00
LDB_EXTENDED_SEQUENCE_NUMBER ,
tseq ,
NULL ,
res ,
ldb_extended_default_callback ,
NULL ) ;
2006-07-10 11:24:46 +00:00
if ( ret ! = LDB_SUCCESS ) {
2008-10-15 14:03:20 -04:00
talloc_free ( res ) ;
2006-07-10 11:24:46 +00:00
return ret ;
}
2008-09-11 18:36:28 -04:00
2008-10-15 14:03:20 -04:00
ret = ldb_next_request ( module , treq ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
ret = ldb_wait ( treq - > handle , LDB_WAIT_ALL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
tseqr = talloc_get_type ( res - > extended - > data ,
struct ldb_seqnum_result ) ;
timestamp = tseqr - > seq_num ;
talloc_free ( res ) ;
2009-06-27 14:16:19 +02:00
/* Skip the lot if 'data' isn't here yet (initialisation) */
2006-09-21 06:44:12 +00:00
for ( i = 0 ; data & & data - > partitions & & data - > partitions [ i ] ; i + + ) {
2008-09-11 18:36:28 -04:00
2008-10-15 14:03:20 -04:00
res = talloc_zero ( req , struct ldb_result ) ;
if ( res = = NULL ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq = talloc_zero ( res , struct ldb_seqnum_request ) ;
if ( tseq = = NULL ) {
talloc_free ( res ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
tseq - > type = LDB_SEQ_HIGHEST_TIMESTAMP ;
2009-06-30 15:12:29 +10:00
ret = ldb_build_extended_req ( & treq , ldb_module_get_ctx ( module ) , res ,
2008-10-15 14:03:20 -04:00
LDB_EXTENDED_SEQUENCE_NUMBER ,
tseq ,
NULL ,
res ,
ldb_extended_default_callback ,
NULL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
2009-09-02 13:36:54 +10:00
if ( ! ldb_request_get_control ( treq , DSDB_CONTROL_CURRENT_PARTITION_OID ) ) {
ret = ldb_request_add_control ( treq ,
DSDB_CONTROL_CURRENT_PARTITION_OID ,
false , data - > partitions [ i ] - > ctrl ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
2008-12-10 17:23:44 +11:00
}
2008-10-15 14:03:20 -04:00
ret = partition_request ( data - > partitions [ i ] - > module , treq ) ;
2006-09-21 06:44:12 +00:00
if ( ret ! = LDB_SUCCESS ) {
2008-10-15 14:03:20 -04:00
talloc_free ( res ) ;
2006-09-21 06:44:12 +00:00
return ret ;
}
2008-10-15 14:03:20 -04:00
ret = ldb_wait ( treq - > handle , LDB_WAIT_ALL ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( res ) ;
return ret ;
}
tseqr = talloc_get_type ( res - > extended - > data ,
struct ldb_seqnum_result ) ;
timestamp = MAX ( timestamp , tseqr - > seq_num ) ;
talloc_free ( res ) ;
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
2006-09-21 06:44:12 +00:00
break ;
}
2008-10-15 14:03:20 -04:00
ext = talloc_zero ( req , struct ldb_extended ) ;
if ( ! ext ) {
return LDB_ERR_OPERATIONS_ERROR ;
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
seqr = talloc_zero ( ext , struct ldb_seqnum_result ) ;
if ( seqr = = NULL ) {
talloc_free ( ext ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
ext - > oid = LDB_EXTENDED_SEQUENCE_NUMBER ;
ext - > data = seqr ;
2006-09-21 06:44:12 +00:00
2008-10-15 14:03:20 -04:00
switch ( seq - > type ) {
2006-09-21 06:44:12 +00:00
case LDB_SEQ_NEXT :
case LDB_SEQ_HIGHEST_SEQ :
2008-09-11 18:36:28 -04:00
2006-09-21 06:44:12 +00:00
/* Has someone above set a timebase sequence? */
if ( timestamp_sequence ) {
2008-10-15 14:03:20 -04:00
seqr - > seq_num = ( ( ( unsigned long long ) timestamp < < 24 ) | ( seq_number & 0xFFFFFF ) ) ;
2006-09-21 06:44:12 +00:00
} else {
2008-10-15 14:03:20 -04:00
seqr - > seq_num = seq_number ;
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
if ( timestamp_sequence > seqr - > seq_num ) {
seqr - > seq_num = timestamp_sequence ;
seqr - > flags | = LDB_SEQ_TIMESTAMP_SEQUENCE ;
2006-09-21 06:44:12 +00:00
}
2008-10-15 14:03:20 -04:00
seqr - > flags | = LDB_SEQ_GLOBAL_SEQUENCE ;
2006-09-21 06:44:12 +00:00
break ;
case LDB_SEQ_HIGHEST_TIMESTAMP :
2008-10-15 14:03:20 -04:00
seqr - > seq_num = timestamp ;
2006-09-21 06:44:12 +00:00
break ;
}
2008-10-15 14:03:20 -04:00
if ( seq - > type = = LDB_SEQ_NEXT ) {
seqr - > seq_num + + ;
2006-07-10 11:24:46 +00:00
}
2008-10-15 14:03:20 -04:00
/* send request done */
return ldb_module_done ( req , NULL , ext , LDB_SUCCESS ) ;
2006-07-10 11:24:46 +00:00
}
2007-01-07 19:11:27 +00:00
/* extended */
static int partition_extended ( struct ldb_module * module , struct ldb_request * req )
{
2008-09-11 18:36:28 -04:00
struct partition_private_data * data ;
2007-01-07 19:11:27 +00:00
struct partition_context * ac ;
2009-10-14 12:57:03 +11:00
int ret ;
2007-01-07 19:11:27 +00:00
2008-09-11 18:36:28 -04:00
data = talloc_get_type ( module - > private_data , struct partition_private_data ) ;
2009-10-02 10:28:29 +10:00
if ( ! data ) {
2008-09-11 18:36:28 -04:00
return ldb_next_request ( module , req ) ;
}
2009-10-14 12:57:03 +11:00
ret = partition_reload_if_required ( module , data ) ;
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2008-10-15 14:03:20 -04:00
if ( strcmp ( req - > op . extended . oid , LDB_EXTENDED_SEQUENCE_NUMBER ) = = 0 ) {
return partition_sequence_number ( module , req ) ;
}
2009-10-02 10:28:29 +10:00
if ( strcmp ( req - > op . extended . oid , DSDB_EXTENDED_CREATE_PARTITION_OID ) = = 0 ) {
return partition_create ( module , req ) ;
}
2007-01-07 19:11:27 +00:00
/*
* as the extended operation has no dn
* we need to send it to all partitions
*/
2008-09-11 18:36:28 -04:00
ac = partition_init_ctx ( module , req ) ;
2007-01-11 09:38:21 +00:00
if ( ! ac ) {
2007-01-07 19:11:27 +00:00
return LDB_ERR_OPERATIONS_ERROR ;
}
2008-09-11 18:36:28 -04:00
2008-01-16 08:49:54 +11:00
return partition_send_all ( module , ac , req ) ;
2007-01-07 19:11:27 +00:00
}
2008-02-20 04:33:43 +01:00
_PUBLIC_ const struct ldb_module_ops ldb_partition_module_ops = {
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
. name = " partition " ,
. init_context = partition_init ,
. search = partition_search ,
. add = partition_add ,
. modify = partition_modify ,
. del = partition_delete ,
. rename = partition_rename ,
2007-01-07 19:11:27 +00:00
. extended = partition_extended ,
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
. start_transaction = partition_start_trans ,
2009-09-03 18:32:42 +10:00
. prepare_commit = partition_prepare_commit ,
r16264: Add, but do not yet enable, the partitions module.
This required changes to the rootDSE module, to allow registration of
partitions. In doing so I renamed the 'register' operation to
'register_control' and 'register_partition', which changed a few more
modules.
Due to the behaviour of certain LDAP servers, we create the baseDN
entry in two parts: Firstly, we allow the admin to export a simple
LDIF file to add to their server. Then we perform a modify to add the
remaining attributes.
To delete all users in partitions, we must now search and delete all
objects in the partition, rather than a simple search from the root.
Against LDAP, this might not delete all objects, so we allow this to
fail.
In testing, we found that the 'Domain Controllers' container was
misnamed, and should be 'CN=', rather than 'OU='.
To avoid the Templates being found in default searches, they have been
moved to CN=Templates from CN=Templates,${BASEDN}.
Andrew Bartlett
(This used to be commit b49a4fbb57f10726bd288fdc9fc95c0cbbe9094a)
2006-06-15 18:04:24 +00:00
. end_transaction = partition_end_trans ,
. del_transaction = partition_del_trans ,
} ;