2006-08-13 23:58:04 +00:00
/*
ldb database mapping module
2005-08-24 22:06:26 +00:00
Copyright ( C ) Jelmer Vernooij 2005
2006-08-13 23:58:04 +00:00
Copyright ( C ) Martin Kuehl < mkhl @ samba . org > 2006
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
* NOTICE : this module is NOT released under the GNU LGPL license as
* other ldb code . This module is release under the GNU GPL v2 or
* later license .
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +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
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
2005-08-24 22:06:26 +00:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2006-08-13 23:58:04 +00:00
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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
2005-08-24 22:06:26 +00:00
*/
2006-08-13 23:58:04 +00:00
/*
* Name : ldb
*
* Component : ldb ldb_map module
*
* Description : Map portions of data into a different format on a
* remote partition .
*
* Author : Jelmer Vernooij , Martin Kuehl
*/
2005-08-24 22:06:26 +00:00
# include "includes.h"
2006-01-10 16:48:32 +00:00
# include "ldb/include/includes.h"
2005-08-31 18:33:57 +00:00
# include "ldb/modules/ldb_map.h"
2006-08-13 23:58:04 +00:00
# include "ldb/modules/ldb_map_private.h"
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Description of the provided ldb requests:
2005-08-29 15:54:10 +00:00
- special attribute ' isMapped '
2006-08-13 23:58:04 +00:00
- search :
- if parse tree can be split
- search remote records w / remote attrs and parse tree
- otherwise
- enumerate all remote records
- for each remote result
- map remote result to local message
- search local result
- is present
- merge local into remote result
- run callback on merged result
- otherwise
- run callback on remote result
- add :
- split message into local and remote part
- if local message is not empty
- add isMapped to local message
- add local message
- add remote message
- modify :
- split message into local and remote part
- if local message is not empty
- add isMapped to local message
- search for local record
- if present
- modify local record
- otherwise
- add local message
- modify remote record
- delete :
- search for local record
- if present
- delete local record
- delete remote record
- rename :
- search for local record
- if present
- rename local record
- modify local isMapped
- rename remote record
*/
2005-08-27 23:47:17 +00:00
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* Private data structures
* = = = = = = = = = = = = = = = = = = = = = = = */
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Global private data */
/* Extract mappings from private data. */
const struct ldb_map_context * map_get_context ( struct ldb_module * module )
2005-08-29 15:54:10 +00:00
{
2006-08-13 23:58:04 +00:00
const struct map_private * data = talloc_get_type ( module - > private_data , struct map_private ) ;
2006-08-22 06:01:47 +00:00
return data - > context ;
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
/* Create a generic request context. */
static struct map_context * map_init_context ( struct ldb_handle * h , struct ldb_request * req )
2005-08-29 15:54:10 +00:00
{
2006-08-13 23:58:04 +00:00
struct map_context * ac ;
2005-08-29 17:31:19 +00:00
2006-08-13 23:58:04 +00:00
ac = talloc_zero ( h , struct map_context ) ;
if ( ac = = NULL ) {
map_oom ( h - > module ) ;
return NULL ;
2005-08-29 17:31:19 +00:00
}
2006-08-13 23:58:04 +00:00
ac - > module = h - > module ;
ac - > orig_req = req ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
return ac ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Create a search request context. */
struct map_search_context * map_init_search_context ( struct map_context * ac , struct ldb_reply * ares )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
struct map_search_context * sc ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
sc = talloc_zero ( ac , struct map_search_context ) ;
if ( sc = = NULL ) {
map_oom ( ac - > module ) ;
return NULL ;
2005-08-25 15:25:22 +00:00
}
2006-08-13 23:58:04 +00:00
sc - > ac = ac ;
sc - > local_res = NULL ;
sc - > remote_res = ares ;
return sc ;
2005-08-24 22:06:26 +00:00
}
2006-08-13 23:58:04 +00:00
/* Create a request context and handle. */
struct ldb_handle * map_init_handle ( struct ldb_request * req , struct ldb_module * module )
2005-09-01 15:33:31 +00:00
{
2006-08-13 23:58:04 +00:00
struct map_context * ac ;
struct ldb_handle * h ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
h = talloc_zero ( req , struct ldb_handle ) ;
if ( h = = NULL ) {
map_oom ( module ) ;
return NULL ;
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
h - > module = module ;
ac = map_init_context ( h , req ) ;
if ( ac = = NULL ) {
talloc_free ( h ) ;
return NULL ;
}
2005-09-01 18:04:23 +00:00
2006-08-13 23:58:04 +00:00
h - > private_data = ( void * ) ac ;
2005-09-01 18:04:23 +00:00
2006-08-13 23:58:04 +00:00
h - > state = LDB_ASYNC_INIT ;
h - > status = LDB_SUCCESS ;
2005-09-01 18:04:23 +00:00
2006-08-13 23:58:04 +00:00
return h ;
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
/* Dealing with DNs for different partitions
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* Check whether any data should be stored in the local partition. */
BOOL map_check_local_db ( struct ldb_module * module )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
const struct ldb_map_context * data = map_get_context ( module ) ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
if ( ! data - > remote_base_dn | | ! data - > local_base_dn ) {
return False ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
return True ;
}
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Copy a DN with the base DN of the local partition. */
2006-11-22 00:59:34 +00:00
static struct ldb_dn * ldb_dn_rebase_local ( void * mem_ctx , const struct ldb_map_context * data , struct ldb_dn * dn )
2006-08-13 23:58:04 +00:00
{
2006-11-22 00:59:34 +00:00
struct ldb_dn * new_dn ;
new_dn = ldb_dn_copy ( mem_ctx , dn ) ;
if ( ! ldb_dn_validate ( new_dn ) ) {
talloc_free ( new_dn ) ;
return NULL ;
}
/* may be we don't need to rebase at all */
if ( ! data - > remote_base_dn | | ! data - > local_base_dn ) {
return new_dn ;
}
if ( ! ldb_dn_remove_base_components ( new_dn , ldb_dn_get_comp_num ( data - > remote_base_dn ) ) ) {
talloc_free ( new_dn ) ;
return NULL ;
}
if ( ! ldb_dn_add_base ( new_dn , data - > local_base_dn ) ) {
talloc_free ( new_dn ) ;
return NULL ;
}
return new_dn ;
2006-08-13 23:58:04 +00:00
}
/* Copy a DN with the base DN of the remote partition. */
2006-11-22 00:59:34 +00:00
static struct ldb_dn * ldb_dn_rebase_remote ( void * mem_ctx , const struct ldb_map_context * data , struct ldb_dn * dn )
2006-08-13 23:58:04 +00:00
{
2006-11-22 00:59:34 +00:00
struct ldb_dn * new_dn ;
new_dn = ldb_dn_copy ( mem_ctx , dn ) ;
if ( ! ldb_dn_validate ( new_dn ) ) {
talloc_free ( new_dn ) ;
return NULL ;
}
/* may be we don't need to rebase at all */
if ( ! data - > remote_base_dn | | ! data - > local_base_dn ) {
return new_dn ;
}
if ( ! ldb_dn_remove_base_components ( new_dn , ldb_dn_get_comp_num ( data - > local_base_dn ) ) ) {
talloc_free ( new_dn ) ;
return NULL ;
}
if ( ! ldb_dn_add_base ( new_dn , data - > remote_base_dn ) ) {
talloc_free ( new_dn ) ;
return NULL ;
}
return new_dn ;
2006-08-13 23:58:04 +00:00
}
/* Run a request and make sure it targets the remote partition. */
/* TODO: free old DNs and messages? */
int ldb_next_remote_request ( struct ldb_module * module , struct ldb_request * request )
{
const struct ldb_map_context * data = map_get_context ( module ) ;
struct ldb_message * msg ;
switch ( request - > operation ) {
case LDB_SEARCH :
if ( request - > op . search . base ) {
request - > op . search . base = ldb_dn_rebase_remote ( request , data , request - > op . search . base ) ;
} else {
request - > op . search . base = data - > remote_base_dn ;
/* TODO: adjust scope? */
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
break ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
case LDB_ADD :
msg = ldb_msg_copy_shallow ( request , request - > op . add . message ) ;
msg - > dn = ldb_dn_rebase_remote ( msg , data , msg - > dn ) ;
request - > op . add . message = msg ;
break ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
case LDB_MODIFY :
msg = ldb_msg_copy_shallow ( request , request - > op . mod . message ) ;
msg - > dn = ldb_dn_rebase_remote ( msg , data , msg - > dn ) ;
request - > op . mod . message = msg ;
break ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
case LDB_DELETE :
request - > op . del . dn = ldb_dn_rebase_remote ( request , data , request - > op . del . dn ) ;
break ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
case LDB_RENAME :
request - > op . rename . olddn = ldb_dn_rebase_remote ( request , data , request - > op . rename . olddn ) ;
request - > op . rename . newdn = ldb_dn_rebase_remote ( request , data , request - > op . rename . newdn ) ;
break ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
default :
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" Invalid remote request! \n " ) ;
return LDB_ERR_OPERATIONS_ERROR ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
return ldb_next_request ( module , request ) ;
}
2005-08-27 23:47:17 +00:00
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Finding mappings for attributes and objectClasses
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
2005-08-27 18:30:00 +00:00
2006-08-13 23:58:04 +00:00
/* Find an objectClass mapping by the local name. */
static const struct ldb_map_objectclass * map_objectclass_find_local ( const struct ldb_map_context * data , const char * name )
{
int i ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
for ( i = 0 ; data - > objectclass_maps & & data - > objectclass_maps [ i ] . local_name ; i + + ) {
if ( ldb_attr_cmp ( data - > objectclass_maps [ i ] . local_name , name ) = = 0 ) {
return & data - > objectclass_maps [ i ] ;
}
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
return NULL ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Find an objectClass mapping by the remote name. */
static const struct ldb_map_objectclass * map_objectclass_find_remote ( const struct ldb_map_context * data , const char * name )
{
int i ;
for ( i = 0 ; data - > objectclass_maps & & data - > objectclass_maps [ i ] . remote_name ; i + + ) {
if ( ldb_attr_cmp ( data - > objectclass_maps [ i ] . remote_name , name ) = = 0 ) {
return & data - > objectclass_maps [ i ] ;
}
2005-08-25 15:25:22 +00:00
}
2006-08-13 23:58:04 +00:00
return NULL ;
2005-08-25 15:25:22 +00:00
}
2006-08-13 23:58:04 +00:00
/* Find an attribute mapping by the local name. */
const struct ldb_map_attribute * map_attr_find_local ( const struct ldb_map_context * data , const char * name )
2005-08-25 15:25:22 +00:00
{
int i ;
2006-08-13 23:58:04 +00:00
for ( i = 0 ; data - > attribute_maps [ i ] . local_name ; i + + ) {
if ( ldb_attr_cmp ( data - > attribute_maps [ i ] . local_name , name ) = = 0 ) {
return & data - > attribute_maps [ i ] ;
}
}
for ( i = 0 ; data - > attribute_maps [ i ] . local_name ; i + + ) {
if ( ldb_attr_cmp ( data - > attribute_maps [ i ] . local_name , " * " ) = = 0 ) {
return & data - > attribute_maps [ i ] ;
}
}
2005-08-27 15:13:15 +00:00
2006-08-13 23:58:04 +00:00
return NULL ;
}
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Find an attribute mapping by the remote name. */
const struct ldb_map_attribute * map_attr_find_remote ( const struct ldb_map_context * data , const char * name )
{
2006-08-22 06:01:47 +00:00
const struct ldb_map_attribute * map ;
2006-08-13 23:58:04 +00:00
const struct ldb_map_attribute * wildcard = NULL ;
int i , j ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
for ( i = 0 ; data - > attribute_maps [ i ] . local_name ; i + + ) {
2006-08-22 06:01:47 +00:00
map = & data - > attribute_maps [ i ] ;
if ( ldb_attr_cmp ( map - > local_name , " * " ) = = 0 ) {
2006-08-13 23:58:04 +00:00
wildcard = & data - > attribute_maps [ i ] ;
}
2005-08-25 15:25:22 +00:00
2006-08-22 06:01:47 +00:00
switch ( map - > type ) {
2005-08-27 23:47:17 +00:00
case MAP_IGNORE :
2006-08-13 23:58:04 +00:00
break ;
2005-08-27 18:30:00 +00:00
2005-08-27 23:47:17 +00:00
case MAP_KEEP :
2006-08-22 06:01:47 +00:00
if ( ldb_attr_cmp ( map - > local_name , name ) = = 0 ) {
return map ;
2006-08-13 23:58:04 +00:00
}
2005-08-27 23:47:17 +00:00
break ;
2006-08-13 23:58:04 +00:00
case MAP_RENAME :
2005-08-27 23:47:17 +00:00
case MAP_CONVERT :
2006-08-22 06:01:47 +00:00
if ( ldb_attr_cmp ( map - > u . rename . remote_name , name ) = = 0 ) {
return map ;
2006-08-13 23:58:04 +00:00
}
2005-08-27 23:47:17 +00:00
break ;
2006-08-13 23:58:04 +00:00
case MAP_GENERATE :
2006-08-22 06:01:47 +00:00
for ( j = 0 ; map - > u . generate . remote_names & & map - > u . generate . remote_names [ j ] ; j + + ) {
if ( ldb_attr_cmp ( map - > u . generate . remote_names [ j ] , name ) = = 0 ) {
return map ;
2006-08-13 23:58:04 +00:00
}
}
2005-08-27 23:47:17 +00:00
break ;
2005-08-25 15:25:22 +00:00
}
}
2005-08-27 15:13:15 +00:00
2006-08-13 23:58:04 +00:00
/* We didn't find it, so return the wildcard record if one was configured */
return wildcard ;
}
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Mapping attributes
* = = = = = = = = = = = = = = = = = = */
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Check whether an attribute will be mapped into the remote partition. */
BOOL map_attr_check_remote ( const struct ldb_map_context * data , const char * attr )
{
const struct ldb_map_attribute * map = map_attr_find_local ( data , attr ) ;
2005-08-27 18:30:00 +00:00
2006-08-13 23:58:04 +00:00
if ( map = = NULL ) {
return False ;
}
if ( map - > type = = MAP_IGNORE ) {
return False ;
2005-08-25 15:25:22 +00:00
}
2006-08-13 23:58:04 +00:00
return True ;
2005-08-25 15:25:22 +00:00
}
2006-08-13 23:58:04 +00:00
/* Map an attribute name into the remote partition. */
const char * map_attr_map_local ( void * mem_ctx , const struct ldb_map_attribute * map , const char * attr )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
if ( map = = NULL ) {
return talloc_strdup ( mem_ctx , attr ) ;
}
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
switch ( map - > type ) {
case MAP_KEEP :
return talloc_strdup ( mem_ctx , attr ) ;
2005-08-27 15:13:15 +00:00
2006-08-13 23:58:04 +00:00
case MAP_RENAME :
case MAP_CONVERT :
return talloc_strdup ( mem_ctx , map - > u . rename . remote_name ) ;
default :
return NULL ;
2005-08-25 15:25:22 +00:00
}
2006-08-13 23:58:04 +00:00
}
/* Map an attribute name back into the local partition. */
const char * map_attr_map_remote ( void * mem_ctx , const struct ldb_map_attribute * map , const char * attr )
{
if ( map = = NULL ) {
return talloc_strdup ( mem_ctx , attr ) ;
2005-08-27 23:47:17 +00:00
}
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
if ( map - > type = = MAP_KEEP ) {
return talloc_strdup ( mem_ctx , attr ) ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
return talloc_strdup ( mem_ctx , map - > local_name ) ;
2005-08-24 22:06:26 +00:00
}
2006-09-14 03:15:30 +00:00
/* Merge two lists of attributes into a single one. */
2006-11-16 09:16:17 +00:00
int map_attrs_merge ( struct ldb_module * module , void * mem_ctx ,
const char * * * attrs , const char * const * more_attrs )
2006-09-14 03:15:30 +00:00
{
int i , j , k ;
for ( i = 0 ; * attrs & & ( * attrs ) [ i ] ; i + + ) /* noop */ ;
for ( j = 0 ; more_attrs & & more_attrs [ j ] ; j + + ) /* noop */ ;
2006-11-16 09:16:17 +00:00
2006-09-14 03:15:30 +00:00
* attrs = talloc_realloc ( mem_ctx , * attrs , const char * , i + j + 1 ) ;
if ( * attrs = = NULL ) {
map_oom ( module ) ;
return - 1 ;
}
for ( k = 0 ; k < j ; k + + ) {
2006-11-16 09:16:17 +00:00
( * attrs ) [ i + k ] = more_attrs [ k ] ;
2006-09-14 03:15:30 +00:00
}
( * attrs ) [ i + k ] = NULL ;
return 0 ;
}
2006-08-13 23:58:04 +00:00
/* Mapping ldb values
* = = = = = = = = = = = = = = = = = = */
2005-08-27 16:33:42 +00:00
2006-08-13 23:58:04 +00:00
/* Map an ldb value into the remote partition. */
2006-11-16 09:16:17 +00:00
struct ldb_val ldb_val_map_local ( struct ldb_module * module , void * mem_ctx ,
const struct ldb_map_attribute * map , const struct ldb_val * val )
2006-08-13 23:58:04 +00:00
{
if ( map & & ( map - > type = = MAP_CONVERT ) & & ( map - > u . convert . convert_local ) ) {
2006-11-01 23:31:26 +00:00
return map - > u . convert . convert_local ( module , mem_ctx , val ) ;
2006-08-13 23:58:04 +00:00
}
2005-08-27 16:33:42 +00:00
2006-11-01 23:31:26 +00:00
return ldb_val_dup ( mem_ctx , val ) ;
2006-08-13 23:58:04 +00:00
}
2005-08-27 16:33:42 +00:00
2006-08-13 23:58:04 +00:00
/* Map an ldb value back into the local partition. */
2006-11-16 09:16:17 +00:00
struct ldb_val ldb_val_map_remote ( struct ldb_module * module , void * mem_ctx ,
const struct ldb_map_attribute * map , const struct ldb_val * val )
2006-08-13 23:58:04 +00:00
{
if ( map & & ( map - > type = = MAP_CONVERT ) & & ( map - > u . convert . convert_remote ) ) {
2006-11-01 23:31:26 +00:00
return map - > u . convert . convert_remote ( module , mem_ctx , val ) ;
2005-08-27 16:33:42 +00:00
}
2006-11-01 23:31:26 +00:00
return ldb_val_dup ( mem_ctx , val ) ;
2005-08-27 16:33:42 +00:00
}
2006-08-13 23:58:04 +00:00
/* Mapping DNs
* = = = = = = = = = = = */
/* Check whether a DN is below the local baseDN. */
2006-11-22 00:59:34 +00:00
BOOL ldb_dn_check_local ( struct ldb_module * module , struct ldb_dn * dn )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
const struct ldb_map_context * data = map_get_context ( module ) ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
if ( ! data - > local_base_dn ) {
return True ;
}
2006-11-22 00:59:34 +00:00
return ldb_dn_compare_base ( data - > local_base_dn , dn ) = = 0 ;
2006-08-13 23:58:04 +00:00
}
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Map a DN into the remote partition. */
2006-11-22 00:59:34 +00:00
struct ldb_dn * ldb_dn_map_local ( struct ldb_module * module , void * mem_ctx , struct ldb_dn * dn )
2006-08-13 23:58:04 +00:00
{
const struct ldb_map_context * data = map_get_context ( module ) ;
struct ldb_dn * newdn ;
const struct ldb_map_attribute * map ;
enum ldb_map_attr_type map_type ;
2006-11-01 23:31:26 +00:00
const char * name ;
struct ldb_val value ;
int i , ret ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
if ( dn = = NULL ) {
return NULL ;
}
2005-08-27 16:33:42 +00:00
2006-08-13 23:58:04 +00:00
newdn = ldb_dn_copy ( mem_ctx , dn ) ;
if ( newdn = = NULL ) {
map_oom ( module ) ;
return NULL ;
2005-08-27 16:33:42 +00:00
}
2006-08-13 23:58:04 +00:00
/* For each RDN, map the component name and possibly the value */
2006-11-01 23:31:26 +00:00
for ( i = 0 ; i < ldb_dn_get_comp_num ( newdn ) ; i + + ) {
map = map_attr_find_local ( data , ldb_dn_get_component_name ( dn , i ) ) ;
2005-08-25 15:25:22 +00:00
2006-08-13 23:58:04 +00:00
/* Unknown attribute - leave this RDN as is and hope the best... */
if ( map = = NULL ) {
map_type = MAP_KEEP ;
} else {
map_type = map - > type ;
}
2005-08-25 15:25:22 +00:00
2005-08-27 23:47:17 +00:00
switch ( map_type ) {
2006-08-13 23:58:04 +00:00
case MAP_IGNORE :
case MAP_GENERATE :
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" MAP_IGNORE/MAP_GENERATE attribute '%s' "
2006-11-01 23:31:26 +00:00
" used in DN! \n " , ldb_dn_get_component_name ( dn , i ) ) ;
2006-08-13 23:58:04 +00:00
goto failed ;
case MAP_CONVERT :
if ( map - > u . convert . convert_local = = NULL ) {
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" 'convert_local' not set for attribute '%s' "
2006-11-01 23:31:26 +00:00
" used in DN! \n " , ldb_dn_get_component_name ( dn , i ) ) ;
2006-08-13 23:58:04 +00:00
goto failed ;
}
/* fall through */
case MAP_KEEP :
case MAP_RENAME :
2006-11-01 23:31:26 +00:00
name = map_attr_map_local ( newdn , map , ldb_dn_get_component_name ( dn , i ) ) ;
if ( name = = NULL ) goto failed ;
value = ldb_val_map_local ( module , newdn , map , ldb_dn_get_component_val ( dn , i ) ) ;
if ( value . data = = NULL ) goto failed ;
ret = ldb_dn_set_component ( newdn , i , name , value ) ;
if ( ret ! = LDB_SUCCESS ) {
goto failed ;
}
2006-08-13 23:58:04 +00:00
break ;
2005-08-25 15:25:22 +00:00
}
}
2006-08-13 23:58:04 +00:00
return newdn ;
2005-08-27 16:33:42 +00:00
2006-08-13 23:58:04 +00:00
failed :
talloc_free ( newdn ) ;
return NULL ;
2005-08-24 22:06:26 +00:00
}
2006-08-13 23:58:04 +00:00
/* Map a DN into the local partition. */
2006-11-22 00:59:34 +00:00
struct ldb_dn * ldb_dn_map_remote ( struct ldb_module * module , void * mem_ctx , struct ldb_dn * dn )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
const struct ldb_map_context * data = map_get_context ( module ) ;
struct ldb_dn * newdn ;
const struct ldb_map_attribute * map ;
enum ldb_map_attr_type map_type ;
2006-11-01 23:31:26 +00:00
const char * name ;
struct ldb_val value ;
int i , ret ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
if ( dn = = NULL ) {
return NULL ;
}
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
newdn = ldb_dn_copy ( mem_ctx , dn ) ;
if ( newdn = = NULL ) {
map_oom ( module ) ;
return NULL ;
2005-09-01 01:11:15 +00:00
}
2006-08-13 23:58:04 +00:00
/* For each RDN, map the component name and possibly the value */
2006-11-01 23:31:26 +00:00
for ( i = 0 ; i < ldb_dn_get_comp_num ( newdn ) ; i + + ) {
map = map_attr_find_remote ( data , ldb_dn_get_component_name ( dn , i ) ) ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Unknown attribute - leave this RDN as is and hope the best... */
if ( map = = NULL ) {
map_type = MAP_KEEP ;
} else {
map_type = map - > type ;
}
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
switch ( map_type ) {
case MAP_IGNORE :
case MAP_GENERATE :
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" MAP_IGNORE/MAP_GENERATE attribute '%s' "
2006-11-01 23:31:26 +00:00
" used in DN! \n " , ldb_dn_get_component_name ( dn , i ) ) ;
2006-08-13 23:58:04 +00:00
goto failed ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
case MAP_CONVERT :
if ( map - > u . convert . convert_remote = = NULL ) {
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" 'convert_remote' not set for attribute '%s' "
2006-11-01 23:31:26 +00:00
" used in DN! \n " , ldb_dn_get_component_name ( dn , i ) ) ;
2006-08-13 23:58:04 +00:00
goto failed ;
}
/* fall through */
case MAP_KEEP :
case MAP_RENAME :
2006-11-01 23:31:26 +00:00
name = map_attr_map_remote ( newdn , map , ldb_dn_get_component_name ( dn , i ) ) ;
if ( name = = NULL ) goto failed ;
value = ldb_val_map_remote ( module , newdn , map , ldb_dn_get_component_val ( dn , i ) ) ;
if ( value . data = = NULL ) goto failed ;
ret = ldb_dn_set_component ( newdn , i , name , value ) ;
if ( ret ! = LDB_SUCCESS ) {
goto failed ;
}
2006-08-13 23:58:04 +00:00
break ;
2005-09-01 18:04:23 +00:00
}
2005-09-01 01:11:15 +00:00
}
2006-08-13 23:58:04 +00:00
return newdn ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
failed :
2005-08-24 22:06:26 +00:00
talloc_free ( newdn ) ;
2006-08-13 23:58:04 +00:00
return NULL ;
2005-08-24 22:06:26 +00:00
}
2006-08-13 23:58:04 +00:00
/* Map a DN and its base into the local partition. */
/* TODO: This should not be required with GUIDs. */
2006-11-22 00:59:34 +00:00
struct ldb_dn * ldb_dn_map_rebase_remote ( struct ldb_module * module , void * mem_ctx , struct ldb_dn * dn )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
const struct ldb_map_context * data = map_get_context ( module ) ;
struct ldb_dn * dn1 , * dn2 ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
dn1 = ldb_dn_rebase_local ( mem_ctx , data , dn ) ;
dn2 = ldb_dn_map_remote ( module , mem_ctx , dn1 ) ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
talloc_free ( dn1 ) ;
return dn2 ;
}
2005-08-29 15:54:10 +00:00
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
/* Converting DNs and objectClasses (as ldb values)
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Map a DN contained in an ldb value into the remote partition. */
static struct ldb_val ldb_dn_convert_local ( struct ldb_module * module , void * mem_ctx , const struct ldb_val * val )
{
struct ldb_dn * dn , * newdn ;
struct ldb_val newval ;
2005-08-29 15:54:10 +00:00
2006-11-22 00:59:34 +00:00
dn = ldb_dn_new ( mem_ctx , module - > ldb , ( char * ) val - > data ) ;
if ( ! ldb_dn_validate ( dn ) ) {
newval . length = 0 ;
newval . data = NULL ;
talloc_free ( dn ) ;
return newval ;
}
2006-08-13 23:58:04 +00:00
newdn = ldb_dn_map_local ( module , mem_ctx , dn ) ;
talloc_free ( dn ) ;
newval . length = 0 ;
2006-11-22 02:05:19 +00:00
newval . data = ( uint8_t * ) ldb_dn_alloc_linearized ( mem_ctx , newdn ) ;
2006-08-13 23:58:04 +00:00
if ( newval . data ) {
newval . length = strlen ( ( char * ) newval . data ) ;
}
talloc_free ( newdn ) ;
return newval ;
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
/* Map a DN contained in an ldb value into the local partition. */
static struct ldb_val ldb_dn_convert_remote ( struct ldb_module * module , void * mem_ctx , const struct ldb_val * val )
2005-08-29 15:54:10 +00:00
{
2006-08-13 23:58:04 +00:00
struct ldb_dn * dn , * newdn ;
struct ldb_val newval ;
2005-08-24 22:06:26 +00:00
2006-11-22 00:59:34 +00:00
dn = ldb_dn_new ( mem_ctx , module - > ldb , ( char * ) val - > data ) ;
if ( ! ldb_dn_validate ( dn ) ) {
newval . length = 0 ;
newval . data = NULL ;
talloc_free ( dn ) ;
return newval ;
}
2006-08-13 23:58:04 +00:00
newdn = ldb_dn_map_remote ( module , mem_ctx , dn ) ;
talloc_free ( dn ) ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
newval . length = 0 ;
2006-11-22 02:05:19 +00:00
newval . data = ( uint8_t * ) ldb_dn_alloc_linearized ( mem_ctx , newdn ) ;
2006-08-13 23:58:04 +00:00
if ( newval . data ) {
newval . length = strlen ( ( char * ) newval . data ) ;
}
talloc_free ( newdn ) ;
2005-08-29 17:31:19 +00:00
2006-08-13 23:58:04 +00:00
return newval ;
2005-08-24 22:06:26 +00:00
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Map an objectClass into the remote partition. */
static struct ldb_val map_objectclass_convert_local ( struct ldb_module * module , void * mem_ctx , const struct ldb_val * val )
2005-08-29 15:54:10 +00:00
{
2006-08-13 23:58:04 +00:00
const struct ldb_map_context * data = map_get_context ( module ) ;
const char * name = ( char * ) val - > data ;
const struct ldb_map_objectclass * map = map_objectclass_find_local ( data , name ) ;
struct ldb_val newval ;
if ( map ) {
newval . data = ( uint8_t * ) talloc_strdup ( mem_ctx , map - > remote_name ) ;
newval . length = strlen ( ( char * ) newval . data ) ;
return newval ;
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
return ldb_val_dup ( mem_ctx , val ) ;
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Generate a remote message with a mapped objectClass. */
static void map_objectclass_generate_remote ( struct ldb_module * module , const char * local_attr , const struct ldb_message * old , struct ldb_message * remote , struct ldb_message * local )
{
struct ldb_message_element * el , * oc ;
struct ldb_val val ;
BOOL found_extensibleObject = False ;
int i ;
/* Find old local objectClass */
2006-11-16 09:16:17 +00:00
oc = ldb_msg_find_element ( old , " objectClass " ) ;
2006-08-13 23:58:04 +00:00
if ( oc = = NULL ) {
return ;
2005-08-31 21:04:17 +00:00
}
2006-08-13 23:58:04 +00:00
/* Prepare new element */
el = talloc_zero ( remote , struct ldb_message_element ) ;
if ( el = = NULL ) {
ldb_oom ( module - > ldb ) ;
return ; /* TODO: fail? */
}
2005-11-29 12:34:03 +00:00
2006-08-13 23:58:04 +00:00
/* Copy local objectClass element, reverse space for an extra value */
el - > num_values = oc - > num_values + 1 ;
el - > values = talloc_array ( el , struct ldb_val , el - > num_values ) ;
if ( el - > values = = NULL ) {
talloc_free ( el ) ;
ldb_oom ( module - > ldb ) ;
return ; /* TODO: fail? */
2005-08-30 00:41:02 +00:00
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Copy local element name "objectClass" */
el - > name = talloc_strdup ( el , local_attr ) ;
2005-11-29 12:34:03 +00:00
2006-08-13 23:58:04 +00:00
/* Convert all local objectClasses */
for ( i = 0 ; i < el - > num_values - 1 ; i + + ) {
el - > values [ i ] = map_objectclass_convert_local ( module , el - > values , & oc - > values [ i ] ) ;
if ( ldb_attr_cmp ( ( char * ) el - > values [ i ] . data , " extensibleObject " ) = = 0 ) {
found_extensibleObject = True ;
}
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
if ( ! found_extensibleObject ) {
val . data = ( uint8_t * ) talloc_strdup ( el - > values , " extensibleObject " ) ;
val . length = strlen ( ( char * ) val . data ) ;
2005-08-31 21:04:17 +00:00
2006-08-13 23:58:04 +00:00
/* Append additional objectClass "extensibleObject" */
el - > values [ i ] = val ;
} else {
el - > num_values - - ;
2005-11-29 12:34:03 +00:00
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Add new objectClass to remote message */
ldb_msg_add ( remote , el , 0 ) ;
2005-08-29 15:54:10 +00:00
}
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Map an objectClass into the local partition. */
static struct ldb_val map_objectclass_convert_remote ( struct ldb_module * module , void * mem_ctx , const struct ldb_val * val )
2005-09-01 15:33:31 +00:00
{
2006-08-13 23:58:04 +00:00
const struct ldb_map_context * data = map_get_context ( module ) ;
const char * name = ( char * ) val - > data ;
const struct ldb_map_objectclass * map = map_objectclass_find_remote ( data , name ) ;
struct ldb_val newval ;
if ( map ) {
newval . data = ( uint8_t * ) talloc_strdup ( mem_ctx , map - > local_name ) ;
newval . length = strlen ( ( char * ) newval . data ) ;
return newval ;
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
return ldb_val_dup ( mem_ctx , val ) ;
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
/* Generate a local message with a mapped objectClass. */
2006-11-16 09:16:17 +00:00
static struct ldb_message_element * map_objectclass_generate_local ( struct ldb_module * module , void * mem_ctx , const char * local_attr , const struct ldb_message * remote )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
struct ldb_message_element * el , * oc ;
struct ldb_val val ;
2005-09-01 15:33:31 +00:00
int i ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Find old remote objectClass */
2006-11-16 09:16:17 +00:00
oc = ldb_msg_find_element ( remote , " objectClass " ) ;
2006-08-13 23:58:04 +00:00
if ( oc = = NULL ) {
return NULL ;
}
2005-08-31 21:04:17 +00:00
2006-08-13 23:58:04 +00:00
/* Prepare new element */
el = talloc_zero ( mem_ctx , struct ldb_message_element ) ;
if ( el = = NULL ) {
ldb_oom ( module - > ldb ) ;
return NULL ;
}
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* Copy remote objectClass element */
el - > num_values = oc - > num_values ;
el - > values = talloc_array ( el , struct ldb_val , el - > num_values ) ;
if ( el - > values = = NULL ) {
talloc_free ( el ) ;
ldb_oom ( module - > ldb ) ;
return NULL ;
}
/* Copy remote element name "objectClass" */
2006-11-16 09:16:17 +00:00
el - > name = talloc_strdup ( el , local_attr ) ;
2006-08-13 23:58:04 +00:00
/* Convert all remote objectClasses */
for ( i = 0 ; i < el - > num_values ; i + + ) {
el - > values [ i ] = map_objectclass_convert_remote ( module , el - > values , & oc - > values [ i ] ) ;
}
val . data = ( uint8_t * ) talloc_strdup ( el - > values , " extensibleObject " ) ;
val . length = strlen ( ( char * ) val . data ) ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* Remove last value if it was "extensibleObject" */
if ( ldb_val_equal_exact ( & val , & el - > values [ i - 1 ] ) ) {
el - > num_values - - ;
el - > values = talloc_realloc ( el , el - > values , struct ldb_val , el - > num_values ) ;
if ( el - > values = = NULL ) {
talloc_free ( el ) ;
ldb_oom ( module - > ldb ) ;
return NULL ;
2005-09-01 15:33:31 +00:00
}
}
2006-08-13 23:58:04 +00:00
return el ;
}
2005-09-01 15:33:31 +00:00
2006-08-14 23:25:04 +00:00
/* Mappings for searches on objectClass= assuming a one-to-one
* mapping . Needed because this is a generate operator for the
* add / modify code */
static int map_objectclass_convert_operator ( struct ldb_module * module , void * mem_ctx ,
struct ldb_parse_tree * * new , const struct ldb_parse_tree * tree )
{
static const struct ldb_map_attribute objectclass_map = {
2006-08-17 20:06:43 +00:00
. local_name = " objectClass " ,
2006-08-14 23:25:04 +00:00
. type = MAP_CONVERT ,
. u = {
. convert = {
2006-08-17 20:06:43 +00:00
. remote_name = " objectClass " ,
2006-08-14 23:25:04 +00:00
. convert_local = map_objectclass_convert_local ,
. convert_remote = map_objectclass_convert_remote ,
} ,
} ,
} ;
2005-09-01 15:33:31 +00:00
2006-08-14 23:25:04 +00:00
return map_subtree_collect_remote_simple ( module , mem_ctx , new , tree , & objectclass_map ) ;
}
2005-09-01 20:28:03 +00:00
2006-08-13 23:58:04 +00:00
/* Auxiliary request construction
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
2005-09-01 20:28:03 +00:00
2006-08-13 23:58:04 +00:00
/* Store the DN of a single search result in context. */
static int map_search_self_callback ( struct ldb_context * ldb , void * context , struct ldb_reply * ares )
{
struct map_context * ac ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
if ( context = = NULL | | ares = = NULL ) {
ldb_set_errstring ( ldb , talloc_asprintf ( ldb , " NULL Context or Result in callback " ) ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
ac = talloc_get_type ( context , struct map_context ) ;
2005-09-01 20:28:03 +00:00
2006-08-13 23:58:04 +00:00
/* We are interested only in the single reply */
if ( ares - > type ! = LDB_REPLY_ENTRY ) {
talloc_free ( ares ) ;
return LDB_SUCCESS ;
}
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* We have already found a remote DN */
if ( ac - > local_dn ) {
ldb_set_errstring ( ldb , talloc_asprintf ( ldb , " Too many results to base search " ) ) ;
talloc_free ( ares ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* Store local DN */
ac - > local_dn = ares - > message - > dn ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
return LDB_SUCCESS ;
}
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
/* Build a request to search a record by its DN. */
2006-11-22 00:59:34 +00:00
struct ldb_request * map_search_base_req ( struct map_context * ac , struct ldb_dn * dn , const char * const * attrs , const struct ldb_parse_tree * tree , void * context , ldb_search_callback callback )
2006-08-13 23:58:04 +00:00
{
struct ldb_request * req ;
req = talloc_zero ( ac , struct ldb_request ) ;
if ( req = = NULL ) {
map_oom ( ac - > module ) ;
return NULL ;
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
req - > operation = LDB_SEARCH ;
req - > op . search . base = dn ;
req - > op . search . scope = LDB_SCOPE_BASE ;
req - > op . search . attrs = attrs ;
if ( tree ) {
req - > op . search . tree = tree ;
} else {
req - > op . search . tree = ldb_parse_tree ( req , NULL ) ;
if ( req - > op . search . tree = = NULL ) {
talloc_free ( req ) ;
return NULL ;
}
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
req - > controls = NULL ;
req - > context = context ;
req - > callback = callback ;
ldb_set_timeout_from_prev_req ( ac - > module - > ldb , ac - > orig_req , req ) ;
2005-09-01 15:33:31 +00:00
2006-08-13 23:58:04 +00:00
return req ;
}
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
/* Build a request to search the local record by its DN. */
2006-11-22 00:59:34 +00:00
struct ldb_request * map_search_self_req ( struct map_context * ac , struct ldb_dn * dn )
2006-08-13 23:58:04 +00:00
{
/* attrs[] is returned from this function in
* ac - > search_req - > op . search . attrs , so it must be static , as
* otherwise the compiler can put it on the stack */
static const char * const attrs [ ] = { IS_MAPPED , NULL } ;
struct ldb_parse_tree * tree ;
/* Limit search to records with 'IS_MAPPED' present */
/* TODO: `tree = ldb_parse_tree(ac, IS_MAPPED);' won't do. */
tree = talloc_zero ( ac , struct ldb_parse_tree ) ;
if ( tree = = NULL ) {
map_oom ( ac - > module ) ;
return NULL ;
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
tree - > operation = LDB_OP_PRESENT ;
tree - > u . present . attr = talloc_strdup ( tree , IS_MAPPED ) ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
return map_search_base_req ( ac , dn , attrs , tree , ac , map_search_self_callback ) ;
2005-08-24 22:06:26 +00:00
}
2006-08-13 23:58:04 +00:00
/* Build a request to update the 'IS_MAPPED' attribute */
2006-11-22 00:59:34 +00:00
struct ldb_request * map_build_fixup_req ( struct map_context * ac , struct ldb_dn * olddn , struct ldb_dn * newdn )
2005-08-24 22:06:26 +00:00
{
2006-08-13 23:58:04 +00:00
struct ldb_request * req ;
struct ldb_message * msg ;
const char * dn ;
/* Prepare request */
req = talloc_zero ( ac , struct ldb_request ) ;
if ( req = = NULL ) {
map_oom ( ac - > module ) ;
return NULL ;
}
2005-09-01 18:55:51 +00:00
2006-08-13 23:58:04 +00:00
/* Prepare message */
msg = ldb_msg_new ( req ) ;
if ( msg = = NULL ) {
map_oom ( ac - > module ) ;
goto failed ;
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Update local 'IS_MAPPED' to the new remote DN */
2006-11-22 02:05:19 +00:00
msg - > dn = ldb_dn_copy ( msg , olddn ) ;
dn = ldb_dn_alloc_linearized ( msg , newdn ) ;
if ( ! dn | | ! ldb_dn_validate ( msg - > dn ) ) {
2006-08-13 23:58:04 +00:00
goto failed ;
}
2006-10-25 01:42:59 +00:00
if ( ldb_msg_add_empty ( msg , IS_MAPPED , LDB_FLAG_MOD_REPLACE , NULL ) ! = 0 ) {
2006-08-13 23:58:04 +00:00
goto failed ;
}
if ( ldb_msg_add_string ( msg , IS_MAPPED , dn ) ! = 0 ) {
goto failed ;
2005-09-01 15:33:31 +00:00
}
2006-08-13 23:58:04 +00:00
req - > operation = LDB_MODIFY ;
req - > op . mod . message = msg ;
req - > controls = NULL ;
req - > handle = NULL ;
req - > context = NULL ;
req - > callback = NULL ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
return req ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
failed :
talloc_free ( req ) ;
return NULL ;
2005-08-24 22:06:26 +00:00
}
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
/* Asynchronous call structure
* = = = = = = = = = = = = = = = = = = = = = = = = = = = */
/* Figure out which request is currently pending. */
static struct ldb_request * map_get_req ( struct map_context * ac )
2005-11-08 00:11:45 +00:00
{
2006-08-13 23:58:04 +00:00
switch ( ac - > step ) {
case MAP_SEARCH_SELF_MODIFY :
case MAP_SEARCH_SELF_DELETE :
case MAP_SEARCH_SELF_RENAME :
return ac - > search_req ;
case MAP_ADD_REMOTE :
case MAP_MODIFY_REMOTE :
case MAP_DELETE_REMOTE :
case MAP_RENAME_REMOTE :
return ac - > remote_req ;
case MAP_RENAME_FIXUP :
return ac - > down_req ;
case MAP_ADD_LOCAL :
case MAP_MODIFY_LOCAL :
case MAP_DELETE_LOCAL :
case MAP_RENAME_LOCAL :
return ac - > local_req ;
case MAP_SEARCH_REMOTE :
/* Can't happen */
break ;
}
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
return NULL ; /* unreachable; silences a warning */
}
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
typedef int ( * map_next_function ) ( struct ldb_handle * handle ) ;
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
/* Figure out the next request to run. */
static map_next_function map_get_next ( struct map_context * ac )
{
switch ( ac - > step ) {
case MAP_SEARCH_REMOTE :
return NULL ;
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
case MAP_ADD_LOCAL :
return map_add_do_remote ;
case MAP_ADD_REMOTE :
return NULL ;
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
case MAP_SEARCH_SELF_MODIFY :
return map_modify_do_local ;
case MAP_MODIFY_LOCAL :
return map_modify_do_remote ;
case MAP_MODIFY_REMOTE :
return NULL ;
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
case MAP_SEARCH_SELF_DELETE :
return map_delete_do_local ;
case MAP_DELETE_LOCAL :
return map_delete_do_remote ;
case MAP_DELETE_REMOTE :
return NULL ;
2005-11-08 00:11:45 +00:00
2006-08-13 23:58:04 +00:00
case MAP_SEARCH_SELF_RENAME :
return map_rename_do_local ;
case MAP_RENAME_LOCAL :
return map_rename_do_fixup ;
case MAP_RENAME_FIXUP :
return map_rename_do_remote ;
case MAP_RENAME_REMOTE :
return NULL ;
2005-11-08 00:11:45 +00:00
}
2006-08-13 23:58:04 +00:00
return NULL ; /* unreachable; silences a warning */
}
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Wait for the current pending request to finish and continue with the next. */
static int map_wait_next ( struct ldb_handle * handle )
2005-08-29 15:54:10 +00:00
{
2006-08-13 23:58:04 +00:00
struct map_context * ac ;
struct ldb_request * req ;
map_next_function next ;
2005-08-29 15:54:10 +00:00
int ret ;
2006-08-13 23:58:04 +00:00
if ( handle = = NULL | | handle - > private_data = = NULL ) {
return LDB_ERR_OPERATIONS_ERROR ;
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
if ( handle - > state = = LDB_ASYNC_DONE ) {
return handle - > status ;
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
handle - > state = LDB_ASYNC_PENDING ;
handle - > status = LDB_SUCCESS ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
ac = talloc_get_type ( handle - > private_data , struct map_context ) ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
if ( ac - > step = = MAP_SEARCH_REMOTE ) {
int i ;
for ( i = 0 ; i < ac - > num_searches ; i + + ) {
req = ac - > search_reqs [ i ] ;
ret = ldb_wait ( req - > handle , LDB_WAIT_NONE ) ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
if ( ret ! = LDB_SUCCESS ) {
handle - > status = ret ;
goto done ;
}
if ( req - > handle - > status ! = LDB_SUCCESS ) {
handle - > status = req - > handle - > status ;
goto done ;
}
if ( req - > handle - > state ! = LDB_ASYNC_DONE ) {
return LDB_SUCCESS ;
}
}
} else {
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
req = map_get_req ( ac ) ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
ret = ldb_wait ( req - > handle , LDB_WAIT_NONE ) ;
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
if ( ret ! = LDB_SUCCESS ) {
handle - > status = ret ;
goto done ;
}
if ( req - > handle - > status ! = LDB_SUCCESS ) {
handle - > status = req - > handle - > status ;
goto done ;
}
if ( req - > handle - > state ! = LDB_ASYNC_DONE ) {
return LDB_SUCCESS ;
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
next = map_get_next ( ac ) ;
if ( next ) {
return next ( handle ) ;
}
2005-08-29 15:54:10 +00:00
}
2006-08-13 23:58:04 +00:00
ret = LDB_SUCCESS ;
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
done :
handle - > state = LDB_ASYNC_DONE ;
return ret ;
}
2005-08-29 15:54:10 +00:00
2006-08-13 23:58:04 +00:00
/* Wait for all current pending requests to finish. */
static int map_wait_all ( struct ldb_handle * handle )
{
int ret ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
while ( handle - > state ! = LDB_ASYNC_DONE ) {
ret = map_wait_next ( handle ) ;
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
return handle - > status ;
}
/* Wait for pending requests to finish. */
static int map_wait ( struct ldb_handle * handle , enum ldb_wait_type type )
{
if ( type = = LDB_WAIT_ALL ) {
return map_wait_all ( handle ) ;
} else {
return map_wait_next ( handle ) ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Module initialization
* = = = = = = = = = = = = = = = = = = = = = */
2005-08-24 22:06:26 +00:00
2006-08-13 23:58:04 +00:00
/* Provided module operations */
static const struct ldb_module_ops map_ops = {
. name = " ldb_map " ,
. add = map_add ,
. modify = map_modify ,
. del = map_delete ,
. rename = map_rename ,
. search = map_search ,
. wait = map_wait ,
} ;
/* Builtin mappings for DNs and objectClasses */
static const struct ldb_map_attribute builtin_attribute_maps [ ] = {
{
. local_name = " dn " ,
. type = MAP_CONVERT ,
. u = {
. convert = {
. remote_name = " dn " ,
. convert_local = ldb_dn_convert_local ,
. convert_remote = ldb_dn_convert_remote ,
} ,
} ,
} ,
{
2006-08-17 20:06:43 +00:00
. local_name = " objectClass " ,
2006-08-13 23:58:04 +00:00
. type = MAP_GENERATE ,
2006-08-14 23:25:04 +00:00
. convert_operator = map_objectclass_convert_operator ,
2006-08-13 23:58:04 +00:00
. u = {
. generate = {
2006-08-17 20:06:43 +00:00
. remote_names = { " objectClass " , NULL } ,
2006-08-13 23:58:04 +00:00
. generate_local = map_objectclass_generate_local ,
. generate_remote = map_objectclass_generate_remote ,
} ,
} ,
} ,
{
. local_name = NULL ,
}
} ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Find the special 'MAP_DN_NAME' record and store local and remote
* base DNs in private data . */
static int map_init_dns ( struct ldb_module * module , struct ldb_map_context * data , const char * name )
2005-08-27 23:47:17 +00:00
{
2006-08-13 23:58:04 +00:00
static const char * const attrs [ ] = { MAP_DN_FROM , MAP_DN_TO , NULL } ;
struct ldb_dn * dn ;
struct ldb_message * msg ;
struct ldb_result * res ;
int ret ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
if ( ! name ) {
data - > local_base_dn = NULL ;
data - > remote_base_dn = NULL ;
return LDB_SUCCESS ;
}
2005-08-27 23:47:17 +00:00
2006-11-22 00:59:34 +00:00
dn = ldb_dn_new_fmt ( data , module - > ldb , " %s=%s " , MAP_DN_NAME , name ) ;
if ( ! ldb_dn_validate ( dn ) ) {
2006-08-13 23:58:04 +00:00
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" Failed to construct '%s' DN! \n " , MAP_DN_NAME ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
ret = ldb_search ( module - > ldb , dn , LDB_SCOPE_BASE , NULL , attrs , & res ) ;
2005-08-27 23:47:17 +00:00
talloc_free ( dn ) ;
2006-08-13 23:58:04 +00:00
if ( ret ! = LDB_SUCCESS ) {
return ret ;
}
if ( res - > count = = 0 ) {
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" No results for '%s=%s'! \n " , MAP_DN_NAME , name ) ;
2006-10-15 23:14:19 +00:00
talloc_free ( res ) ;
2006-08-13 23:58:04 +00:00
return LDB_ERR_CONSTRAINT_VIOLATION ;
}
if ( res - > count > 1 ) {
ldb_debug ( module - > ldb , LDB_DEBUG_ERROR , " ldb_map: "
" Too many results for '%s=%s'! \n " , MAP_DN_NAME , name ) ;
2006-10-15 23:14:19 +00:00
talloc_free ( res ) ;
2006-08-13 23:58:04 +00:00
return LDB_ERR_CONSTRAINT_VIOLATION ;
2005-09-01 18:55:51 +00:00
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
msg = res - > msgs [ 0 ] ;
2006-11-22 00:59:34 +00:00
data - > local_base_dn = ldb_msg_find_attr_as_dn ( module - > ldb , data , msg , MAP_DN_FROM ) ;
data - > remote_base_dn = ldb_msg_find_attr_as_dn ( module - > ldb , data , msg , MAP_DN_TO ) ;
2006-08-13 23:58:04 +00:00
talloc_free ( res ) ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
return LDB_SUCCESS ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
/* Store attribute maps and objectClass maps in private data. */
2006-09-14 03:15:30 +00:00
static int map_init_maps ( struct ldb_module * module , struct ldb_map_context * data ,
const struct ldb_map_attribute * attrs ,
const struct ldb_map_objectclass * ocls ,
const char * const * wildcard_attributes )
2005-08-27 23:47:17 +00:00
{
2006-08-13 23:58:04 +00:00
int i , j , last ;
last = 0 ;
/* Count specified attribute maps */
for ( i = 0 ; attrs [ i ] . local_name ; i + + ) /* noop */ ;
/* Count built-in attribute maps */
for ( j = 0 ; builtin_attribute_maps [ j ] . local_name ; j + + ) /* noop */ ;
/* Store list of attribute maps */
data - > attribute_maps = talloc_array ( data , struct ldb_map_attribute , i + j + 1 ) ;
if ( data - > attribute_maps = = NULL ) {
map_oom ( module ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Specified ones go first */
for ( i = 0 ; attrs [ i ] . local_name ; i + + ) {
data - > attribute_maps [ last ] = attrs [ i ] ;
last + + ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Built-in ones go last */
for ( i = 0 ; builtin_attribute_maps [ i ] . local_name ; i + + ) {
data - > attribute_maps [ last ] = builtin_attribute_maps [ i ] ;
last + + ;
2005-09-01 20:28:03 +00:00
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Ensure 'local_name == NULL' for the last entry */
memset ( & data - > attribute_maps [ last ] , 0 , sizeof ( struct ldb_map_attribute ) ) ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Store list of objectClass maps */
data - > objectclass_maps = ocls ;
2006-09-14 03:15:30 +00:00
data - > wildcard_attributes = wildcard_attributes ;
2006-08-13 23:58:04 +00:00
return LDB_SUCCESS ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
/* Copy the list of provided module operations. */
2006-10-29 17:40:19 +00:00
_PUBLIC_ struct ldb_module_ops ldb_map_get_ops ( void )
2005-08-27 23:47:17 +00:00
{
2006-08-13 23:58:04 +00:00
return map_ops ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Initialize global private data. */
2006-10-29 17:40:19 +00:00
_PUBLIC_ int ldb_map_init ( struct ldb_module * module , const struct ldb_map_attribute * attrs ,
2006-09-14 03:15:30 +00:00
const struct ldb_map_objectclass * ocls ,
const char * const * wildcard_attributes ,
const char * name )
2006-08-13 23:58:04 +00:00
{
struct map_private * data ;
int ret ;
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Prepare private data */
data = talloc_zero ( module , struct map_private ) ;
if ( data = = NULL ) {
map_oom ( module ) ;
return LDB_ERR_OPERATIONS_ERROR ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
module - > private_data = data ;
2005-08-27 23:47:17 +00:00
2006-08-22 06:01:47 +00:00
data - > context = talloc_zero ( data , struct ldb_map_context ) ;
if ( ! data - > context ) {
map_oom ( module ) ;
return LDB_ERR_OPERATIONS_ERROR ;
}
2006-08-13 23:58:04 +00:00
/* Store local and remote baseDNs */
2006-08-22 06:01:47 +00:00
ret = map_init_dns ( module , data - > context , name ) ;
2006-08-13 23:58:04 +00:00
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( data ) ;
return ret ;
}
2005-08-27 23:47:17 +00:00
2006-08-13 23:58:04 +00:00
/* Store list of attribute and objectClass maps */
2006-09-14 03:15:30 +00:00
ret = map_init_maps ( module , data - > context , attrs , ocls , wildcard_attributes ) ;
2006-08-13 23:58:04 +00:00
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( data ) ;
return ret ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
return LDB_SUCCESS ;
2005-08-27 23:47:17 +00:00
}
2006-08-13 23:58:04 +00:00
/* Usage note for initialization of this module:
*
* ldb_map is meant to be used from a different module that sets up
* the mappings and gets registered in ldb .
*
* ' ldb_map_init ' initializes the private data of this module and
* stores the attribute and objectClass maps in there . It also looks
* up the ' @ MAP ' special DN so requests can be redirected to the
* remote partition .
*
* This function should be called from the ' init_context ' op of the
* module using ldb_map .
*
* ' ldb_map_get_ops ' returns a copy of ldb_maps module operations .
*
* It should be called from the initialize function of the using
* module , which should then override the ' init_context ' op with a
* function making the appropriate calls to ' ldb_map_init ' .
*/