2007-04-23 00:43:47 +00:00
/*
Unix SMB / CIFS mplementation .
DSDB replication service
Copyright ( C ) Stefan Metzmacher 2007
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
2007-04-23 00:43:47 +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/>.
2007-04-23 00:43:47 +00:00
*/
# include "includes.h"
# include "dsdb/samdb/samdb.h"
# include "auth/auth.h"
# include "smbd/service.h"
# include "lib/events/events.h"
# include "lib/messaging/irpc.h"
# include "dsdb/repl/drepl_service.h"
# include "lib/ldb/include/ldb_errors.h"
2008-10-11 21:31:42 +02:00
# include "../lib/util/dlinklist.h"
2007-04-23 00:43:47 +00:00
# include "librpc/gen_ndr/ndr_misc.h"
# include "librpc/gen_ndr/ndr_drsuapi.h"
# include "librpc/gen_ndr/ndr_drsblobs.h"
2008-01-01 22:05:13 -06:00
# include "param/param.h"
2007-04-23 00:43:47 +00:00
WERROR dreplsrv_load_partitions ( struct dreplsrv_service * s )
{
WERROR status ;
struct ldb_dn * basedn ;
struct ldb_result * r ;
struct ldb_message_element * el ;
2009-12-20 11:06:23 +11:00
static const char * attrs [ ] = { " hasMasterNCs " , NULL } ;
2007-04-23 00:43:47 +00:00
uint32_t i ;
int ret ;
2009-12-20 11:06:23 +11:00
basedn = samdb_ntds_settings_dn ( s - > samdb ) ;
2007-04-23 00:43:47 +00:00
W_ERROR_HAVE_NO_MEMORY ( basedn ) ;
2008-09-23 14:30:06 -04:00
ret = ldb_search ( s - > samdb , s , & r , basedn , LDB_SCOPE_BASE , attrs ,
" (objectClass=*) " ) ;
2007-04-23 00:43:47 +00:00
if ( ret ! = LDB_SUCCESS ) {
return WERR_FOOBAR ;
} else if ( r - > count ! = 1 ) {
talloc_free ( r ) ;
return WERR_FOOBAR ;
}
2009-12-20 11:06:23 +11:00
el = ldb_msg_find_element ( r - > msgs [ 0 ] , " hasMasterNCs " ) ;
2007-04-23 00:43:47 +00:00
if ( ! el ) {
return WERR_FOOBAR ;
}
for ( i = 0 ; el & & i < el - > num_values ; i + + ) {
const char * v = ( const char * ) el - > values [ i ] . data ;
struct ldb_dn * pdn ;
struct dreplsrv_partition * p ;
pdn = ldb_dn_new ( s , s - > samdb , v ) ;
if ( ! ldb_dn_validate ( pdn ) ) {
return WERR_FOOBAR ;
}
p = talloc_zero ( s , struct dreplsrv_partition ) ;
W_ERROR_HAVE_NO_MEMORY ( p ) ;
p - > dn = talloc_steal ( p , pdn ) ;
DLIST_ADD ( s - > partitions , p ) ;
DEBUG ( 2 , ( " dreplsrv_partition[%s] loaded \n " , v ) ) ;
}
talloc_free ( r ) ;
status = dreplsrv_refresh_partitions ( s ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
return WERR_OK ;
}
2010-01-06 14:54:12 +11:00
WERROR dreplsrv_out_connection_attach ( struct dreplsrv_service * s ,
const struct repsFromTo1 * rft ,
struct dreplsrv_out_connection * * _conn )
2007-04-23 00:43:47 +00:00
{
struct dreplsrv_out_connection * cur , * conn = NULL ;
const char * hostname ;
if ( ! rft - > other_info ) {
return WERR_FOOBAR ;
}
if ( ! rft - > other_info - > dns_name ) {
return WERR_FOOBAR ;
}
hostname = rft - > other_info - > dns_name ;
for ( cur = s - > connections ; cur ; cur = cur - > next ) {
if ( strcmp ( cur - > binding - > host , hostname ) = = 0 ) {
conn = cur ;
break ;
}
}
if ( ! conn ) {
NTSTATUS nt_status ;
char * binding_str ;
conn = talloc_zero ( s , struct dreplsrv_out_connection ) ;
W_ERROR_HAVE_NO_MEMORY ( conn ) ;
conn - > service = s ;
binding_str = talloc_asprintf ( conn , " ncacn_ip_tcp:%s[krb5,seal] " ,
hostname ) ;
W_ERROR_HAVE_NO_MEMORY ( binding_str ) ;
nt_status = dcerpc_parse_binding ( conn , binding_str , & conn - > binding ) ;
talloc_free ( binding_str ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return ntstatus_to_werror ( nt_status ) ;
}
DLIST_ADD_END ( s - > connections , conn , struct dreplsrv_out_connection * ) ;
DEBUG ( 2 , ( " dreplsrv_out_connection_attach(%s): create \n " , conn - > binding - > host ) ) ;
} else {
DEBUG ( 2 , ( " dreplsrv_out_connection_attach(%s): attach \n " , conn - > binding - > host ) ) ;
}
* _conn = conn ;
return WERR_OK ;
}
static WERROR dreplsrv_partition_add_source_dsa ( struct dreplsrv_service * s ,
struct dreplsrv_partition * p ,
const struct ldb_val * val )
{
WERROR status ;
2007-11-09 19:24:51 +01:00
enum ndr_err_code ndr_err ;
2009-09-12 15:22:26 +10:00
struct dreplsrv_partition_source_dsa * source , * s2 ;
2007-04-23 00:43:47 +00:00
source = talloc_zero ( p , struct dreplsrv_partition_source_dsa ) ;
W_ERROR_HAVE_NO_MEMORY ( source ) ;
2008-01-01 22:05:13 -06:00
ndr_err = ndr_pull_struct_blob ( val , source ,
2008-01-02 01:52:18 -06:00
lp_iconv_convenience ( s - > task - > lp_ctx ) , & source - > _repsFromBlob ,
2007-11-09 19:24:51 +01:00
( ndr_pull_flags_fn_t ) ndr_pull_repsFromToBlob ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
NTSTATUS nt_status = ndr_map_error2ntstatus ( ndr_err ) ;
2009-09-12 15:22:26 +10:00
talloc_free ( source ) ;
2007-04-23 00:43:47 +00:00
return ntstatus_to_werror ( nt_status ) ;
}
/* NDR_PRINT_DEBUG(repsFromToBlob, &source->_repsFromBlob); */
if ( source - > _repsFromBlob . version ! = 1 ) {
2009-09-12 15:22:26 +10:00
talloc_free ( source ) ;
2007-04-23 00:43:47 +00:00
return WERR_DS_DRA_INTERNAL_ERROR ;
}
source - > partition = p ;
source - > repsFrom1 = & source - > _repsFromBlob . ctr . ctr1 ;
status = dreplsrv_out_connection_attach ( s , source - > repsFrom1 , & source - > conn ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
2009-09-12 15:22:26 +10:00
/* remove any existing source with the same GUID */
for ( s2 = p - > sources ; s2 ; s2 = s2 - > next ) {
if ( GUID_compare ( & s2 - > repsFrom1 - > source_dsa_obj_guid ,
& source - > repsFrom1 - > source_dsa_obj_guid ) = = 0 ) {
talloc_free ( s2 - > repsFrom1 - > other_info ) ;
* s2 - > repsFrom1 = * source - > repsFrom1 ;
talloc_steal ( s2 , s2 - > repsFrom1 - > other_info ) ;
talloc_free ( source ) ;
return WERR_OK ;
}
}
2007-04-23 00:43:47 +00:00
DLIST_ADD_END ( p - > sources , source , struct dreplsrv_partition_source_dsa * ) ;
return WERR_OK ;
}
2010-01-09 14:29:39 +11:00
/*
convert from one udv format to the other
*/
static WERROR udv_convert ( TALLOC_CTX * mem_ctx ,
const struct replUpToDateVectorCtr2 * udv ,
struct drsuapi_DsReplicaCursorCtrEx * udv_ex )
{
int i ;
udv_ex - > version = 2 ;
udv_ex - > reserved1 = 0 ;
udv_ex - > reserved2 = 0 ;
udv_ex - > count = udv - > count ;
udv_ex - > cursors = talloc_array ( mem_ctx , struct drsuapi_DsReplicaCursor , udv - > count ) ;
W_ERROR_HAVE_NO_MEMORY ( udv_ex - > cursors ) ;
for ( i = 0 ; i < udv - > count ; i + + ) {
udv_ex - > cursors [ i ] . source_dsa_invocation_id = udv - > cursors [ i ] . source_dsa_invocation_id ;
udv_ex - > cursors [ i ] . highest_usn = udv - > cursors [ i ] . highest_usn ;
}
return WERR_OK ;
}
/*
add our local UDV element for the partition
*/
static WERROR add_local_udv ( struct dreplsrv_service * s ,
struct dreplsrv_partition * p ,
const struct GUID * our_invocation_id ,
struct drsuapi_DsReplicaCursorCtrEx * udv )
{
int ret ;
uint64_t highest_usn ;
int i ;
2010-01-13 14:08:56 -02:00
ret = dsdb_load_partition_usn ( s - > samdb , p - > dn , & highest_usn , NULL ) ;
2010-01-09 14:29:39 +11:00
if ( ret ! = LDB_SUCCESS ) {
/* nothing to add */
return WERR_OK ;
}
for ( i = 0 ; i < udv - > count ; i + + ) {
if ( GUID_equal ( our_invocation_id , & udv - > cursors [ i ] . source_dsa_invocation_id ) ) {
udv - > cursors [ i ] . highest_usn = highest_usn ;
return WERR_OK ;
}
}
udv - > cursors = talloc_realloc ( p , udv - > cursors , struct drsuapi_DsReplicaCursor , udv - > count + 1 ) ;
W_ERROR_HAVE_NO_MEMORY ( udv - > cursors ) ;
udv - > cursors [ udv - > count ] . source_dsa_invocation_id = * our_invocation_id ;
udv - > cursors [ udv - > count ] . highest_usn = highest_usn ;
udv - > count + + ;
return WERR_OK ;
}
2007-04-23 00:43:47 +00:00
static WERROR dreplsrv_refresh_partition ( struct dreplsrv_service * s ,
2009-09-12 15:22:26 +10:00
struct dreplsrv_partition * p )
2007-04-23 00:43:47 +00:00
{
WERROR status ;
struct dom_sid * nc_sid ;
struct ldb_message_element * orf_el = NULL ;
struct ldb_result * r ;
uint32_t i ;
int ret ;
2009-09-12 15:22:26 +10:00
TALLOC_CTX * mem_ctx = talloc_new ( p ) ;
2007-12-20 00:02:15 +01:00
static const char * attrs [ ] = {
2007-04-23 00:43:47 +00:00
" objectSid " ,
" objectGUID " ,
" repsFrom " ,
NULL
} ;
DEBUG ( 2 , ( " dreplsrv_refresh_partition(%s) \n " ,
ldb_dn_get_linearized ( p - > dn ) ) ) ;
2008-09-23 14:30:06 -04:00
ret = ldb_search ( s - > samdb , mem_ctx , & r , p - > dn , LDB_SCOPE_BASE , attrs ,
" (objectClass=*) " ) ;
2007-04-23 00:43:47 +00:00
if ( ret ! = LDB_SUCCESS ) {
2009-09-12 15:22:26 +10:00
talloc_free ( mem_ctx ) ;
2007-04-23 00:43:47 +00:00
return WERR_FOOBAR ;
}
2009-09-12 15:22:26 +10:00
talloc_free ( discard_const ( p - > nc . dn ) ) ;
2007-04-23 00:43:47 +00:00
ZERO_STRUCT ( p - > nc ) ;
p - > nc . dn = ldb_dn_alloc_linearized ( p , p - > dn ) ;
W_ERROR_HAVE_NO_MEMORY ( p - > nc . dn ) ;
p - > nc . guid = samdb_result_guid ( r - > msgs [ 0 ] , " objectGUID " ) ;
nc_sid = samdb_result_dom_sid ( p , r - > msgs [ 0 ] , " objectSid " ) ;
if ( nc_sid ) {
p - > nc . sid = * nc_sid ;
2009-09-12 15:22:26 +10:00
talloc_free ( nc_sid ) ;
2007-04-23 00:43:47 +00:00
}
2010-01-09 14:29:39 +11:00
talloc_free ( p - > uptodatevector . cursors ) ;
talloc_free ( p - > uptodatevector_ex . cursors ) ;
ZERO_STRUCT ( p - > uptodatevector ) ;
ZERO_STRUCT ( p - > uptodatevector_ex ) ;
2010-01-16 11:08:44 +11:00
ret = dsdb_load_udv_v2 ( s - > samdb , p - > dn , p , & p - > uptodatevector . cursors , & p - > uptodatevector . count ) ;
if ( ret = = LDB_SUCCESS ) {
2010-01-09 14:29:39 +11:00
status = udv_convert ( p , & p - > uptodatevector , & p - > uptodatevector_ex ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
}
2007-04-23 00:43:47 +00:00
2010-01-09 20:53:27 +11:00
status = add_local_udv ( s , p , samdb_ntds_invocation_id ( s - > samdb ) , & p - > uptodatevector_ex ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
2007-04-23 00:43:47 +00:00
orf_el = ldb_msg_find_element ( r - > msgs [ 0 ] , " repsFrom " ) ;
if ( orf_el ) {
for ( i = 0 ; i < orf_el - > num_values ; i + + ) {
status = dreplsrv_partition_add_source_dsa ( s , p , & orf_el - > values [ i ] ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
}
}
2009-09-12 15:22:26 +10:00
talloc_free ( mem_ctx ) ;
2007-04-23 00:43:47 +00:00
return WERR_OK ;
}
2009-09-11 22:47:11 +10:00
WERROR dreplsrv_refresh_partitions ( struct dreplsrv_service * s )
2007-04-23 00:43:47 +00:00
{
WERROR status ;
struct dreplsrv_partition * p ;
for ( p = s - > partitions ; p ; p = p - > next ) {
2009-09-12 15:22:26 +10:00
status = dreplsrv_refresh_partition ( s , p ) ;
2007-04-23 00:43:47 +00:00
W_ERROR_NOT_OK_RETURN ( status ) ;
}
return WERR_OK ;
}