2005-06-17 02:47:26 +00:00
/*
ldb database library - ildap backend
Copyright ( C ) Andrew Tridgell 2005
* * NOTE ! The following LGPL license applies to the ldb
* * library . This does NOT imply that all of Samba is released
* * under the LGPL
This library is free software ; you can redistribute it and / or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation ; either
version 2 of the License , or ( at your option ) any later version .
This library is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
Lesser General Public License for more details .
You should have received a copy of the GNU Lesser General Public
License along with this library ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
/*
This is a ldb backend for the internal ldap client library in
Samba4 . By using this backend we are independent of a system ldap
library
*/
# include "includes.h"
# include "ldb/include/ldb.h"
# include "ldb/include/ldb_private.h"
# include "libcli/ldap/ldap.h"
# include "libcli/ldap/ldap_client.h"
2005-06-18 09:01:09 +00:00
# include "lib/cmdline/popt_common.h"
2005-06-17 02:47:26 +00:00
struct ildb_private {
struct ldap_connection * ldap ;
NTSTATUS last_rc ;
2005-06-18 09:48:17 +00:00
struct ldb_message * rootDSE ;
2005-06-17 02:47:26 +00:00
} ;
/*
rename a record
*/
2005-08-18 15:02:01 +00:00
static int ildb_rename ( struct ldb_module * module , const struct ldb_dn * olddn , const struct ldb_dn * newdn )
2005-06-17 02:47:26 +00:00
{
2005-08-18 15:02:01 +00:00
TALLOC_CTX * local_ctx ;
2005-06-17 02:47:26 +00:00
struct ildb_private * ildb = module - > private_data ;
int ret = 0 ;
2005-08-18 15:02:01 +00:00
char * old_dn ;
char * newrdn , * parentdn ;
2005-06-17 02:47:26 +00:00
/* ignore ltdb specials */
2005-08-18 15:02:01 +00:00
if ( ldb_dn_is_special ( olddn ) | | ldb_dn_is_special ( newdn ) ) {
2005-06-17 02:47:26 +00:00
return 0 ;
}
2005-08-18 15:02:01 +00:00
local_ctx = talloc_named ( ildb , 0 , " ildb_rename local context " ) ;
if ( local_ctx = = NULL ) {
2005-06-17 02:47:26 +00:00
return - 1 ;
}
2005-08-18 15:02:01 +00:00
old_dn = ldb_dn_linearize ( local_ctx , olddn ) ;
if ( old_dn = = NULL ) {
goto failed ;
}
newrdn = talloc_asprintf ( local_ctx , " %s=%s " ,
newdn - > components [ 0 ] . name ,
ldb_dn_escape_value ( ildb , newdn - > components [ 0 ] . value ) ) ;
if ( newrdn = = NULL ) {
goto failed ;
}
parentdn = ldb_dn_linearize ( local_ctx , ldb_dn_get_parent ( ildb , newdn ) ) ;
if ( parentdn = = NULL ) {
goto failed ;
2005-06-17 02:47:26 +00:00
}
2005-08-18 15:02:01 +00:00
ildb - > last_rc = ildap_rename ( ildb - > ldap , old_dn , newrdn , parentdn , True ) ;
2005-06-17 02:47:26 +00:00
if ( ! NT_STATUS_IS_OK ( ildb - > last_rc ) ) {
ret = - 1 ;
}
2005-08-18 15:02:01 +00:00
talloc_free ( local_ctx ) ;
2005-06-17 02:47:26 +00:00
return ret ;
2005-08-18 15:02:01 +00:00
failed :
talloc_free ( local_ctx ) ;
return - 1 ;
2005-06-17 02:47:26 +00:00
}
/*
delete a record
*/
2005-08-18 15:02:01 +00:00
static int ildb_delete ( struct ldb_module * module , const struct ldb_dn * dn )
2005-06-17 02:47:26 +00:00
{
struct ildb_private * ildb = module - > private_data ;
2005-08-18 15:02:01 +00:00
char * del_dn ;
2005-06-17 02:47:26 +00:00
int ret = 0 ;
/* ignore ltdb specials */
2005-08-18 15:02:01 +00:00
if ( ldb_dn_is_special ( dn ) ) {
2005-06-17 02:47:26 +00:00
return 0 ;
}
2005-08-18 15:02:01 +00:00
del_dn = ldb_dn_linearize ( ildb , dn ) ;
ildb - > last_rc = ildap_delete ( ildb - > ldap , del_dn ) ;
2005-06-17 02:47:26 +00:00
if ( ! NT_STATUS_IS_OK ( ildb - > last_rc ) ) {
ret = - 1 ;
}
2005-08-18 15:02:01 +00:00
talloc_free ( del_dn ) ;
2005-06-17 02:47:26 +00:00
return ret ;
}
2005-06-18 09:48:17 +00:00
static void ildb_rootdse ( struct ldb_module * module ) ;
2005-06-17 02:47:26 +00:00
/*
search for matching records
*/
2005-08-18 15:02:01 +00:00
static int ildb_search ( struct ldb_module * module , const struct ldb_dn * base ,
2005-06-17 02:47:26 +00:00
enum ldb_scope scope , const char * expression ,
const char * const * attrs , struct ldb_message * * * res )
{
struct ildb_private * ildb = module - > private_data ;
int count , i ;
struct ldap_message * * ldapres , * msg ;
2005-08-18 15:02:01 +00:00
char * search_base ;
2005-06-17 02:47:26 +00:00
2005-06-18 09:01:09 +00:00
if ( scope = = LDB_SCOPE_DEFAULT ) {
scope = LDB_SCOPE_SUBTREE ;
}
2005-06-17 02:47:26 +00:00
if ( base = = NULL ) {
2005-06-18 09:48:17 +00:00
if ( ildb - > rootDSE = = NULL ) {
ildb_rootdse ( module ) ;
}
if ( ildb - > rootDSE ! = NULL ) {
2005-08-18 15:02:01 +00:00
search_base = talloc_strdup ( ildb ,
ldb_msg_find_string ( ildb - > rootDSE ,
" defaultNamingContext " , " " ) ) ;
2005-06-19 10:37:45 +00:00
} else {
2005-08-18 15:02:01 +00:00
search_base = talloc_strdup ( ildb , " " ) ;
2005-06-18 09:48:17 +00:00
}
2005-08-18 15:02:01 +00:00
} else {
search_base = ldb_dn_linearize ( ildb , base ) ;
}
if ( search_base = = NULL ) {
return - 1 ;
2005-06-17 02:47:26 +00:00
}
if ( expression = = NULL | | expression [ 0 ] = = ' \0 ' ) {
expression = " objectClass=* " ;
}
2005-08-18 15:02:01 +00:00
ildb - > last_rc = ildap_search ( ildb - > ldap , search_base , scope , expression , attrs ,
2005-06-17 02:47:26 +00:00
0 , & ldapres ) ;
2005-08-18 15:02:01 +00:00
talloc_free ( search_base ) ;
2005-06-17 02:47:26 +00:00
if ( ! NT_STATUS_IS_OK ( ildb - > last_rc ) ) {
return - 1 ;
}
count = ildap_count_entries ( ildb - > ldap , ldapres ) ;
if ( count = = - 1 | | count = = 0 ) {
talloc_free ( ldapres ) ;
return count ;
}
( * res ) = talloc_array ( ildb , struct ldb_message * , count + 1 ) ;
if ( ! * res ) {
talloc_free ( ldapres ) ;
return - 1 ;
}
( * res ) [ 0 ] = NULL ;
/* loop over all messages */
for ( i = 0 ; i < count ; i + + ) {
struct ldap_SearchResEntry * search ;
msg = ldapres [ i ] ;
search = & msg - > r . SearchResultEntry ;
( * res ) [ i ] = talloc ( * res , struct ldb_message ) ;
if ( ! ( * res ) [ i ] ) {
goto failed ;
}
( * res ) [ i + 1 ] = NULL ;
2005-08-18 15:02:01 +00:00
( * res ) [ i ] - > dn = ldb_dn_explode ( ( * res ) [ i ] , search - > dn ) ;
if ( ( * res ) [ i ] - > dn = = NULL ) {
goto failed ;
}
2005-06-17 02:47:26 +00:00
( * res ) [ i ] - > num_elements = search - > num_attributes ;
( * res ) [ i ] - > elements = talloc_steal ( ( * res ) [ i ] , search - > attributes ) ;
( * res ) [ i ] - > private_data = NULL ;
}
talloc_free ( ldapres ) ;
return count ;
failed :
if ( * res ) talloc_free ( * res ) ;
return - 1 ;
}
/*
search for matching records using a ldb_parse_tree
*/
2005-08-18 15:02:01 +00:00
static int ildb_search_bytree ( struct ldb_module * module , const struct ldb_dn * base ,
2005-06-17 02:47:26 +00:00
enum ldb_scope scope , struct ldb_parse_tree * tree ,
const char * const * attrs , struct ldb_message * * * res )
{
struct ildb_private * ildb = module - > private_data ;
char * expression ;
int ret ;
expression = ldb_filter_from_tree ( ildb , tree ) ;
if ( expression = = NULL ) {
return - 1 ;
}
ret = ildb_search ( module , base , scope , expression , attrs , res ) ;
talloc_free ( expression ) ;
return ret ;
}
/*
convert a ldb_message structure to a list of ldap_mod structures
ready for ildap_add ( ) or ildap_modify ( )
*/
static struct ldap_mod * * ildb_msg_to_mods ( struct ldb_context * ldb ,
const struct ldb_message * msg , int use_flags )
{
struct ldap_mod * * mods ;
unsigned int i ;
int num_mods = 0 ;
/* allocate maximum number of elements needed */
mods = talloc_array ( ldb , struct ldap_mod * , msg - > num_elements + 1 ) ;
if ( ! mods ) {
errno = ENOMEM ;
return NULL ;
}
mods [ 0 ] = NULL ;
for ( i = 0 ; i < msg - > num_elements ; i + + ) {
const struct ldb_message_element * el = & msg - > elements [ i ] ;
mods [ num_mods ] = talloc ( ldb , struct ldap_mod ) ;
if ( ! mods [ num_mods ] ) {
goto failed ;
}
mods [ num_mods + 1 ] = NULL ;
mods [ num_mods ] - > type = 0 ;
mods [ num_mods ] - > attrib = * el ;
if ( use_flags ) {
switch ( el - > flags & LDB_FLAG_MOD_MASK ) {
case LDB_FLAG_MOD_ADD :
mods [ num_mods ] - > type = LDAP_MODIFY_ADD ;
break ;
case LDB_FLAG_MOD_DELETE :
mods [ num_mods ] - > type = LDAP_MODIFY_DELETE ;
break ;
case LDB_FLAG_MOD_REPLACE :
mods [ num_mods ] - > type = LDAP_MODIFY_REPLACE ;
break ;
}
}
num_mods + + ;
}
return mods ;
failed :
talloc_free ( mods ) ;
return NULL ;
}
/*
add a record
*/
static int ildb_add ( struct ldb_module * module , const struct ldb_message * msg )
{
struct ldb_context * ldb = module - > ldb ;
struct ildb_private * ildb = module - > private_data ;
struct ldap_mod * * mods ;
2005-08-18 15:02:01 +00:00
char * dn ;
2005-06-17 02:47:26 +00:00
int ret = 0 ;
/* ignore ltdb specials */
2005-08-18 15:02:01 +00:00
if ( ldb_dn_is_special ( msg - > dn ) ) {
2005-06-17 02:47:26 +00:00
return 0 ;
}
mods = ildb_msg_to_mods ( ldb , msg , 0 ) ;
2005-08-18 15:02:01 +00:00
if ( mods = = NULL ) {
return - 1 ;
}
2005-06-17 02:47:26 +00:00
2005-08-18 15:02:01 +00:00
dn = ldb_dn_linearize ( mods , msg - > dn ) ;
if ( dn = = NULL ) {
talloc_free ( mods ) ;
return - 1 ;
}
ildb - > last_rc = ildap_add ( ildb - > ldap , dn , mods ) ;
2005-06-17 02:47:26 +00:00
if ( ! NT_STATUS_IS_OK ( ildb - > last_rc ) ) {
ret = - 1 ;
}
talloc_free ( mods ) ;
return ret ;
}
/*
modify a record
*/
static int ildb_modify ( struct ldb_module * module , const struct ldb_message * msg )
{
struct ldb_context * ldb = module - > ldb ;
struct ildb_private * ildb = module - > private_data ;
struct ldap_mod * * mods ;
2005-08-18 15:02:01 +00:00
char * dn ;
2005-06-17 02:47:26 +00:00
int ret = 0 ;
/* ignore ltdb specials */
2005-08-18 15:02:01 +00:00
if ( ldb_dn_is_special ( msg - > dn ) ) {
2005-06-17 02:47:26 +00:00
return 0 ;
}
mods = ildb_msg_to_mods ( ldb , msg , 1 ) ;
2005-08-18 15:02:01 +00:00
if ( mods = = NULL ) {
return - 1 ;
}
dn = ldb_dn_linearize ( mods , msg - > dn ) ;
if ( dn = = NULL ) {
talloc_free ( mods ) ;
return - 1 ;
}
2005-06-17 02:47:26 +00:00
2005-08-18 15:02:01 +00:00
ildb - > last_rc = ildap_modify ( ildb - > ldap , dn , mods ) ;
2005-06-17 02:47:26 +00:00
if ( ! NT_STATUS_IS_OK ( ildb - > last_rc ) ) {
ret = - 1 ;
}
talloc_free ( mods ) ;
return ret ;
}
2005-09-17 19:25:50 +00:00
static int ildb_start_trans ( struct ldb_module * module )
2005-06-17 02:47:26 +00:00
{
/* TODO implement a local locking mechanism here */
2005-09-17 19:25:50 +00:00
return 0 ;
2005-06-17 02:47:26 +00:00
}
2005-09-17 19:25:50 +00:00
static int ildb_end_trans ( struct ldb_module * module , int status )
2005-06-17 02:47:26 +00:00
{
2005-09-17 19:25:50 +00:00
/* TODO implement a local transaction mechanism here */
2005-06-17 02:47:26 +00:00
2005-09-17 19:25:50 +00:00
return status ;
2005-06-17 02:47:26 +00:00
}
/*
return extended error information
*/
static const char * ildb_errstring ( struct ldb_module * module )
{
2005-06-18 13:18:43 +00:00
struct ildb_private * ildb = talloc_get_type ( module - > private_data ,
struct ildb_private ) ;
if ( ildb = = NULL ) {
return " ildap not connected " ;
}
2005-06-17 02:47:26 +00:00
return ldap_errstr ( ildb - > ldap , ildb - > last_rc ) ;
}
static const struct ldb_module_ops ildb_ops = {
2005-09-17 19:25:50 +00:00
. name = " ldap " ,
. search = ildb_search ,
. search_bytree = ildb_search_bytree ,
. add_record = ildb_add ,
. modify_record = ildb_modify ,
. delete_record = ildb_delete ,
. rename_record = ildb_rename ,
. start_transaction = ildb_start_trans ,
. end_transaction = ildb_end_trans ,
. errstring = ildb_errstring
2005-06-17 02:47:26 +00:00
} ;
2005-06-18 09:48:17 +00:00
/*
fetch the rootDSE
*/
static void ildb_rootdse ( struct ldb_module * module )
{
struct ildb_private * ildb = module - > private_data ;
struct ldb_message * * res = NULL ;
2005-08-18 15:02:01 +00:00
struct ldb_dn * empty_dn = ldb_dn_new ( ildb ) ;
2005-06-18 09:48:17 +00:00
int ret ;
2005-08-18 15:02:01 +00:00
ret = ildb_search ( module , empty_dn , LDB_SCOPE_BASE , " dn=dc=rootDSE " , NULL , & res ) ;
2005-06-18 09:48:17 +00:00
if ( ret = = 1 ) {
ildb - > rootDSE = talloc_steal ( ildb , res [ 0 ] ) ;
}
2005-08-18 15:02:01 +00:00
if ( ret ! = - 1 ) talloc_free ( res ) ;
talloc_free ( empty_dn ) ;
2005-06-18 09:48:17 +00:00
}
2005-06-17 02:47:26 +00:00
/*
connect to the database
*/
2005-06-18 07:42:21 +00:00
int ildb_connect ( struct ldb_context * ldb , const char * url ,
unsigned int flags , const char * options [ ] )
2005-06-17 02:47:26 +00:00
{
struct ildb_private * ildb = NULL ;
NTSTATUS status ;
ildb = talloc ( ldb , struct ildb_private ) ;
if ( ! ildb ) {
2005-06-18 07:42:21 +00:00
ldb_oom ( ldb ) ;
2005-06-17 02:47:26 +00:00
goto failed ;
}
2005-06-18 09:48:17 +00:00
ildb - > rootDSE = NULL ;
2005-06-20 04:56:43 +00:00
ildb - > ldap = ldap_new_connection ( ildb , ldb_get_opaque ( ldb , " EventContext " ) ) ;
2005-06-17 02:47:26 +00:00
if ( ! ildb - > ldap ) {
2005-06-18 07:42:21 +00:00
ldb_oom ( ldb ) ;
2005-06-17 02:47:26 +00:00
goto failed ;
}
status = ldap_connect ( ildb - > ldap , url ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-06-18 07:42:21 +00:00
ldb_debug ( ldb , LDB_DEBUG_ERROR , " Failed to connect to ldap URL '%s' - %s \n " ,
url , ldap_errstr ( ildb - > ldap , status ) ) ;
2005-06-17 02:47:26 +00:00
goto failed ;
}
ldb - > modules = talloc ( ldb , struct ldb_module ) ;
if ( ! ldb - > modules ) {
2005-06-18 07:42:21 +00:00
ldb_oom ( ldb ) ;
2005-06-17 02:47:26 +00:00
goto failed ;
}
ldb - > modules - > ldb = ldb ;
ldb - > modules - > prev = ldb - > modules - > next = NULL ;
ldb - > modules - > private_data = ildb ;
ldb - > modules - > ops = & ildb_ops ;
2005-08-29 04:30:22 +00:00
if ( cmdline_credentials ! = NULL & & cli_credentials_authentication_requested ( cmdline_credentials ) ) {
2005-06-18 09:01:09 +00:00
status = ldap_bind_sasl ( ildb - > ldap , cmdline_credentials ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
ldb_debug ( ldb , LDB_DEBUG_ERROR , " Failed to bind - %s \n " ,
ldap_errstr ( ildb - > ldap , status ) ) ;
goto failed ;
}
}
2005-06-18 07:42:21 +00:00
return 0 ;
2005-06-17 02:47:26 +00:00
failed :
2005-06-18 13:18:43 +00:00
if ( ldb - > modules ) {
ldb - > modules - > private_data = NULL ;
}
2005-06-18 07:42:21 +00:00
talloc_free ( ildb ) ;
return - 1 ;
2005-06-17 02:47:26 +00:00
}