2006-05-30 17:57:35 +00:00
/*
Unix SMB / CIFS mplementation .
LDAP schema tests
Copyright ( C ) Stefan Metzmacher 2006
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 ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
# include "libcli/ldap/ldap_client.h"
# include "lib/cmdline/popt_common.h"
# include "db_wrap.h"
# include "lib/ldb/include/ldb.h"
# include "lib/ldb/include/ldb_errors.h"
# include "dsdb/samdb/samdb.h"
2006-08-30 11:29:34 +00:00
# include "lib/util/dlinklist.h"
2006-05-30 17:57:35 +00:00
# include "torture/torture.h"
# include "torture/ldap/proto.h"
struct test_rootDSE {
const char * defaultdn ;
const char * rootdn ;
const char * configdn ;
const char * schemadn ;
} ;
2006-07-22 16:56:33 +00:00
struct test_schema_ctx {
2006-05-30 17:57:35 +00:00
struct ldb_paged_control * ctrl ;
uint32_t count ;
BOOL pending ;
int ( * callback ) ( void * , struct ldb_context * ldb , struct ldb_message * ) ;
void * private_data ;
} ;
static BOOL test_search_rootDSE ( struct ldb_context * ldb , struct test_rootDSE * root )
{
int ret ;
struct ldb_message * msg ;
struct ldb_result * r ;
d_printf ( " Testing RootDSE Search \n " ) ;
2006-11-22 00:59:34 +00:00
ret = ldb_search ( ldb , ldb_dn_new ( ldb , ldb , NULL ) , LDB_SCOPE_BASE ,
2006-05-30 17:57:35 +00:00
NULL , NULL , & r ) ;
if ( ret ! = LDB_SUCCESS ) {
return False ;
} else if ( r - > count ! = 1 ) {
2006-10-15 23:14:19 +00:00
talloc_free ( r ) ;
2006-05-30 17:57:35 +00:00
return False ;
}
msg = r - > msgs [ 0 ] ;
2006-08-13 08:00:36 +00:00
root - > defaultdn = ldb_msg_find_attr_as_string ( msg , " defaultNamingContext " , NULL ) ;
2006-05-30 17:57:35 +00:00
talloc_steal ( ldb , root - > defaultdn ) ;
2006-08-13 08:00:36 +00:00
root - > rootdn = ldb_msg_find_attr_as_string ( msg , " rootDomainNamingContext " , NULL ) ;
2006-05-30 17:57:35 +00:00
talloc_steal ( ldb , root - > rootdn ) ;
2006-08-13 08:00:36 +00:00
root - > configdn = ldb_msg_find_attr_as_string ( msg , " configurationNamingContext " , NULL ) ;
2006-05-30 17:57:35 +00:00
talloc_steal ( ldb , root - > configdn ) ;
2006-08-13 08:00:36 +00:00
root - > schemadn = ldb_msg_find_attr_as_string ( msg , " schemaNamingContext " , NULL ) ;
2006-05-30 17:57:35 +00:00
talloc_steal ( ldb , root - > schemadn ) ;
talloc_free ( r ) ;
return True ;
}
2006-07-22 16:56:33 +00:00
static int test_schema_search_callback ( struct ldb_context * ldb , void * context , struct ldb_reply * ares )
2006-05-30 17:57:35 +00:00
{
2006-07-22 16:56:33 +00:00
struct test_schema_ctx * actx = talloc_get_type ( context , struct test_schema_ctx ) ;
2006-05-30 17:57:35 +00:00
int ret = LDB_SUCCESS ;
switch ( ares - > type ) {
case LDB_REPLY_ENTRY :
actx - > count + + ;
ret = actx - > callback ( actx - > private_data , ldb , ares - > message ) ;
break ;
case LDB_REPLY_REFERRAL :
break ;
case LDB_REPLY_DONE :
if ( ares - > controls ) {
struct ldb_paged_control * ctrl = NULL ;
int i ;
for ( i = 0 ; ares - > controls [ i ] ; i + + ) {
if ( strcmp ( LDB_CONTROL_PAGED_RESULTS_OID , ares - > controls [ i ] - > oid ) = = 0 ) {
ctrl = talloc_get_type ( ares - > controls [ i ] - > data , struct ldb_paged_control ) ;
break ;
}
}
if ( ! ctrl ) break ;
talloc_free ( actx - > ctrl - > cookie ) ;
actx - > ctrl - > cookie = talloc_steal ( actx - > ctrl - > cookie , ctrl - > cookie ) ;
actx - > ctrl - > cookie_len = ctrl - > cookie_len ;
if ( actx - > ctrl - > cookie_len > 0 ) {
actx - > pending = True ;
}
}
break ;
default :
d_printf ( " %s: unknown Reply Type %u \n " , __location__ , ares - > type ) ;
return LDB_ERR_OTHER ;
}
if ( talloc_free ( ares ) = = - 1 ) {
d_printf ( " talloc_free failed \n " ) ;
actx - > pending = 0 ;
return LDB_ERR_OPERATIONS_ERROR ;
}
if ( ret ) {
return LDB_ERR_OPERATIONS_ERROR ;
}
return LDB_SUCCESS ;
}
static BOOL test_create_schema_type ( struct ldb_context * ldb , struct test_rootDSE * root ,
const char * filter ,
int ( * callback ) ( void * , struct ldb_context * ldb , struct ldb_message * ) ,
void * private_data )
{
struct ldb_control * * ctrl ;
struct ldb_paged_control * control ;
struct ldb_request * req ;
int ret ;
2006-07-22 16:56:33 +00:00
struct test_schema_ctx * actx ;
2006-05-30 17:57:35 +00:00
req = talloc ( ldb , struct ldb_request ) ;
2006-07-22 16:56:33 +00:00
actx = talloc ( req , struct test_schema_ctx ) ;
2006-05-30 17:57:35 +00:00
ctrl = talloc_array ( req , struct ldb_control * , 2 ) ;
ctrl [ 0 ] = talloc ( ctrl , struct ldb_control ) ;
ctrl [ 0 ] - > oid = LDB_CONTROL_PAGED_RESULTS_OID ;
ctrl [ 0 ] - > critical = True ;
control = talloc ( ctrl [ 0 ] , struct ldb_paged_control ) ;
control - > size = 1000 ;
control - > cookie = NULL ;
control - > cookie_len = 0 ;
ctrl [ 0 ] - > data = control ;
ctrl [ 1 ] = NULL ;
req - > operation = LDB_SEARCH ;
2006-11-22 00:59:34 +00:00
req - > op . search . base = ldb_dn_new ( req , ldb , root - > schemadn ) ;
2006-05-30 17:57:35 +00:00
req - > op . search . scope = LDB_SCOPE_SUBTREE ;
2006-10-16 11:58:19 +00:00
req - > op . search . tree = ldb_parse_tree ( req , filter ) ;
2006-05-30 17:57:35 +00:00
if ( req - > op . search . tree = = NULL ) return - 1 ;
req - > op . search . attrs = NULL ;
req - > controls = ctrl ;
2006-07-22 17:21:59 +00:00
req - > context = actx ;
req - > callback = test_schema_search_callback ;
2006-06-04 05:28:13 +00:00
ldb_set_timeout ( ldb , req , 0 ) ;
2006-05-30 17:57:35 +00:00
actx - > count = 0 ;
actx - > ctrl = control ;
actx - > callback = callback ;
actx - > private_data = private_data ;
again :
actx - > pending = False ;
ret = ldb_request ( ldb , req ) ;
if ( ret ! = LDB_SUCCESS ) {
d_printf ( " search failed - %s \n " , ldb_errstring ( ldb ) ) ;
return False ;
}
2006-07-22 17:21:59 +00:00
ret = ldb_wait ( req - > handle , LDB_WAIT_ALL ) ;
2006-05-30 17:57:35 +00:00
if ( ret ! = LDB_SUCCESS ) {
d_printf ( " search error - %s \n " , ldb_errstring ( ldb ) ) ;
return False ;
}
if ( actx - > pending )
goto again ;
d_printf ( " filter[%s] count[%u] \n " , filter , actx - > count ) ;
return True ;
}
static int test_add_attribute ( void * ptr , struct ldb_context * ldb , struct ldb_message * msg )
{
struct dsdb_schema * schema = talloc_get_type ( ptr , struct dsdb_schema ) ;
2006-12-17 02:19:56 +00:00
struct dsdb_attribute * attr = NULL ;
WERROR status ;
2006-05-30 17:57:35 +00:00
attr = talloc_zero ( schema , struct dsdb_attribute ) ;
2006-12-17 02:19:56 +00:00
if ( ! attr ) {
goto failed ;
}
2006-05-30 17:57:35 +00:00
2006-12-17 02:19:56 +00:00
status = dsdb_attribute_from_ldb ( msg , attr , attr ) ;
if ( ! W_ERROR_IS_OK ( status ) ) {
goto failed ;
}
2006-05-30 17:57:35 +00:00
DLIST_ADD_END ( schema - > attributes , attr , struct dsdb_attribute * ) ;
return LDB_SUCCESS ;
failed :
2006-12-17 02:19:56 +00:00
talloc_free ( attr ) ;
2006-05-30 17:57:35 +00:00
return LDB_ERR_OTHER ;
}
static int test_add_class ( void * ptr , struct ldb_context * ldb , struct ldb_message * msg )
{
struct dsdb_schema * schema = talloc_get_type ( ptr , struct dsdb_schema ) ;
2006-12-17 02:19:56 +00:00
struct dsdb_class * obj ;
WERROR status ;
2006-05-30 17:57:35 +00:00
2006-12-17 02:19:56 +00:00
obj = talloc_zero ( schema , struct dsdb_class ) ;
if ( ! obj ) {
goto failed ;
}
2006-05-30 17:57:35 +00:00
2006-12-17 02:19:56 +00:00
status = dsdb_class_from_ldb ( msg , obj , obj ) ;
if ( ! W_ERROR_IS_OK ( status ) ) {
goto failed ;
}
2006-05-30 17:57:35 +00:00
2006-12-17 02:19:56 +00:00
DLIST_ADD_END ( schema - > classes , obj , struct dsdb_class * ) ;
2006-05-30 17:57:35 +00:00
return LDB_SUCCESS ;
failed :
return LDB_ERR_OTHER ;
}
static BOOL test_create_schema ( struct ldb_context * ldb , struct test_rootDSE * root , struct dsdb_schema * * _schema )
{
BOOL ret = True ;
struct dsdb_schema * schema ;
schema = talloc_zero ( ldb , struct dsdb_schema ) ;
d_printf ( " Fetching attributes... \n " ) ;
ret & = test_create_schema_type ( ldb , root , " (objectClass=attributeSchema) " ,
test_add_attribute , schema ) ;
d_printf ( " Fetching objectClasses... \n " ) ;
ret & = test_create_schema_type ( ldb , root , " (objectClass=classSchema) " ,
test_add_class , schema ) ;
if ( ret = = True ) {
* _schema = schema ;
}
return ret ;
}
static BOOL test_dump_not_replicated ( struct ldb_context * ldb , struct test_rootDSE * root , struct dsdb_schema * schema )
{
struct dsdb_attribute * a ;
uint32_t a_i = 1 ;
d_printf ( " Dumping not replicated attributes \n " ) ;
for ( a = schema - > attributes ; a ; a = a - > next ) {
if ( ! ( a - > systemFlags & 0x00000001 ) ) continue ;
d_printf ( " attr[%4u]: '%s' \n " , a_i + + ,
a - > lDAPDisplayName ) ;
}
return True ;
}
static BOOL test_dump_partial ( struct ldb_context * ldb , struct test_rootDSE * root , struct dsdb_schema * schema )
{
struct dsdb_attribute * a ;
uint32_t a_i = 1 ;
d_printf ( " Dumping attributes which are provided by the global catalog \n " ) ;
for ( a = schema - > attributes ; a ; a = a - > next ) {
if ( ! ( a - > systemFlags & 0x00000002 ) & & ! a - > isMemberOfPartialAttributeSet ) continue ;
d_printf ( " attr[%4u]: %u %u '%s' \n " , a_i + + ,
a - > systemFlags & 0x00000002 , a - > isMemberOfPartialAttributeSet ,
a - > lDAPDisplayName ) ;
}
return True ;
}
static BOOL test_dump_contructed ( struct ldb_context * ldb , struct test_rootDSE * root , struct dsdb_schema * schema )
{
struct dsdb_attribute * a ;
uint32_t a_i = 1 ;
d_printf ( " Dumping constructed attributes \n " ) ;
for ( a = schema - > attributes ; a ; a = a - > next ) {
if ( ! ( a - > systemFlags & 0x00000004 ) ) continue ;
d_printf ( " attr[%4u]: '%s' \n " , a_i + + ,
a - > lDAPDisplayName ) ;
}
return True ;
}
static BOOL test_dump_sorted_syntax ( struct ldb_context * ldb , struct test_rootDSE * root , struct dsdb_schema * schema )
{
struct dsdb_attribute * a ;
uint32_t a_i = 1 ;
uint32_t i ;
const char * syntaxes [ ] = {
" 2.5.5.0 " ,
" 2.5.5.1 " ,
" 2.5.5.2 " ,
" 2.5.5.3 " ,
" 2.5.5.4 " ,
" 2.5.5.5 " ,
" 2.5.5.6 " ,
" 2.5.5.7 " ,
" 2.5.5.8 " ,
" 2.5.5.9 " ,
" 2.5.5.10 " ,
" 2.5.5.11 " ,
" 2.5.5.12 " ,
" 2.5.5.13 " ,
" 2.5.5.14 " ,
" 2.5.5.15 " ,
" 2.5.5.16 " ,
" 2.5.5.17 "
} ;
for ( i = 0 ; i < ARRAY_SIZE ( syntaxes ) ; i + + ) {
for ( a = schema - > attributes ; a ; a = a - > next ) {
2006-12-17 02:19:56 +00:00
if ( strcmp ( syntaxes [ i ] , a - > attributeSyntax_oid ) ! = 0 ) continue ;
2006-05-30 17:57:35 +00:00
d_printf ( " attr[%4u]: %s %u '%s' \n " , a_i + + ,
2006-12-17 02:19:56 +00:00
a - > attributeSyntax_oid , a - > oMSyntax ,
2006-05-30 17:57:35 +00:00
a - > lDAPDisplayName ) ;
}
}
return True ;
}
BOOL torture_ldap_schema ( struct torture_context * torture )
{
struct ldb_context * ldb ;
BOOL ret = True ;
2006-10-18 14:23:19 +00:00
const char * host = torture_setting_string ( torture , " host " , NULL ) ;
2006-05-30 17:57:35 +00:00
char * url ;
struct test_rootDSE rootDSE ;
struct dsdb_schema * schema = NULL ;
ZERO_STRUCT ( rootDSE ) ;
2006-12-17 02:19:56 +00:00
url = talloc_asprintf ( torture , " ldap://%s/ " , host ) ;
2006-05-30 17:57:35 +00:00
2006-12-17 02:19:56 +00:00
ldb = ldb_wrap_connect ( torture , url ,
2006-05-30 17:57:35 +00:00
NULL ,
cmdline_credentials ,
0 , NULL ) ;
2006-05-30 19:08:44 +00:00
if ( ! ldb ) goto failed ;
2006-05-30 17:57:35 +00:00
ret & = test_search_rootDSE ( ldb , & rootDSE ) ;
if ( ! ret ) goto failed ;
ret & = test_create_schema ( ldb , & rootDSE , & schema ) ;
if ( ! ret ) goto failed ;
ret & = test_dump_not_replicated ( ldb , & rootDSE , schema ) ;
ret & = test_dump_partial ( ldb , & rootDSE , schema ) ;
ret & = test_dump_contructed ( ldb , & rootDSE , schema ) ;
ret & = test_dump_sorted_syntax ( ldb , & rootDSE , schema ) ;
failed :
return ret ;
}