2015-01-16 13:49:46 +01:00
/*
Unix SMB / CIFS implementation .
test suite for rpc witness operations
Copyright ( C ) Guenther Deschner 2015
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 .
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 "torture/rpc/torture_rpc.h"
# include "librpc/gen_ndr/ndr_witness_c.h"
# include "librpc/gen_ndr/ndr_srvsvc_c.h"
2015-03-26 13:12:06 +01:00
# include "librpc/gen_ndr/ndr_clusapi_c.h"
2015-01-16 13:49:46 +01:00
# include "param/param.h"
2015-03-26 13:12:35 +01:00
# include <tevent.h>
2020-12-02 17:24:22 +01:00
# include "lib/cmdline/cmdline.h"
2015-01-16 13:49:46 +01:00
2015-03-26 13:12:06 +01:00
struct torture_test_clusapi_state {
struct dcerpc_pipe * p ;
} ;
2015-01-16 13:49:46 +01:00
struct torture_test_witness_state {
const char * net_name ;
const char * share_name ;
struct witness_interfaceList * list ;
struct policy_handle context_handle ;
2015-03-26 13:12:06 +01:00
struct torture_test_clusapi_state clusapi ;
2015-01-16 13:49:46 +01:00
} ;
static bool test_witness_GetInterfaceList ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
void * data )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct witness_GetInterfaceList r ;
struct witness_interfaceList * l ;
struct torture_test_witness_state * state =
( struct torture_test_witness_state * ) data ;
r . out . interface_list = & l ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_GetInterfaceList_r ( b , tctx , & r ) ,
" GetInterfaceList failed " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" GetInterfaceList failed " ) ;
state - > list = l ;
return true ;
}
static bool find_sofs_share ( struct torture_context * tctx ,
const char * * sofs_sharename )
{
struct dcerpc_pipe * p ;
struct dcerpc_binding_handle * b ;
struct srvsvc_NetShareEnumAll r ;
struct srvsvc_NetShareInfoCtr info_ctr ;
struct srvsvc_NetShareCtr1 ctr1 ;
uint32_t resume_handle = 0 ;
uint32_t totalentries = 0 ;
int i ;
torture_assert_ntstatus_ok ( tctx ,
torture_rpc_connection_transport ( tctx , & p , & ndr_table_srvsvc ,
2016-09-15 08:36:32 +02:00
NCACN_NP , 0 , 0 ) ,
2015-01-16 13:49:46 +01:00
" failed to setup srvsvc connection " ) ;
b = p - > binding_handle ;
ZERO_STRUCT ( ctr1 ) ;
info_ctr . level = 1 ;
info_ctr . ctr . ctr1 = & ctr1 ;
r . in . server_unc = dcerpc_server_name ( p ) ;
r . in . max_buffer = - 1 ;
r . in . info_ctr = & info_ctr ;
r . in . resume_handle = & resume_handle ;
r . out . totalentries = & totalentries ;
r . out . info_ctr = & info_ctr ;
r . out . resume_handle = & resume_handle ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_srvsvc_NetShareEnumAll_r ( b , tctx , & r ) ,
" failed to call srvsvc_NetShareEnumAll " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" failed to call srvsvc_NetShareEnumAll " ) ;
for ( i = 0 ; i < r . out . info_ctr - > ctr . ctr1 - > count ; i + + ) {
if ( r . out . info_ctr - > ctr . ctr1 - > array [ i ] . type = = STYPE_CLUSTER_SOFS ) {
* sofs_sharename = talloc_strdup ( tctx , r . out . info_ctr - > ctr . ctr1 - > array [ i ] . name ) ;
if ( * sofs_sharename = = NULL ) {
return false ;
}
torture_comment ( tctx , " using SOFS share: %s \n " , * sofs_sharename ) ;
return true ;
}
if ( r . out . info_ctr - > ctr . ctr1 - > array [ i ] . type = = STYPE_DISKTREE ) {
* sofs_sharename = talloc_strdup ( tctx , r . out . info_ctr - > ctr . ctr1 - > array [ i ] . name ) ;
if ( * sofs_sharename = = NULL ) {
return false ;
}
torture_comment ( tctx , " assuming SOFS share: %s \n " , * sofs_sharename ) ;
return true ;
}
}
return false ;
}
static bool init_witness_test_state ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
struct torture_test_witness_state * state )
{
if ( state - > net_name = = NULL ) {
state - > net_name = lpcfg_parm_string ( tctx - > lp_ctx , NULL , " torture " , " net_name " ) ;
}
if ( state - > list = = NULL ) {
torture_assert ( tctx ,
test_witness_GetInterfaceList ( tctx , p , state ) ,
" failed to retrieve GetInterfaceList " ) ;
}
if ( state - > share_name = = NULL ) {
find_sofs_share ( tctx , & state - > share_name ) ;
}
return true ;
}
static bool test_witness_UnRegister_with_handle ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
struct policy_handle * context_handle )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct witness_UnRegister r ;
r . in . context_handle = * context_handle ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_UnRegister_r ( b , tctx , & r ) ,
" UnRegister failed " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" UnRegister failed " ) ;
/* make sure we are not able/allowed to reuse context handles after they
* have been unregistered */
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_UnRegister_r ( b , tctx , & r ) ,
" UnRegister failed " ) ;
torture_assert_werr_equal ( tctx ,
r . out . result ,
2015-12-03 15:24:27 +01:00
WERR_INVALID_PARAMETER ,
2015-01-16 13:49:46 +01:00
" UnRegister failed " ) ;
return true ;
}
static bool test_witness_UnRegister ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
void * data )
{
/* acquire handle and free afterwards */
return true ;
}
static bool get_ip_address_from_interface ( struct torture_context * tctx ,
struct witness_interfaceInfo * i ,
const char * * ip_address )
{
if ( i - > flags & WITNESS_INFO_IPv4_VALID ) {
* ip_address = talloc_strdup ( tctx , i - > ipv4 ) ;
torture_assert ( tctx , * ip_address , " talloc_strdup failed " ) ;
return true ;
}
if ( i - > flags & WITNESS_INFO_IPv6_VALID ) {
* ip_address = talloc_strdup ( tctx , i - > ipv6 ) ;
torture_assert ( tctx , * ip_address , " talloc_strdup failed " ) ;
return true ;
}
return false ;
}
static bool check_valid_interface ( struct torture_context * tctx ,
struct witness_interfaceInfo * i )
{
/* continue looking for an interface that allows witness
* registration */
if ( ! ( i - > flags & WITNESS_INFO_WITNESS_IF ) ) {
return false ;
}
/* witness should be available of course */
if ( i - > state ! = WITNESS_STATE_AVAILABLE ) {
return false ;
}
return true ;
}
static bool test_witness_Register ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
void * data )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct witness_Register r ;
struct policy_handle context_handle ;
struct torture_test_witness_state * state =
( struct torture_test_witness_state * ) data ;
int i ;
struct {
enum witness_version version ;
const char * net_name ;
const char * ip_address ;
const char * client_computer_name ;
NTSTATUS expected_status ;
WERROR expected_result ;
} tests [ ] = {
{
. version = 0 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = 1 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = 123456 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = - 1 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = WITNESS_V2 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = WITNESS_V1 ,
. net_name = " " ,
. ip_address = " " ,
. client_computer_name = " " ,
. expected_status = NT_STATUS_OK ,
2015-12-03 15:24:27 +01:00
. expected_result = WERR_INVALID_PARAMETER
2015-01-16 13:49:46 +01:00
} , {
. version = WITNESS_V1 ,
. net_name = NULL ,
. ip_address = NULL ,
. client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ,
. expected_status = NT_STATUS_OK ,
2015-12-03 15:24:27 +01:00
. expected_result = WERR_INVALID_PARAMETER
2015-01-16 13:49:46 +01:00
} , {
. version = WITNESS_V2 ,
. net_name = NULL ,
. ip_address = NULL ,
. client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = WITNESS_V1 ,
. net_name = dcerpc_server_name ( p ) ,
2015-05-13 10:47:12 +02:00
. ip_address = NULL ,
2015-01-16 13:49:46 +01:00
. client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ,
. expected_status = NT_STATUS_OK ,
2015-12-03 15:24:27 +01:00
. expected_result = WERR_INVALID_PARAMETER
2015-01-16 13:49:46 +01:00
}
} ;
for ( i = 0 ; i < ARRAY_SIZE ( tests ) ; i + + ) {
ZERO_STRUCT ( r ) ;
r . out . context_handle = & context_handle ;
r . in . version = tests [ i ] . version ;
r . in . net_name = tests [ i ] . net_name ;
r . in . ip_address = tests [ i ] . ip_address ;
r . in . client_computer_name = tests [ i ] . client_computer_name ;
torture_assert_ntstatus_equal ( tctx ,
dcerpc_witness_Register_r ( b , tctx , & r ) ,
tests [ i ] . expected_status ,
" Register failed " ) ;
torture_assert_werr_equal ( tctx ,
r . out . result ,
tests [ i ] . expected_result ,
" Register failed " ) ;
if ( W_ERROR_IS_OK ( r . out . result ) ) {
/* we have a handle, make sure to unregister it */
torture_assert ( tctx ,
test_witness_UnRegister_with_handle ( tctx , p , r . out . context_handle ) ,
" Failed to unregister " ) ;
}
}
init_witness_test_state ( tctx , p , state ) ;
for ( i = 0 ; state - > list & & i < state - > list - > num_interfaces ; i + + ) {
const char * ip_address ;
struct witness_interfaceInfo interface = state - > list - > interfaces [ i ] ;
if ( ! check_valid_interface ( tctx , & interface ) ) {
continue ;
}
torture_assert ( tctx ,
get_ip_address_from_interface ( tctx , & interface , & ip_address ) ,
" failed to get ip_address from interface " ) ;
r . in . version = WITNESS_V1 ;
r . in . net_name = state - > net_name ;
r . in . ip_address = ip_address ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_Register_r ( b , tctx , & r ) ,
" Register failed " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" Register failed " ) ;
torture_assert ( tctx ,
test_witness_UnRegister_with_handle ( tctx , p , r . out . context_handle ) ,
" Failed to unregister " ) ;
}
return true ;
}
static bool test_witness_RegisterEx ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
void * data )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct witness_RegisterEx r ;
struct policy_handle context_handle ;
struct torture_test_witness_state * state =
( struct torture_test_witness_state * ) data ;
int i ;
struct {
enum witness_version version ;
const char * net_name ;
const char * ip_address ;
const char * client_computer_name ;
NTSTATUS expected_status ;
WERROR expected_result ;
} tests [ ] = {
{
. version = 0 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = 1 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = 123456 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = - 1 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = WITNESS_V1 ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = WITNESS_V2 ,
. net_name = " " ,
. ip_address = " " ,
. client_computer_name = " " ,
. expected_status = NT_STATUS_OK ,
2015-12-03 15:24:27 +01:00
. expected_result = WERR_INVALID_PARAMETER
2015-01-16 13:49:46 +01:00
} , {
. version = WITNESS_V2 ,
. net_name = NULL ,
. ip_address = NULL ,
. client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ,
. expected_status = NT_STATUS_OK ,
2015-12-03 15:24:27 +01:00
. expected_result = WERR_INVALID_PARAMETER
2015-01-16 13:49:46 +01:00
} , {
. version = WITNESS_V1 ,
. net_name = NULL ,
. ip_address = NULL ,
. client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ,
. expected_status = NT_STATUS_OK ,
. expected_result = WERR_REVISION_MISMATCH
} , {
. version = WITNESS_V2 ,
. net_name = dcerpc_server_name ( p ) ,
2015-05-13 10:47:12 +02:00
. ip_address = NULL ,
2015-01-16 13:49:46 +01:00
. client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ,
. expected_status = NT_STATUS_OK ,
2015-12-03 15:24:27 +01:00
. expected_result = WERR_INVALID_PARAMETER
2015-01-16 13:49:46 +01:00
}
} ;
for ( i = 0 ; i < ARRAY_SIZE ( tests ) ; i + + ) {
ZERO_STRUCT ( r ) ;
r . out . context_handle = & context_handle ;
r . in . version = tests [ i ] . version ;
r . in . net_name = tests [ i ] . net_name ;
r . in . ip_address = tests [ i ] . ip_address ;
r . in . client_computer_name = tests [ i ] . client_computer_name ;
torture_assert_ntstatus_equal ( tctx ,
dcerpc_witness_RegisterEx_r ( b , tctx , & r ) ,
tests [ i ] . expected_status ,
" RegisterEx failed " ) ;
torture_assert_werr_equal ( tctx ,
r . out . result ,
tests [ i ] . expected_result ,
" RegisterEx failed " ) ;
if ( W_ERROR_IS_OK ( r . out . result ) ) {
/* we have a handle, make sure to unregister it */
torture_assert ( tctx ,
test_witness_UnRegister_with_handle ( tctx , p , r . out . context_handle ) ,
" Failed to unregister " ) ;
}
}
init_witness_test_state ( tctx , p , state ) ;
for ( i = 0 ; state - > list & & i < state - > list - > num_interfaces ; i + + ) {
const char * ip_address ;
struct witness_interfaceInfo interface = state - > list - > interfaces [ i ] ;
if ( ! check_valid_interface ( tctx , & interface ) ) {
continue ;
}
torture_assert ( tctx ,
get_ip_address_from_interface ( tctx , & interface , & ip_address ) ,
" failed to get ip_address from interface " ) ;
r . in . version = WITNESS_V2 ;
r . in . net_name = state - > net_name ;
r . in . ip_address = ip_address ;
2015-04-22 13:56:35 +02:00
/*
* a valid request with an invalid sharename fails with
* WERR_INVALID_STATE
*/
r . in . share_name = " any_invalid_share_name " ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_RegisterEx_r ( b , tctx , & r ) ,
" RegisterEx failed " ) ;
torture_assert_werr_equal ( tctx ,
r . out . result ,
WERR_INVALID_STATE ,
" RegisterEx failed " ) ;
r . in . share_name = NULL ;
2015-01-16 13:49:46 +01:00
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_RegisterEx_r ( b , tctx , & r ) ,
" RegisterEx failed " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" RegisterEx failed " ) ;
torture_assert ( tctx ,
test_witness_UnRegister_with_handle ( tctx , p , r . out . context_handle ) ,
" Failed to unregister " ) ;
}
return true ;
}
2015-03-26 13:12:06 +01:00
static bool setup_clusapi_connection ( struct torture_context * tctx ,
struct torture_test_witness_state * s )
{
2015-07-01 19:29:28 +02:00
struct dcerpc_binding * binding ;
2015-05-13 10:47:37 +02:00
2015-03-26 13:12:06 +01:00
if ( s - > clusapi . p ) {
return true ;
}
2015-07-01 19:29:28 +02:00
torture_assert_ntstatus_ok ( tctx ,
torture_rpc_binding ( tctx , & binding ) ,
" failed to retrieve torture binding " ) ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_binding_set_transport ( binding , NCACN_IP_TCP ) ,
" failed to set transport " ) ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_binding_set_flags ( binding , DCERPC_SEAL , 0 ) ,
" failed to set dcerpc flags " ) ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_pipe_connect_b ( tctx , & s - > clusapi . p , binding ,
& ndr_table_clusapi ,
2020-12-02 17:24:22 +01:00
samba_cmdline_get_creds ( ) ,
2015-07-01 19:29:28 +02:00
tctx - > ev , tctx - > lp_ctx ) ,
" failed to connect dcerpc pipe " ) ;
2015-03-26 13:12:06 +01:00
return true ;
}
#if 0
static bool cluster_get_nodes ( struct torture_context * tctx ,
struct torture_test_witness_state * s )
{
struct clusapi_CreateEnum r ;
struct ENUM_LIST * ReturnEnum ;
WERROR rpc_status ;
struct dcerpc_binding_handle * b ;
torture_assert ( tctx ,
setup_clusapi_connection ( tctx , s ) ,
" failed to setup clusapi connection " ) ;
b = s - > clusapi . p - > binding_handle ;
r . in . dwType = CLUSTER_ENUM_NODE ;
r . out . ReturnEnum = & ReturnEnum ;
r . out . rpc_status = & rpc_status ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_clusapi_CreateEnum_r ( b , tctx , & r ) ,
" failed to enumerate nodes " ) ;
return true ;
}
# endif
2015-06-25 21:59:20 +02:00
static bool test_GetResourceState_int ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
struct policy_handle * hResource ,
enum clusapi_ClusterResourceState * State )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct clusapi_GetResourceState r ;
const char * NodeName ;
const char * GroupName ;
WERROR rpc_status ;
r . in . hResource = * hResource ;
r . out . State = State ;
r . out . NodeName = & NodeName ;
r . out . GroupName = & GroupName ;
r . out . rpc_status = & rpc_status ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_clusapi_GetResourceState_r ( b , tctx , & r ) ,
" GetResourceState failed " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" GetResourceState failed " ) ;
return true ;
}
static bool toggle_cluster_resource_state ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
2015-07-01 15:14:19 +02:00
const char * resource_name ,
enum clusapi_ClusterResourceState * old_state ,
enum clusapi_ClusterResourceState * new_state )
2015-06-25 21:59:20 +02:00
{
struct policy_handle hResource ;
enum clusapi_ClusterResourceState State ;
torture_assert ( tctx ,
test_OpenResource_int ( tctx , p , resource_name , & hResource ) ,
" failed to open resource " ) ;
torture_assert ( tctx ,
test_GetResourceState_int ( tctx , p , & hResource , & State ) ,
" failed to query resource state " ) ;
2015-07-01 15:14:19 +02:00
if ( old_state ) {
* old_state = State ;
}
2015-06-25 21:59:20 +02:00
switch ( State ) {
case ClusterResourceOffline :
if ( ! test_OnlineResource_int ( tctx , p , & hResource ) ) {
test_CloseResource_int ( tctx , p , & hResource ) ;
torture_warning ( tctx , " failed to set resource online " ) ;
return false ;
}
break ;
case ClusterResourceOnline :
if ( ! test_OfflineResource_int ( tctx , p , & hResource ) ) {
test_CloseResource_int ( tctx , p , & hResource ) ;
torture_warning ( tctx , " failed to set resource offline " ) ;
return false ;
}
break ;
default :
break ;
}
2015-07-01 15:14:19 +02:00
torture_assert ( tctx ,
test_GetResourceState_int ( tctx , p , & hResource , & State ) ,
" failed to query resource state " ) ;
if ( new_state ) {
* new_state = State ;
}
2015-06-25 21:59:20 +02:00
test_CloseResource_int ( tctx , p , & hResource ) ;
return true ;
}
2015-01-16 13:49:46 +01:00
static bool test_witness_AsyncNotify ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
void * data )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct witness_AsyncNotify r ;
struct witness_notifyResponse * response ;
struct torture_test_witness_state * state =
( struct torture_test_witness_state * ) data ;
int i ;
init_witness_test_state ( tctx , p , state ) ;
2015-03-26 13:12:06 +01:00
setup_clusapi_connection ( tctx , state ) ;
2015-01-16 13:49:46 +01:00
for ( i = 0 ; state - > list & & i < state - > list - > num_interfaces ; i + + ) {
const char * ip_address ;
struct witness_interfaceInfo interface = state - > list - > interfaces [ i ] ;
struct witness_Register reg ;
2015-03-26 13:12:35 +01:00
struct tevent_req * req ;
2015-07-01 15:14:19 +02:00
enum clusapi_ClusterResourceState old_state , new_state ;
2015-01-16 13:49:46 +01:00
if ( ! check_valid_interface ( tctx , & interface ) ) {
continue ;
}
torture_assert ( tctx ,
get_ip_address_from_interface ( tctx , & interface , & ip_address ) ,
" failed to get ip_address from interface " ) ;
reg . in . version = WITNESS_V1 ;
reg . in . net_name = state - > net_name ;
reg . in . ip_address = ip_address ;
reg . in . client_computer_name = lpcfg_netbios_name ( tctx - > lp_ctx ) ;
reg . out . context_handle = & state - > context_handle ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_Register_r ( b , tctx , & reg ) ,
" Register failed " ) ;
torture_assert_werr_ok ( tctx ,
reg . out . result ,
" Register failed " ) ;
r . in . context_handle = state - > context_handle ;
r . out . response = & response ;
2015-03-26 13:12:35 +01:00
req = dcerpc_witness_AsyncNotify_r_send ( tctx , tctx - > ev , b , & r ) ;
torture_assert ( tctx , req , " failed to create request " ) ;
torture_assert ( tctx ,
2015-07-01 15:14:19 +02:00
toggle_cluster_resource_state ( tctx , state - > clusapi . p , state - > net_name , & old_state , & new_state ) ,
2015-03-26 13:12:35 +01:00
" failed to toggle cluster resource state " ) ;
2015-07-01 15:14:19 +02:00
torture_assert ( tctx , old_state ! = new_state , " failed to change cluster resource state " ) ;
2015-03-26 13:12:35 +01:00
torture_assert ( tctx ,
tevent_req_poll ( req , tctx - > ev ) ,
" failed to call event loop " ) ;
2015-01-16 13:49:46 +01:00
torture_assert_ntstatus_ok ( tctx ,
2015-03-26 13:12:35 +01:00
dcerpc_witness_AsyncNotify_r_recv ( req , tctx ) ,
" failed to receive reply " ) ;
2015-01-16 13:49:46 +01:00
2015-07-01 15:40:06 +02:00
torture_assert_int_equal ( tctx , response - > num , 1 , " num " ) ;
torture_assert_int_equal ( tctx , response - > type , WITNESS_NOTIFY_RESOURCE_CHANGE , " type " ) ;
/*
* TODO : find out how ClusterResourceOfflinePending and
* ClusterResourceOnlinePending are represented as witness
* types .
*/
if ( new_state = = ClusterResourceOffline ) {
torture_assert_int_equal ( tctx , response - > messages [ 0 ] . resource_change . type , WITNESS_RESOURCE_STATE_UNAVAILABLE , " resource_change.type " ) ;
}
if ( new_state = = ClusterResourceOnline ) {
torture_assert_int_equal ( tctx , response - > messages [ 0 ] . resource_change . type , WITNESS_RESOURCE_STATE_AVAILABLE , " resource_change.type " ) ;
}
2015-01-16 13:49:46 +01:00
torture_assert ( tctx ,
test_witness_UnRegister_with_handle ( tctx , p , & state - > context_handle ) ,
" Failed to unregister " ) ;
ZERO_STRUCT ( state - > context_handle ) ;
2015-07-01 15:14:19 +02:00
torture_assert ( tctx ,
toggle_cluster_resource_state ( tctx , state - > clusapi . p , state - > net_name , & old_state , & new_state ) ,
" failed to toggle cluster resource state " ) ;
torture_assert ( tctx , old_state ! = new_state , " failed to change cluster resource state " ) ;
2015-01-16 13:49:46 +01:00
}
return true ;
}
2015-05-13 10:49:54 +02:00
static bool test_do_witness_RegisterEx ( struct torture_context * tctx ,
struct dcerpc_binding_handle * b ,
uint32_t version ,
const char * net_name ,
const char * share_name ,
const char * ip_address ,
const char * client_computer_name ,
uint32_t flags ,
uint32_t timeout ,
struct policy_handle * context_handle )
{
struct witness_RegisterEx r ;
r . in . version = version ;
r . in . net_name = net_name ;
r . in . share_name = NULL ;
r . in . ip_address = ip_address ;
r . in . client_computer_name = client_computer_name ;
r . in . flags = flags ;
r . in . timeout = timeout ;
r . out . context_handle = context_handle ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_RegisterEx_r ( b , tctx , & r ) ,
" RegisterEx failed " ) ;
torture_assert_werr_ok ( tctx ,
r . out . result ,
" RegisterEx failed " ) ;
return true ;
}
static void torture_subunit_report_time ( struct torture_context * tctx )
{
struct timespec tp ;
struct tm * tmp ;
char timestr [ 200 ] ;
if ( clock_gettime ( CLOCK_REALTIME , & tp ) ! = 0 ) {
torture_comment ( tctx , " failed to call clock_gettime " ) ;
return ;
}
2018-01-25 17:23:06 +13:00
tmp = gmtime ( & tp . tv_sec ) ;
2015-05-13 10:49:54 +02:00
if ( ! tmp ) {
2018-01-25 17:23:06 +13:00
torture_comment ( tctx , " failed to call gmtime " ) ;
2015-05-13 10:49:54 +02:00
return ;
}
if ( strftime ( timestr , sizeof ( timestr ) , " %Y-%m-%d %H:%M:%S " , tmp ) < = 0 ) {
torture_comment ( tctx , " failed to call strftime " ) ;
return ;
}
torture_comment ( tctx , " time: %s.%06ld \n " , timestr , tp . tv_nsec / 1000 ) ;
}
static bool test_witness_AsyncNotify_timeouts ( struct torture_context * tctx ,
struct dcerpc_pipe * p ,
void * data )
{
struct dcerpc_binding_handle * b = p - > binding_handle ;
struct witness_AsyncNotify r ;
struct witness_notifyResponse * response ;
struct torture_test_witness_state * state =
( struct torture_test_witness_state * ) data ;
int i ;
init_witness_test_state ( tctx , p , state ) ;
setup_clusapi_connection ( tctx , state ) ;
for ( i = 0 ; state - > list & & i < state - > list - > num_interfaces ; i + + ) {
const char * ip_address ;
struct witness_interfaceInfo interface = state - > list - > interfaces [ i ] ;
uint32_t timeouts [ ] = {
0 , 1 , 10 , 100 , 120
} ;
int t ;
uint32_t old_timeout ;
if ( ! check_valid_interface ( tctx , & interface ) ) {
continue ;
}
torture_assert ( tctx ,
get_ip_address_from_interface ( tctx , & interface , & ip_address ) ,
" failed to get ip_address from interface " ) ;
for ( t = 0 ; t < ARRAY_SIZE ( timeouts ) ; t + + ) {
torture_comment ( tctx , " Testing Async Notify with timeout of %d milliseconds " , timeouts [ t ] ) ;
torture_assert ( tctx ,
test_do_witness_RegisterEx ( tctx , b ,
WITNESS_V2 ,
state - > net_name ,
NULL ,
ip_address ,
lpcfg_netbios_name ( tctx - > lp_ctx ) ,
0 ,
timeouts [ t ] ,
& state - > context_handle ) ,
" failed to RegisterEx " ) ;
r . in . context_handle = state - > context_handle ;
r . out . response = & response ;
old_timeout = dcerpc_binding_handle_set_timeout ( b , UINT_MAX ) ;
torture_subunit_report_time ( tctx ) ;
torture_assert_ntstatus_ok ( tctx ,
dcerpc_witness_AsyncNotify_r ( b , tctx , & r ) ,
" AsyncNotify failed " ) ;
torture_assert_werr_equal ( tctx ,
r . out . result ,
WERR_TIMEOUT ,
" AsyncNotify failed " ) ;
torture_subunit_report_time ( tctx ) ;
dcerpc_binding_handle_set_timeout ( b , old_timeout ) ;
torture_assert ( tctx ,
test_witness_UnRegister_with_handle ( tctx , p , & state - > context_handle ) ,
" Failed to unregister " ) ;
ZERO_STRUCT ( state - > context_handle ) ;
}
}
return true ;
}
2015-01-16 13:49:46 +01:00
struct torture_suite * torture_rpc_witness ( TALLOC_CTX * mem_ctx )
{
struct torture_rpc_tcase * tcase ;
struct torture_suite * suite = torture_suite_create ( mem_ctx , " witness " ) ;
struct torture_test_witness_state * state ;
tcase = torture_suite_add_rpc_iface_tcase ( suite , " witness " ,
& ndr_table_witness ) ;
state = talloc_zero ( tcase , struct torture_test_witness_state ) ;
torture_rpc_tcase_add_test_ex ( tcase , " GetInterfaceList " ,
test_witness_GetInterfaceList , state ) ;
torture_rpc_tcase_add_test_ex ( tcase , " Register " ,
test_witness_Register , state ) ;
torture_rpc_tcase_add_test_ex ( tcase , " UnRegister " ,
test_witness_UnRegister , state ) ;
torture_rpc_tcase_add_test_ex ( tcase , " RegisterEx " ,
test_witness_RegisterEx , state ) ;
torture_rpc_tcase_add_test_ex ( tcase , " AsyncNotify " ,
test_witness_AsyncNotify , state ) ;
2015-05-13 10:49:54 +02:00
torture_rpc_tcase_add_test_ex ( tcase , " AsyncNotify_timeouts " ,
test_witness_AsyncNotify_timeouts , state ) ;
2015-01-16 13:49:46 +01:00
return suite ;
}