2010-10-17 10:23:41 +02:00
/*
2007-11-29 15:08:27 +01:00
Unix SMB / CIFS implementation .
LDB wrap functions
2009-10-23 14:27:00 +11:00
Copyright ( C ) Andrew Tridgell 2004 - 2009
2010-10-17 10:23:41 +02:00
2007-11-29 15:08:27 +01: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 3 of the License , or
( at your option ) any later version .
2010-10-17 10:23:41 +02:00
2007-11-29 15:08:27 +01:00
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 .
2010-10-17 10:23:41 +02:00
2007-11-29 15:08:27 +01:00
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
/*
the stupidity of the unix fcntl locking design forces us to never
allow a database file to be opened twice in the same process . These
wrappers provide convenient access to a tdb or ldb , taking advantage
of talloc destructors to ensure that only a single open is done
*/
# include "includes.h"
# include "lib/events/events.h"
2011-02-10 14:12:51 +11:00
# include <ldb.h>
# include <ldb_errors.h>
2007-11-29 15:08:27 +01:00
# include "lib/ldb-samba/ldif_handlers.h"
# include "ldb_wrap.h"
# include "dsdb/samdb/samdb.h"
2018-05-31 15:12:46 +12:00
# include "dsdb/common/util.h"
2007-11-29 15:08:27 +01:00
# include "param/param.h"
2009-12-20 18:05:38 +01:00
# include "../lib/util/dlinklist.h"
2017-06-22 16:10:52 +02:00
# include "lib/util/util_paths.h"
2012-06-19 12:43:10 +09:30
# include <tdb.h>
2018-04-19 09:39:17 +10:00
# include <unistd.h>
2007-11-29 15:08:27 +01:00
2013-12-10 17:45:14 +01:00
# undef DBGC_CLASS
2013-07-09 13:56:08 +02:00
# define DBGC_CLASS DBGC_LDB
2007-11-29 15:08:27 +01:00
/*
this is used to catch debug messages from ldb
*/
2010-10-17 10:23:41 +02:00
static void ldb_wrap_debug ( void * context , enum ldb_debug_level level ,
2007-11-29 15:08:27 +01:00
const char * fmt , va_list ap ) PRINTF_ATTRIBUTE ( 3 , 0 ) ;
2010-10-17 10:23:41 +02:00
static void ldb_wrap_debug ( void * context , enum ldb_debug_level level ,
2007-11-29 15:08:27 +01:00
const char * fmt , va_list ap )
{
2008-05-15 18:09:56 +02:00
int samba_level = - 1 ;
2007-11-29 15:08:27 +01:00
switch ( level ) {
case LDB_DEBUG_FATAL :
samba_level = 0 ;
break ;
case LDB_DEBUG_ERROR :
samba_level = 1 ;
break ;
case LDB_DEBUG_WARNING :
samba_level = 2 ;
break ;
case LDB_DEBUG_TRACE :
2013-07-09 13:56:35 +02:00
samba_level = 10 ;
2007-11-29 15:08:27 +01:00
break ;
2010-10-10 16:56:55 +02:00
2007-11-29 15:08:27 +01:00
} ;
2012-08-09 19:41:05 +10:00
if ( CHECK_DEBUGLVL ( samba_level ) ) {
char * s = NULL ;
2015-06-23 18:29:53 +02:00
int ret ;
ret = vasprintf ( & s , fmt , ap ) ;
if ( ret = = - 1 ) {
return ;
}
2012-08-09 19:41:05 +10:00
DEBUG ( samba_level , ( " ldb: %s \n " , s ) ) ;
free ( s ) ;
}
2007-11-29 15:08:27 +01:00
}
2009-10-23 14:27:00 +11:00
/*
connecting to a ldb can be a relatively expensive operation because
of the schema and partition loads . We keep a list of open ldb
2010-10-17 10:23:41 +02:00
contexts here , and try to re - use when possible .
2009-10-23 14:27:00 +11:00
This means callers of ldb_wrap_connect ( ) must use talloc_unlink ( ) or
the free of a parent to destroy the context
*/
static struct ldb_wrap {
struct ldb_wrap * next , * prev ;
struct ldb_wrap_context {
/* the context is what we use to tell if two ldb
2010-10-17 10:23:41 +02:00
* connections are exactly equivalent
*/
2018-03-13 16:43:54 +13:00
pid_t pid ; /* We want to re-open in a new PID due to
* the LMDB backend */
2009-10-23 14:27:00 +11:00
const char * url ;
struct tevent_context * ev ;
struct loadparm_context * lp_ctx ;
struct auth_session_info * session_info ;
struct cli_credentials * credentials ;
unsigned int flags ;
} context ;
struct ldb_context * ldb ;
} * ldb_wrap_list ;
2010-10-17 10:23:41 +02:00
/*
2009-10-23 14:27:00 +11:00
free a ldb_wrap structure
*/
static int ldb_wrap_destructor ( struct ldb_wrap * w )
{
DLIST_REMOVE ( ldb_wrap_list , w ) ;
2008-09-24 23:59:59 +02:00
return 0 ;
2009-10-23 14:27:00 +11:00
}
2007-11-29 15:08:27 +01:00
2010-10-17 10:20:40 +02:00
/*
* The casefolder for s4 ' s LDB databases - Unicode - safe
*/
char * wrap_casefold ( void * context , void * mem_ctx , const char * s , size_t n )
{
return strupper_talloc_n ( mem_ctx , s , n ) ;
}
2010-10-10 17:34:21 +02:00
struct ldb_context * samba_ldb_init ( TALLOC_CTX * mem_ctx ,
2010-10-17 10:23:41 +02:00
struct tevent_context * ev ,
struct loadparm_context * lp_ctx ,
struct auth_session_info * session_info ,
struct cli_credentials * credentials )
2007-11-29 15:08:27 +01:00
{
2010-10-10 17:34:21 +02:00
struct ldb_context * ldb ;
int ret ;
2008-06-14 11:24:17 -04:00
ldb = ldb_init ( mem_ctx , ev ) ;
if ( ldb = = NULL ) {
2008-04-21 17:58:23 -04:00
return NULL ;
2008-04-17 12:23:44 +02:00
}
2007-11-29 15:08:27 +01:00
2011-06-06 14:37:06 +10:00
ldb_set_modules_dir ( ldb , modules_path ( ldb , " ldb " ) ) ;
2008-06-14 11:24:17 -04:00
2010-10-10 17:15:11 +02:00
ldb_set_debug ( ldb , ldb_wrap_debug , NULL ) ;
ldb_set_utf8_fns ( ldb , NULL , wrap_casefold ) ;
2010-06-23 21:15:43 +10:00
if ( session_info ) {
2018-05-31 15:12:46 +12:00
if ( ldb_set_opaque ( ldb , DSDB_SESSION_INFO , session_info ) ) {
2010-06-23 21:15:43 +10:00
talloc_free ( ldb ) ;
return NULL ;
}
2007-11-29 15:08:27 +01:00
}
2010-06-23 21:15:43 +10:00
if ( credentials ) {
if ( ldb_set_opaque ( ldb , " credentials " , credentials ) ) {
talloc_free ( ldb ) ;
return NULL ;
}
2007-11-29 15:08:27 +01:00
}
2007-12-02 19:04:33 +01:00
if ( ldb_set_opaque ( ldb , " loadparm " , lp_ctx ) ) {
talloc_free ( ldb ) ;
return NULL ;
}
2007-11-29 15:08:27 +01:00
2008-08-21 17:29:47 +10:00
/* This must be done before we load the schema, as these
* handlers for objectSid and objectGUID etc must take
* precedence over the ' binary attribute ' declaration in the
* schema */
2007-11-29 15:08:27 +01:00
ret = ldb_register_samba_handlers ( ldb ) ;
2010-09-11 17:41:38 +02:00
if ( ret ! = LDB_SUCCESS ) {
2007-11-29 15:08:27 +01:00
talloc_free ( ldb ) ;
return NULL ;
}
/* we usually want Samba databases to be private. If we later
find we need one public , we will need to add a parameter to
ldb_wrap_connect ( ) */
ldb_set_create_perms ( ldb , 0600 ) ;
2010-10-10 17:15:11 +02:00
return ldb ;
}
2010-10-10 17:34:21 +02:00
struct ldb_context * ldb_wrap_find ( const char * url ,
2010-10-17 10:23:41 +02:00
struct tevent_context * ev ,
struct loadparm_context * lp_ctx ,
struct auth_session_info * session_info ,
struct cli_credentials * credentials ,
2011-04-10 19:54:31 +02:00
unsigned int flags )
2010-10-10 17:34:21 +02:00
{
2018-03-13 16:43:54 +13:00
pid_t pid = getpid ( ) ;
2010-10-10 17:34:21 +02:00
struct ldb_wrap * w ;
/* see if we can re-use an existing ldb */
for ( w = ldb_wrap_list ; w ; w = w - > next ) {
2018-03-13 16:43:54 +13:00
if ( w - > context . pid = = pid & &
w - > context . ev = = ev & &
2010-10-10 17:34:21 +02:00
w - > context . lp_ctx = = lp_ctx & &
w - > context . session_info = = session_info & &
w - > context . credentials = = credentials & &
w - > context . flags = = flags & &
( w - > context . url = = url | | strcmp ( w - > context . url , url ) = = 0 ) )
return w - > ldb ;
}
return NULL ;
}
2010-10-17 10:23:41 +02:00
int samba_ldb_connect ( struct ldb_context * ldb , struct loadparm_context * lp_ctx ,
2011-04-10 19:54:31 +02:00
const char * url , unsigned int flags )
2010-10-10 17:15:11 +02:00
{
int ret ;
char * real_url = NULL ;
2010-10-10 23:25:38 +02:00
/* allow admins to force non-sync ldb for all databases */
if ( lpcfg_parm_bool ( lp_ctx , NULL , " ldb " , " nosync " , false ) ) {
flags | = LDB_FLG_NOSYNC ;
}
2010-10-10 17:15:11 +02:00
2010-10-10 23:25:38 +02:00
if ( DEBUGLVL ( 10 ) ) {
flags | = LDB_FLG_ENABLE_TRACING ;
2010-10-10 17:15:11 +02:00
}
2011-04-29 12:47:11 +10:00
real_url = lpcfg_private_path ( ldb , lp_ctx , url ) ;
2010-10-10 17:15:11 +02:00
if ( real_url = = NULL ) {
2010-10-10 23:25:38 +02:00
return LDB_ERR_OPERATIONS_ERROR ;
2010-10-10 17:15:11 +02:00
}
2009-10-23 14:27:00 +11:00
ret = ldb_connect ( ldb , real_url , flags , NULL ) ;
2010-10-10 23:25:38 +02:00
2007-11-29 15:08:27 +01:00
if ( ret ! = LDB_SUCCESS ) {
2010-10-10 23:25:38 +02:00
return ret ;
2007-11-29 15:08:27 +01:00
}
/* setup for leak detection */
ldb_set_opaque ( ldb , " wrap_url " , real_url ) ;
2010-10-10 23:25:38 +02:00
return LDB_SUCCESS ;
}
bool ldb_wrap_add ( const char * url , struct tevent_context * ev ,
2010-10-17 10:23:41 +02:00
struct loadparm_context * lp_ctx ,
struct auth_session_info * session_info ,
struct cli_credentials * credentials ,
2011-04-10 19:54:31 +02:00
unsigned int flags ,
2010-10-17 10:23:41 +02:00
struct ldb_context * ldb )
2010-10-10 23:25:38 +02:00
{
struct ldb_wrap * w ;
struct ldb_wrap_context c ;
2009-10-23 14:27:00 +11:00
/* add to the list of open ldb contexts */
w = talloc ( ldb , struct ldb_wrap ) ;
if ( w = = NULL ) {
2010-10-10 23:25:38 +02:00
return false ;
2009-10-23 14:27:00 +11:00
}
2018-03-13 16:43:54 +13:00
c . pid = getpid ( ) ;
2010-10-10 17:34:21 +02:00
c . url = url ;
c . ev = ev ;
c . lp_ctx = lp_ctx ;
c . session_info = session_info ;
c . credentials = credentials ;
c . flags = flags ;
2009-10-23 14:27:00 +11:00
w - > context = c ;
w - > context . url = talloc_strdup ( w , url ) ;
if ( w - > context . url = = NULL ) {
2010-10-10 23:25:38 +02:00
return false ;
2009-10-23 14:27:00 +11:00
}
2010-08-17 18:29:42 +10:00
if ( session_info ) {
/* take a reference to the session_info, as it is
* possible for the ldb to live longer than the
* session_info . This happens when a DRS DsBind call
* reuses a handle , but the original connection is
* shutdown . The token for the new connection is still
* valid , so we need the session_info to remain valid for
* ldb modules to use
*/
if ( talloc_reference ( w , session_info ) = = NULL ) {
2010-10-10 23:25:38 +02:00
return false ;
2010-08-17 18:29:42 +10:00
}
}
2009-10-23 14:27:00 +11:00
w - > ldb = ldb ;
DLIST_ADD ( ldb_wrap_list , w ) ;
talloc_set_destructor ( w , ldb_wrap_destructor ) ;
2007-11-29 15:08:27 +01:00
2010-10-10 23:25:38 +02:00
return true ;
}
/*
wrapped connection to a ldb database
to close just talloc_free ( ) the returned ldb_context
TODO : We need an error_string parameter
*/
struct ldb_context * ldb_wrap_connect ( TALLOC_CTX * mem_ctx ,
2010-10-17 10:23:41 +02:00
struct tevent_context * ev ,
struct loadparm_context * lp_ctx ,
const char * url ,
struct auth_session_info * session_info ,
struct cli_credentials * credentials ,
unsigned int flags )
2010-10-10 23:25:38 +02:00
{
struct ldb_context * ldb ;
int ret ;
2018-03-15 13:44:52 +13:00
/*
* Unlike samdb_connect_url ( ) do not try and cache the LDB
* handle , get a new one each time . Only sam . ldb is
* punitively expensive to open and helpful caches like this
* cause challenges ( such as if the value for ' private dir '
* changes ) .
*/
2010-10-10 23:25:38 +02:00
ldb = samba_ldb_init ( mem_ctx , ev , lp_ctx , session_info , credentials ) ;
if ( ldb = = NULL )
return NULL ;
ret = samba_ldb_connect ( ldb , lp_ctx , url , flags ) ;
if ( ret ! = LDB_SUCCESS ) {
talloc_free ( ldb ) ;
return NULL ;
}
DEBUG ( 3 , ( " ldb_wrap open of %s \n " , url ) ) ;
2007-11-29 15:08:27 +01:00
return ldb ;
}
2009-10-23 14:27:00 +11:00
/*
2018-03-15 13:42:17 +13:00
call tdb_reopen_all ( ) in case there is a TDB open so we are
not blocked from re - opening it inside ldb_tdb .
*/
2010-06-16 13:43:38 +02:00
void ldb_wrap_fork_hook ( void )
2009-10-23 14:27:00 +11:00
{
2012-01-30 09:25:50 +10:30
if ( tdb_reopen_all ( 1 ) ! = 0 ) {
2009-10-23 14:50:56 +11:00
smb_panic ( " tdb_reopen_all failed \n " ) ;
}
2009-10-23 14:27:00 +11:00
}
2010-10-10 23:45:23 +02:00
char * ldb_relative_path ( struct ldb_context * ldb ,
2010-10-17 10:23:41 +02:00
TALLOC_CTX * mem_ctx ,
const char * name )
2010-10-10 23:45:23 +02:00
{
2010-10-17 10:23:41 +02:00
const char * base_url =
2010-10-10 23:45:23 +02:00
( const char * ) ldb_get_opaque ( ldb , " ldb_url " ) ;
char * path , * p , * full_name ;
if ( name = = NULL ) {
return NULL ;
}
if ( strncmp ( " tdb:// " , base_url , 6 ) = = 0 ) {
base_url = base_url + 6 ;
2018-03-06 13:40:21 +13:00
} else if ( strncmp ( " mdb:// " , base_url , 6 ) = = 0 ) {
base_url = base_url + 6 ;
2018-03-06 13:40:21 +13:00
} else if ( strncmp ( " ldb:// " , base_url , 6 ) = = 0 ) {
base_url = base_url + 6 ;
2010-10-10 23:45:23 +02:00
}
path = talloc_strdup ( mem_ctx , base_url ) ;
if ( path = = NULL ) {
return NULL ;
}
if ( ( p = strrchr ( path , ' / ' ) ) ! = NULL ) {
p [ 0 ] = ' \0 ' ;
full_name = talloc_asprintf ( mem_ctx , " %s/%s " , path , name ) ;
} else {
full_name = talloc_asprintf ( mem_ctx , " ./%s " , name ) ;
}
talloc_free ( path ) ;
return full_name ;
}