mirror of
https://github.com/samba-team/samba.git
synced 2025-01-14 19:24:43 +03:00
97bef6613e
clusapi only works via DCE/RPC sealed connections in Windows 2012R2. Guenther Signed-off-by: Günther Deschner <gd@samba.org> Reviewed-by: Jose A. Rivera <jarrpa@samba.org>
912 lines
24 KiB
C
912 lines
24 KiB
C
/*
|
|
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 02139, 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"
|
|
#include "librpc/gen_ndr/ndr_clusapi_c.h"
|
|
#include "param/param.h"
|
|
#include <tevent.h>
|
|
#include "lib/cmdline/popt_common.h"
|
|
|
|
struct torture_test_clusapi_state {
|
|
struct dcerpc_pipe *p;
|
|
};
|
|
|
|
struct torture_test_witness_state {
|
|
const char *net_name;
|
|
const char *share_name;
|
|
struct witness_interfaceList *list;
|
|
struct policy_handle context_handle;
|
|
struct torture_test_clusapi_state clusapi;
|
|
};
|
|
|
|
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,
|
|
NCACN_NP, 0),
|
|
"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,
|
|
WERR_INVALID_PARAM,
|
|
"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,
|
|
.expected_result = WERR_INVALID_PARAM
|
|
},{
|
|
.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_INVALID_PARAM
|
|
},{
|
|
.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),
|
|
.ip_address = NULL,
|
|
.client_computer_name = lpcfg_netbios_name(tctx->lp_ctx),
|
|
.expected_status = NT_STATUS_OK,
|
|
.expected_result = WERR_INVALID_PARAM
|
|
}
|
|
|
|
};
|
|
|
|
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,
|
|
.expected_result = WERR_INVALID_PARAM
|
|
},{
|
|
.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_INVALID_PARAM
|
|
},{
|
|
.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),
|
|
.ip_address = NULL,
|
|
.client_computer_name = lpcfg_netbios_name(tctx->lp_ctx),
|
|
.expected_status = NT_STATUS_OK,
|
|
.expected_result = WERR_INVALID_PARAM
|
|
}
|
|
|
|
};
|
|
|
|
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;
|
|
|
|
/*
|
|
* 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;
|
|
|
|
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;
|
|
}
|
|
|
|
static bool setup_clusapi_connection(struct torture_context *tctx,
|
|
struct torture_test_witness_state *s)
|
|
{
|
|
struct dcerpc_binding *binding;
|
|
|
|
if (s->clusapi.p) {
|
|
return true;
|
|
}
|
|
|
|
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,
|
|
cmdline_credentials,
|
|
tctx->ev, tctx->lp_ctx),
|
|
"failed to connect dcerpc pipe");
|
|
|
|
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
|
|
|
|
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,
|
|
const char *resource_name,
|
|
enum clusapi_ClusterResourceState *old_state,
|
|
enum clusapi_ClusterResourceState *new_state)
|
|
{
|
|
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");
|
|
|
|
if (old_state) {
|
|
*old_state = State;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
torture_assert(tctx,
|
|
test_GetResourceState_int(tctx, p, &hResource, &State),
|
|
"failed to query resource state");
|
|
|
|
if (new_state) {
|
|
*new_state = State;
|
|
}
|
|
|
|
test_CloseResource_int(tctx, p, &hResource);
|
|
|
|
return true;
|
|
}
|
|
|
|
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);
|
|
|
|
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];
|
|
struct witness_Register reg;
|
|
struct tevent_req *req;
|
|
enum clusapi_ClusterResourceState old_state, new_state;
|
|
|
|
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, ®),
|
|
"Register failed");
|
|
|
|
torture_assert_werr_ok(tctx,
|
|
reg.out.result,
|
|
"Register failed");
|
|
|
|
r.in.context_handle = state->context_handle;
|
|
r.out.response = &response;
|
|
|
|
req = dcerpc_witness_AsyncNotify_r_send(tctx, tctx->ev, b, &r);
|
|
torture_assert(tctx, req, "failed to create request");
|
|
|
|
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");
|
|
|
|
torture_assert(tctx,
|
|
tevent_req_poll(req, tctx->ev),
|
|
"failed to call event loop");
|
|
|
|
torture_assert_ntstatus_ok(tctx,
|
|
dcerpc_witness_AsyncNotify_r_recv(req, tctx),
|
|
"failed to receive reply");
|
|
|
|
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");
|
|
}
|
|
torture_assert(tctx,
|
|
test_witness_UnRegister_with_handle(tctx, p, &state->context_handle),
|
|
"Failed to unregister");
|
|
|
|
ZERO_STRUCT(state->context_handle);
|
|
|
|
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");
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
tmp = localtime(&tp.tv_sec);
|
|
if (!tmp) {
|
|
torture_comment(tctx, "failed to call localtime");
|
|
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;
|
|
}
|
|
|
|
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);
|
|
torture_rpc_tcase_add_test_ex(tcase, "AsyncNotify_timeouts",
|
|
test_witness_AsyncNotify_timeouts, state);
|
|
|
|
return suite;
|
|
}
|