LCOV - code coverage report
Current view: top level - source4/torture/rpc - witness.c (source / functions) Hit Total Coverage
Test: coverage report for master 70ed9daf Lines: 13 325 4.0 %
Date: 2024-01-11 09:59:51 Functions: 1 17 5.9 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/CIFS implementation.
       3             :    test suite for rpc witness operations
       4             : 
       5             :    Copyright (C) Guenther Deschner 2015
       6             : 
       7             :    This program is free software; you can redistribute it and/or modify
       8             :    it under the terms of the GNU General Public License as published by
       9             :    the Free Software Foundation; either version 3 of the License, or
      10             :    (at your option) any later version.
      11             : 
      12             :    This program is distributed in the hope that it will be useful,
      13             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      14             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      15             :    GNU General Public License for more details.
      16             : 
      17             :    You should have received a copy of the GNU General Public License
      18             :    along with this program; if not, write to the Free Software
      19             :    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
      20             : */
      21             : 
      22             : #include "includes.h"
      23             : #include "torture/rpc/torture_rpc.h"
      24             : #include "librpc/gen_ndr/ndr_witness_c.h"
      25             : #include "librpc/gen_ndr/ndr_srvsvc_c.h"
      26             : #include "librpc/gen_ndr/ndr_clusapi_c.h"
      27             : #include "param/param.h"
      28             : #include <tevent.h>
      29             : #include "lib/cmdline/cmdline.h"
      30             : 
      31             : struct torture_test_clusapi_state {
      32             :         struct dcerpc_pipe *p;
      33             : };
      34             : 
      35             : struct torture_test_witness_state {
      36             :         const char *net_name;
      37             :         const char *share_name;
      38             :         struct witness_interfaceList *list;
      39             :         struct policy_handle context_handle;
      40             :         struct torture_test_clusapi_state clusapi;
      41             : };
      42             : 
      43           0 : static bool test_witness_GetInterfaceList(struct torture_context *tctx,
      44             :                                           struct dcerpc_pipe *p,
      45             :                                           void *data)
      46             : {
      47           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
      48           0 :         struct witness_GetInterfaceList r;
      49           0 :         struct witness_interfaceList *l;
      50           0 :         struct torture_test_witness_state *state =
      51             :                 (struct torture_test_witness_state *)data;
      52             : 
      53           0 :         r.out.interface_list = &l;
      54             : 
      55           0 :         torture_assert_ntstatus_ok(tctx,
      56             :                 dcerpc_witness_GetInterfaceList_r(b, tctx, &r),
      57             :                 "GetInterfaceList failed");
      58             : 
      59           0 :         torture_assert_werr_ok(tctx,
      60             :                 r.out.result,
      61             :                 "GetInterfaceList failed");
      62             : 
      63           0 :         state->list = l;
      64             : 
      65           0 :         return true;
      66             : }
      67             : 
      68           0 : static bool find_sofs_share(struct torture_context *tctx,
      69             :                             const char **sofs_sharename)
      70             : {
      71           0 :         struct dcerpc_pipe *p;
      72           0 :         struct dcerpc_binding_handle *b;
      73           0 :         struct srvsvc_NetShareEnumAll r;
      74           0 :         struct srvsvc_NetShareInfoCtr info_ctr;
      75           0 :         struct srvsvc_NetShareCtr1 ctr1;
      76           0 :         uint32_t resume_handle = 0;
      77           0 :         uint32_t totalentries = 0;
      78           0 :         int i;
      79             : 
      80           0 :         torture_assert_ntstatus_ok(tctx,
      81             :                 torture_rpc_connection_transport(tctx, &p, &ndr_table_srvsvc,
      82             :                                                  NCACN_NP, 0, 0),
      83             :                 "failed to setup srvsvc connection");
      84             : 
      85           0 :         b = p->binding_handle;
      86             : 
      87           0 :         ZERO_STRUCT(ctr1);
      88             : 
      89           0 :         info_ctr.level = 1;
      90           0 :         info_ctr.ctr.ctr1 = &ctr1;
      91             : 
      92           0 :         r.in.server_unc = dcerpc_server_name(p);
      93           0 :         r.in.max_buffer = -1;
      94           0 :         r.in.info_ctr = &info_ctr;
      95           0 :         r.in.resume_handle = &resume_handle;
      96           0 :         r.out.totalentries = &totalentries;
      97           0 :         r.out.info_ctr = &info_ctr;
      98           0 :         r.out.resume_handle = &resume_handle;
      99             : 
     100           0 :         torture_assert_ntstatus_ok(tctx,
     101             :                 dcerpc_srvsvc_NetShareEnumAll_r(b, tctx, &r),
     102             :                 "failed to call srvsvc_NetShareEnumAll");
     103             : 
     104           0 :         torture_assert_werr_ok(tctx,
     105             :                 r.out.result,
     106             :                 "failed to call srvsvc_NetShareEnumAll");
     107             : 
     108           0 :         for (i=0; i < r.out.info_ctr->ctr.ctr1->count; i++) {
     109             : 
     110           0 :                 if (r.out.info_ctr->ctr.ctr1->array[i].type == STYPE_CLUSTER_SOFS) {
     111           0 :                         *sofs_sharename = talloc_strdup(tctx, r.out.info_ctr->ctr.ctr1->array[i].name);
     112           0 :                         if (*sofs_sharename == NULL) {
     113           0 :                                 return false;
     114             :                         }
     115           0 :                         torture_comment(tctx, "using SOFS share: %s\n", *sofs_sharename);
     116           0 :                         return true;
     117             :                 }
     118           0 :                 if (r.out.info_ctr->ctr.ctr1->array[i].type == STYPE_DISKTREE) {
     119           0 :                         *sofs_sharename = talloc_strdup(tctx, r.out.info_ctr->ctr.ctr1->array[i].name);
     120           0 :                         if (*sofs_sharename == NULL) {
     121           0 :                                 return false;
     122             :                         }
     123           0 :                         torture_comment(tctx, "assuming SOFS share: %s\n", *sofs_sharename);
     124           0 :                         return true;
     125             :                 }
     126             :         }
     127             : 
     128           0 :         return false;
     129             : }
     130             : 
     131           0 : static bool init_witness_test_state(struct torture_context *tctx,
     132             :                                     struct dcerpc_pipe *p,
     133             :                                     struct torture_test_witness_state *state)
     134             : {
     135           0 :         if (state->net_name == NULL) {
     136           0 :                 state->net_name = lpcfg_parm_string(tctx->lp_ctx, NULL, "torture", "net_name");
     137             :         }
     138             : 
     139           0 :         if (state->list == NULL) {
     140           0 :                 torture_assert(tctx,
     141             :                         test_witness_GetInterfaceList(tctx, p, state),
     142             :                         "failed to retrieve GetInterfaceList");
     143             :         }
     144             : 
     145           0 :         if (state->share_name == NULL) {
     146           0 :                 find_sofs_share(tctx, &state->share_name);
     147             :         }
     148             : 
     149           0 :         return true;
     150             : }
     151             : 
     152           0 : static bool test_witness_UnRegister_with_handle(struct torture_context *tctx,
     153             :                                                 struct dcerpc_pipe *p,
     154             :                                                 struct policy_handle *context_handle)
     155             : {
     156           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
     157           0 :         struct witness_UnRegister r;
     158             : 
     159           0 :         r.in.context_handle = *context_handle;
     160             : 
     161           0 :         torture_assert_ntstatus_ok(tctx,
     162             :                 dcerpc_witness_UnRegister_r(b, tctx, &r),
     163             :                 "UnRegister failed");
     164             : 
     165           0 :         torture_assert_werr_ok(tctx,
     166             :                 r.out.result,
     167             :                 "UnRegister failed");
     168             : 
     169             :         /* make sure we are not able/allowed to reuse context handles after they
     170             :          * have been unregistered */
     171             : 
     172           0 :         torture_assert_ntstatus_ok(tctx,
     173             :                 dcerpc_witness_UnRegister_r(b, tctx, &r),
     174             :                 "UnRegister failed");
     175             : 
     176           0 :         torture_assert_werr_equal(tctx,
     177             :                 r.out.result,
     178             :                 WERR_INVALID_PARAMETER,
     179             :                 "UnRegister failed");
     180             : 
     181           0 :         return true;
     182             : }
     183             : 
     184           0 : static bool test_witness_UnRegister(struct torture_context *tctx,
     185             :                                     struct dcerpc_pipe *p,
     186             :                                     void *data)
     187             : {
     188             :         /* acquire handle and free afterwards */
     189           0 :         return true;
     190             : }
     191             : 
     192           0 : static bool get_ip_address_from_interface(struct torture_context *tctx,
     193             :                                           struct witness_interfaceInfo *i,
     194             :                                           const char **ip_address)
     195             : {
     196           0 :         if (i->flags & WITNESS_INFO_IPv4_VALID) {
     197           0 :                 *ip_address = talloc_strdup(tctx, i->ipv4);
     198           0 :                 torture_assert(tctx, *ip_address, "talloc_strdup failed");
     199           0 :                 return true;
     200             :         }
     201             : 
     202           0 :         if (i->flags & WITNESS_INFO_IPv6_VALID) {
     203           0 :                 *ip_address = talloc_strdup(tctx, i->ipv6);
     204           0 :                 torture_assert(tctx, *ip_address, "talloc_strdup failed");
     205           0 :                 return true;
     206             :         }
     207             : 
     208           0 :         return false;
     209             : }
     210             : 
     211           0 : static bool check_valid_interface(struct torture_context *tctx,
     212             :                                   struct witness_interfaceInfo *i)
     213             : {
     214             :         /* continue looking for an interface that allows witness
     215             :          * registration */
     216           0 :         if (!(i->flags & WITNESS_INFO_WITNESS_IF)) {
     217           0 :                 return false;
     218             :         }
     219             : 
     220             :         /* witness should be available of course */
     221           0 :         if (i->state != WITNESS_STATE_AVAILABLE) {
     222           0 :                 return false;
     223             :         }
     224             : 
     225           0 :         return true;
     226             : }
     227             : 
     228           0 : static bool test_witness_Register(struct torture_context *tctx,
     229             :                                   struct dcerpc_pipe *p,
     230             :                                   void *data)
     231             : {
     232           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
     233           0 :         struct witness_Register r;
     234           0 :         struct policy_handle context_handle;
     235           0 :         struct torture_test_witness_state *state =
     236             :                 (struct torture_test_witness_state *)data;
     237           0 :         int i;
     238             : 
     239           0 :         struct {
     240             :                 enum witness_version version;
     241             :                 const char *net_name;
     242             :                 const char *ip_address;
     243             :                 const char *client_computer_name;
     244             :                 NTSTATUS expected_status;
     245             :                 WERROR expected_result;
     246           0 :         } tests[] = {
     247             :                 {
     248             :                         .version                = 0,
     249             :                         .expected_status        = NT_STATUS_OK,
     250             :                         .expected_result        = WERR_REVISION_MISMATCH
     251             :                 },{
     252             :                         .version                = 1,
     253             :                         .expected_status        = NT_STATUS_OK,
     254             :                         .expected_result        = WERR_REVISION_MISMATCH
     255             :                 },{
     256             :                         .version                = 123456,
     257             :                         .expected_status        = NT_STATUS_OK,
     258             :                         .expected_result        = WERR_REVISION_MISMATCH
     259             :                 },{
     260             :                         .version                = -1,
     261             :                         .expected_status        = NT_STATUS_OK,
     262             :                         .expected_result        = WERR_REVISION_MISMATCH
     263             :                 },{
     264             :                         .version                = WITNESS_V2,
     265             :                         .expected_status        = NT_STATUS_OK,
     266             :                         .expected_result        = WERR_REVISION_MISMATCH
     267             :                 },{
     268             :                         .version                = WITNESS_V1,
     269             :                         .net_name               = "",
     270             :                         .ip_address             = "",
     271             :                         .client_computer_name   = "",
     272             :                         .expected_status        = NT_STATUS_OK,
     273             :                         .expected_result        = WERR_INVALID_PARAMETER
     274             :                 },{
     275             :                         .version                = WITNESS_V1,
     276             :                         .net_name               = NULL,
     277             :                         .ip_address             = NULL,
     278           0 :                         .client_computer_name   = lpcfg_netbios_name(tctx->lp_ctx),
     279             :                         .expected_status        = NT_STATUS_OK,
     280             :                         .expected_result        = WERR_INVALID_PARAMETER
     281             :                 },{
     282             :                         .version                = WITNESS_V2,
     283             :                         .net_name               = NULL,
     284             :                         .ip_address             = NULL,
     285           0 :                         .client_computer_name   = lpcfg_netbios_name(tctx->lp_ctx),
     286             :                         .expected_status        = NT_STATUS_OK,
     287             :                         .expected_result        = WERR_REVISION_MISMATCH
     288             :                 },{
     289             :                         .version                = WITNESS_V1,
     290           0 :                         .net_name               = dcerpc_server_name(p),
     291             :                         .ip_address             = NULL,
     292           0 :                         .client_computer_name   = lpcfg_netbios_name(tctx->lp_ctx),
     293             :                         .expected_status        = NT_STATUS_OK,
     294             :                         .expected_result        = WERR_INVALID_PARAMETER
     295             :                 }
     296             : 
     297             :         };
     298             : 
     299           0 :         for (i=0; i < ARRAY_SIZE(tests); i++) {
     300             : 
     301           0 :                 ZERO_STRUCT(r);
     302             : 
     303           0 :                 r.out.context_handle = &context_handle;
     304             : 
     305           0 :                 r.in.version = tests[i].version;
     306           0 :                 r.in.net_name = tests[i].net_name;
     307           0 :                 r.in.ip_address = tests[i].ip_address;
     308           0 :                 r.in.client_computer_name = tests[i].client_computer_name;
     309             : 
     310           0 :                 torture_assert_ntstatus_equal(tctx,
     311             :                         dcerpc_witness_Register_r(b, tctx, &r),
     312             :                         tests[i].expected_status,
     313             :                         "Register failed");
     314             : 
     315           0 :                 torture_assert_werr_equal(tctx,
     316             :                         r.out.result,
     317             :                         tests[i].expected_result,
     318             :                         "Register failed");
     319             : 
     320           0 :                 if (W_ERROR_IS_OK(r.out.result)) {
     321             : 
     322             :                         /* we have a handle, make sure to unregister it */
     323           0 :                         torture_assert(tctx,
     324             :                                 test_witness_UnRegister_with_handle(tctx, p, r.out.context_handle),
     325             :                                 "Failed to unregister");
     326             :                 }
     327             :         }
     328             : 
     329           0 :         init_witness_test_state(tctx, p, state);
     330             : 
     331           0 :         for (i=0; state->list && i < state->list->num_interfaces; i++) {
     332             : 
     333           0 :                 const char *ip_address;
     334           0 :                 struct witness_interfaceInfo interface = state->list->interfaces[i];
     335             : 
     336           0 :                 if (!check_valid_interface(tctx, &interface)) {
     337           0 :                         continue;
     338             :                 }
     339             : 
     340           0 :                 torture_assert(tctx,
     341             :                         get_ip_address_from_interface(tctx, &interface, &ip_address),
     342             :                         "failed to get ip_address from interface");
     343             : 
     344           0 :                 r.in.version = WITNESS_V1;
     345           0 :                 r.in.net_name = state->net_name;
     346           0 :                 r.in.ip_address = ip_address;
     347             : 
     348           0 :                 torture_assert_ntstatus_ok(tctx,
     349             :                         dcerpc_witness_Register_r(b, tctx, &r),
     350             :                         "Register failed");
     351             : 
     352           0 :                 torture_assert_werr_ok(tctx,
     353             :                         r.out.result,
     354             :                         "Register failed");
     355             : 
     356           0 :                 torture_assert(tctx,
     357             :                         test_witness_UnRegister_with_handle(tctx, p, r.out.context_handle),
     358             :                         "Failed to unregister");
     359             :         }
     360             : 
     361           0 :         return true;
     362             : }
     363             : 
     364           0 : static bool test_witness_RegisterEx(struct torture_context *tctx,
     365             :                                     struct dcerpc_pipe *p,
     366             :                                     void *data)
     367             : {
     368           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
     369           0 :         struct witness_RegisterEx r;
     370           0 :         struct policy_handle context_handle;
     371           0 :         struct torture_test_witness_state *state =
     372             :                 (struct torture_test_witness_state *)data;
     373           0 :         int i;
     374             : 
     375           0 :         struct {
     376             :                 enum witness_version version;
     377             :                 const char *net_name;
     378             :                 const char *ip_address;
     379             :                 const char *client_computer_name;
     380             :                 NTSTATUS expected_status;
     381             :                 WERROR expected_result;
     382           0 :         } tests[] = {
     383             :                 {
     384             :                         .version                = 0,
     385             :                         .expected_status        = NT_STATUS_OK,
     386             :                         .expected_result        = WERR_REVISION_MISMATCH
     387             :                 },{
     388             :                         .version                = 1,
     389             :                         .expected_status        = NT_STATUS_OK,
     390             :                         .expected_result        = WERR_REVISION_MISMATCH
     391             :                 },{
     392             :                         .version                = 123456,
     393             :                         .expected_status        = NT_STATUS_OK,
     394             :                         .expected_result        = WERR_REVISION_MISMATCH
     395             :                 },{
     396             :                         .version                = -1,
     397             :                         .expected_status        = NT_STATUS_OK,
     398             :                         .expected_result        = WERR_REVISION_MISMATCH
     399             :                 },{
     400             :                         .version                = WITNESS_V1,
     401             :                         .expected_status        = NT_STATUS_OK,
     402             :                         .expected_result        = WERR_REVISION_MISMATCH
     403             :                 },{
     404             :                         .version                = WITNESS_V2,
     405             :                         .net_name               = "",
     406             :                         .ip_address             = "",
     407             :                         .client_computer_name   = "",
     408             :                         .expected_status        = NT_STATUS_OK,
     409             :                         .expected_result        = WERR_INVALID_PARAMETER
     410             :                 },{
     411             :                         .version                = WITNESS_V2,
     412             :                         .net_name               = NULL,
     413             :                         .ip_address             = NULL,
     414           0 :                         .client_computer_name   = lpcfg_netbios_name(tctx->lp_ctx),
     415             :                         .expected_status        = NT_STATUS_OK,
     416             :                         .expected_result        = WERR_INVALID_PARAMETER
     417             :                 },{
     418             :                         .version                = WITNESS_V1,
     419             :                         .net_name               = NULL,
     420             :                         .ip_address             = NULL,
     421           0 :                         .client_computer_name   = lpcfg_netbios_name(tctx->lp_ctx),
     422             :                         .expected_status        = NT_STATUS_OK,
     423             :                         .expected_result        = WERR_REVISION_MISMATCH
     424             :                 },{
     425             :                         .version                = WITNESS_V2,
     426           0 :                         .net_name               = dcerpc_server_name(p),
     427             :                         .ip_address             = NULL,
     428           0 :                         .client_computer_name   = lpcfg_netbios_name(tctx->lp_ctx),
     429             :                         .expected_status        = NT_STATUS_OK,
     430             :                         .expected_result        = WERR_INVALID_PARAMETER
     431             :                 }
     432             : 
     433             :         };
     434             : 
     435           0 :         for (i=0; i < ARRAY_SIZE(tests); i++) {
     436             : 
     437           0 :                 ZERO_STRUCT(r);
     438             : 
     439           0 :                 r.out.context_handle = &context_handle;
     440             : 
     441           0 :                 r.in.version = tests[i].version;
     442           0 :                 r.in.net_name = tests[i].net_name;
     443           0 :                 r.in.ip_address = tests[i].ip_address;
     444           0 :                 r.in.client_computer_name = tests[i].client_computer_name;
     445             : 
     446           0 :                 torture_assert_ntstatus_equal(tctx,
     447             :                         dcerpc_witness_RegisterEx_r(b, tctx, &r),
     448             :                         tests[i].expected_status,
     449             :                         "RegisterEx failed");
     450             : 
     451           0 :                 torture_assert_werr_equal(tctx,
     452             :                         r.out.result,
     453             :                         tests[i].expected_result,
     454             :                         "RegisterEx failed");
     455             : 
     456           0 :                 if (W_ERROR_IS_OK(r.out.result)) {
     457             : 
     458             :                         /* we have a handle, make sure to unregister it */
     459           0 :                         torture_assert(tctx,
     460             :                                 test_witness_UnRegister_with_handle(tctx, p, r.out.context_handle),
     461             :                                 "Failed to unregister");
     462             :                 }
     463             :         }
     464             : 
     465           0 :         init_witness_test_state(tctx, p, state);
     466             : 
     467           0 :         for (i=0; state->list && i < state->list->num_interfaces; i++) {
     468             : 
     469           0 :                 const char *ip_address;
     470           0 :                 struct witness_interfaceInfo interface = state->list->interfaces[i];
     471             : 
     472           0 :                 if (!check_valid_interface(tctx, &interface)) {
     473           0 :                         continue;
     474             :                 }
     475             : 
     476           0 :                 torture_assert(tctx,
     477             :                         get_ip_address_from_interface(tctx, &interface, &ip_address),
     478             :                         "failed to get ip_address from interface");
     479             : 
     480           0 :                 r.in.version = WITNESS_V2;
     481           0 :                 r.in.net_name = state->net_name;
     482           0 :                 r.in.ip_address = ip_address;
     483             : 
     484             :                 /*
     485             :                  * a valid request with an invalid sharename fails with
     486             :                  * WERR_INVALID_STATE
     487             :                  */
     488           0 :                 r.in.share_name = "any_invalid_share_name";
     489             : 
     490           0 :                 torture_assert_ntstatus_ok(tctx,
     491             :                         dcerpc_witness_RegisterEx_r(b, tctx, &r),
     492             :                         "RegisterEx failed");
     493             : 
     494           0 :                 torture_assert_werr_equal(tctx,
     495             :                         r.out.result,
     496             :                         WERR_INVALID_STATE,
     497             :                         "RegisterEx failed");
     498             : 
     499           0 :                 r.in.share_name = NULL;
     500             : 
     501           0 :                 torture_assert_ntstatus_ok(tctx,
     502             :                         dcerpc_witness_RegisterEx_r(b, tctx, &r),
     503             :                         "RegisterEx failed");
     504             : 
     505           0 :                 torture_assert_werr_ok(tctx,
     506             :                         r.out.result,
     507             :                         "RegisterEx failed");
     508             : 
     509           0 :                 torture_assert(tctx,
     510             :                         test_witness_UnRegister_with_handle(tctx, p, r.out.context_handle),
     511             :                         "Failed to unregister");
     512             :         }
     513             : 
     514           0 :         return true;
     515             : }
     516             : 
     517           0 : static bool setup_clusapi_connection(struct torture_context *tctx,
     518             :                                      struct torture_test_witness_state *s)
     519             : {
     520           0 :         struct dcerpc_binding *binding;
     521             : 
     522           0 :         if (s->clusapi.p) {
     523           0 :                 return true;
     524             :         }
     525             : 
     526           0 :         torture_assert_ntstatus_ok(tctx,
     527             :                 torture_rpc_binding(tctx, &binding),
     528             :                 "failed to retrieve torture binding");
     529             : 
     530           0 :         torture_assert_ntstatus_ok(tctx,
     531             :                 dcerpc_binding_set_transport(binding, NCACN_IP_TCP),
     532             :                 "failed to set transport");
     533             : 
     534           0 :         torture_assert_ntstatus_ok(tctx,
     535             :                 dcerpc_binding_set_flags(binding, DCERPC_SEAL, 0),
     536             :                 "failed to set dcerpc flags");
     537             : 
     538           0 :         torture_assert_ntstatus_ok(tctx,
     539             :                 dcerpc_pipe_connect_b(tctx, &s->clusapi.p, binding,
     540             :                                       &ndr_table_clusapi,
     541             :                                       samba_cmdline_get_creds(),
     542             :                                       tctx->ev, tctx->lp_ctx),
     543             :                 "failed to connect dcerpc pipe");
     544             : 
     545           0 :         return true;
     546             : }
     547             : 
     548             : #if 0
     549             : static bool cluster_get_nodes(struct torture_context *tctx,
     550             :                               struct torture_test_witness_state *s)
     551             : {
     552             :         struct clusapi_CreateEnum r;
     553             :         struct ENUM_LIST *ReturnEnum;
     554             :         WERROR rpc_status;
     555             :         struct dcerpc_binding_handle *b;
     556             : 
     557             :         torture_assert(tctx,
     558             :                 setup_clusapi_connection(tctx, s),
     559             :                 "failed to setup clusapi connection");
     560             : 
     561             :         b = s->clusapi.p->binding_handle;
     562             : 
     563             :         r.in.dwType = CLUSTER_ENUM_NODE;
     564             :         r.out.ReturnEnum = &ReturnEnum;
     565             :         r.out.rpc_status = &rpc_status;
     566             : 
     567             :         torture_assert_ntstatus_ok(tctx,
     568             :                 dcerpc_clusapi_CreateEnum_r(b, tctx, &r),
     569             :                 "failed to enumerate nodes");
     570             : 
     571             :         return true;
     572             : }
     573             : #endif
     574             : 
     575           0 : static bool test_GetResourceState_int(struct torture_context *tctx,
     576             :                                       struct dcerpc_pipe *p,
     577             :                                       struct policy_handle *hResource,
     578             :                                       enum clusapi_ClusterResourceState *State)
     579             : {
     580           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
     581           0 :         struct clusapi_GetResourceState r;
     582           0 :         const char *NodeName;
     583           0 :         const char *GroupName;
     584           0 :         WERROR rpc_status;
     585             : 
     586           0 :         r.in.hResource = *hResource;
     587           0 :         r.out.State = State;
     588           0 :         r.out.NodeName = &NodeName;
     589           0 :         r.out.GroupName = &GroupName;
     590           0 :         r.out.rpc_status = &rpc_status;
     591             : 
     592           0 :         torture_assert_ntstatus_ok(tctx,
     593             :                 dcerpc_clusapi_GetResourceState_r(b, tctx, &r),
     594             :                 "GetResourceState failed");
     595           0 :         torture_assert_werr_ok(tctx,
     596             :                 r.out.result,
     597             :                 "GetResourceState failed");
     598             : 
     599           0 :         return true;
     600             : }
     601             : 
     602           0 : static bool toggle_cluster_resource_state(struct torture_context *tctx,
     603             :                                           struct dcerpc_pipe *p,
     604             :                                           const char *resource_name,
     605             :                                           enum clusapi_ClusterResourceState *old_state,
     606             :                                           enum clusapi_ClusterResourceState *new_state)
     607             : {
     608           0 :         struct policy_handle hResource;
     609           0 :         enum clusapi_ClusterResourceState State;
     610             : 
     611           0 :         torture_assert(tctx,
     612             :                 test_OpenResource_int(tctx, p, resource_name, &hResource),
     613             :                 "failed to open resource");
     614           0 :         torture_assert(tctx,
     615             :                 test_GetResourceState_int(tctx, p, &hResource, &State),
     616             :                 "failed to query resource state");
     617             : 
     618           0 :         if (old_state) {
     619           0 :                 *old_state = State;
     620             :         }
     621             : 
     622           0 :         switch (State) {
     623           0 :         case ClusterResourceOffline:
     624           0 :                 if (!test_OnlineResource_int(tctx, p, &hResource)) {
     625           0 :                         test_CloseResource_int(tctx, p, &hResource);
     626           0 :                         torture_warning(tctx, "failed to set resource online");
     627           0 :                         return false;
     628             :                 }
     629           0 :                 break;
     630           0 :         case ClusterResourceOnline:
     631           0 :                 if (!test_OfflineResource_int(tctx, p, &hResource)) {
     632           0 :                         test_CloseResource_int(tctx, p, &hResource);
     633           0 :                         torture_warning(tctx, "failed to set resource offline");
     634           0 :                         return false;
     635             :                 }
     636           0 :                 break;
     637             : 
     638           0 :         default:
     639           0 :                 break;
     640             :         }
     641             : 
     642           0 :         torture_assert(tctx,
     643             :                 test_GetResourceState_int(tctx, p, &hResource, &State),
     644             :                 "failed to query resource state");
     645             : 
     646           0 :         if (new_state) {
     647           0 :                 *new_state = State;
     648             :         }
     649             : 
     650           0 :         test_CloseResource_int(tctx, p, &hResource);
     651             : 
     652           0 :         return true;
     653             : }
     654             : 
     655           0 : static bool test_witness_AsyncNotify(struct torture_context *tctx,
     656             :                                      struct dcerpc_pipe *p,
     657             :                                      void *data)
     658             : {
     659           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
     660           0 :         struct witness_AsyncNotify r;
     661           0 :         struct witness_notifyResponse *response;
     662           0 :         struct torture_test_witness_state *state =
     663             :                 (struct torture_test_witness_state *)data;
     664           0 :         int i;
     665             : 
     666           0 :         init_witness_test_state(tctx, p, state);
     667             : 
     668           0 :         setup_clusapi_connection(tctx, state);
     669             : 
     670           0 :         for (i=0; state->list && i < state->list->num_interfaces; i++) {
     671             : 
     672           0 :                 const char *ip_address;
     673           0 :                 struct witness_interfaceInfo interface = state->list->interfaces[i];
     674           0 :                 struct witness_Register reg;
     675           0 :                 struct tevent_req *req;
     676           0 :                 enum clusapi_ClusterResourceState old_state, new_state;
     677             : 
     678           0 :                 if (!check_valid_interface(tctx, &interface)) {
     679           0 :                         continue;
     680             :                 }
     681             : 
     682           0 :                 torture_assert(tctx,
     683             :                         get_ip_address_from_interface(tctx, &interface, &ip_address),
     684             :                         "failed to get ip_address from interface");
     685             : 
     686           0 :                 reg.in.version = WITNESS_V1;
     687           0 :                 reg.in.net_name = state->net_name;
     688           0 :                 reg.in.ip_address = ip_address;
     689           0 :                 reg.in.client_computer_name = lpcfg_netbios_name(tctx->lp_ctx);
     690           0 :                 reg.out.context_handle = &state->context_handle;
     691             : 
     692           0 :                 torture_assert_ntstatus_ok(tctx,
     693             :                         dcerpc_witness_Register_r(b, tctx, &reg),
     694             :                         "Register failed");
     695             : 
     696           0 :                 torture_assert_werr_ok(tctx,
     697             :                         reg.out.result,
     698             :                         "Register failed");
     699             : 
     700           0 :                 r.in.context_handle = state->context_handle;
     701           0 :                 r.out.response = &response;
     702             : 
     703           0 :                 req = dcerpc_witness_AsyncNotify_r_send(tctx, tctx->ev, b, &r);
     704           0 :                 torture_assert(tctx, req, "failed to create request");
     705             : 
     706           0 :                 torture_assert(tctx,
     707             :                         toggle_cluster_resource_state(tctx, state->clusapi.p, state->net_name, &old_state, &new_state),
     708             :                         "failed to toggle cluster resource state");
     709           0 :                 torture_assert(tctx, old_state != new_state, "failed to change cluster resource state");
     710             : 
     711           0 :                 torture_assert(tctx,
     712             :                         tevent_req_poll(req, tctx->ev),
     713             :                         "failed to call event loop");
     714             : 
     715           0 :                 torture_assert_ntstatus_ok(tctx,
     716             :                         dcerpc_witness_AsyncNotify_r_recv(req, tctx),
     717             :                         "failed to receive reply");
     718             : 
     719           0 :                 torture_assert_int_equal(tctx, response->num, 1, "num");
     720           0 :                 torture_assert_int_equal(tctx, response->type, WITNESS_NOTIFY_RESOURCE_CHANGE, "type");
     721             : 
     722             :                 /*
     723             :                  * TODO: find out how ClusterResourceOfflinePending and
     724             :                  * ClusterResourceOnlinePending are represented as witness
     725             :                  * types.
     726             :                  */
     727             : 
     728           0 :                 if (new_state == ClusterResourceOffline) {
     729           0 :                         torture_assert_int_equal(tctx, response->messages[0].resource_change.type, WITNESS_RESOURCE_STATE_UNAVAILABLE, "resource_change.type");
     730             :                 }
     731           0 :                 if (new_state == ClusterResourceOnline) {
     732           0 :                         torture_assert_int_equal(tctx, response->messages[0].resource_change.type, WITNESS_RESOURCE_STATE_AVAILABLE, "resource_change.type");
     733             :                 }
     734           0 :                 torture_assert(tctx,
     735             :                         test_witness_UnRegister_with_handle(tctx, p, &state->context_handle),
     736             :                         "Failed to unregister");
     737             : 
     738           0 :                 ZERO_STRUCT(state->context_handle);
     739             : 
     740           0 :                 torture_assert(tctx,
     741             :                         toggle_cluster_resource_state(tctx, state->clusapi.p, state->net_name, &old_state, &new_state),
     742             :                         "failed to toggle cluster resource state");
     743           0 :                 torture_assert(tctx, old_state != new_state, "failed to change cluster resource state");
     744             :         }
     745             : 
     746           0 :         return true;
     747             : }
     748             : 
     749           0 : static bool test_do_witness_RegisterEx(struct torture_context *tctx,
     750             :                                        struct dcerpc_binding_handle *b,
     751             :                                        uint32_t version,
     752             :                                        const char *net_name,
     753             :                                        const char *share_name,
     754             :                                        const char *ip_address,
     755             :                                        const char *client_computer_name,
     756             :                                        uint32_t flags,
     757             :                                        uint32_t timeout,
     758             :                                        struct policy_handle *context_handle)
     759             : {
     760           0 :         struct witness_RegisterEx r;
     761             : 
     762           0 :         r.in.version = version;
     763           0 :         r.in.net_name = net_name;
     764           0 :         r.in.share_name = NULL;
     765           0 :         r.in.ip_address = ip_address;
     766           0 :         r.in.client_computer_name = client_computer_name;
     767           0 :         r.in.flags = flags;
     768           0 :         r.in.timeout = timeout;
     769           0 :         r.out.context_handle = context_handle;
     770             : 
     771           0 :         torture_assert_ntstatus_ok(tctx,
     772             :                 dcerpc_witness_RegisterEx_r(b, tctx, &r),
     773             :                 "RegisterEx failed");
     774             : 
     775           0 :         torture_assert_werr_ok(tctx,
     776             :                 r.out.result,
     777             :                 "RegisterEx failed");
     778             : 
     779           0 :         return true;
     780             : }
     781             : 
     782           0 : static void torture_subunit_report_time(struct torture_context *tctx)
     783             : {
     784           0 :         struct timespec tp;
     785           0 :         struct tm *tmp;
     786           0 :         char timestr[200];
     787             : 
     788           0 :         if (clock_gettime(CLOCK_REALTIME, &tp) != 0) {
     789           0 :                 torture_comment(tctx, "failed to call clock_gettime");
     790           0 :                 return;
     791             :         }
     792             : 
     793           0 :         tmp = gmtime(&tp.tv_sec);
     794           0 :         if (!tmp) {
     795           0 :                 torture_comment(tctx, "failed to call gmtime");
     796           0 :                 return;
     797             :         }
     798             : 
     799           0 :         if (strftime(timestr, sizeof(timestr), "%Y-%m-%d %H:%M:%S", tmp) <= 0) {
     800           0 :                 torture_comment(tctx, "failed to call strftime");
     801           0 :                 return;
     802             :         }
     803             : 
     804           0 :         torture_comment(tctx, "time: %s.%06ld\n", timestr, tp.tv_nsec / 1000);
     805             : }
     806             : 
     807           0 : static bool test_witness_AsyncNotify_timeouts(struct torture_context *tctx,
     808             :                                               struct dcerpc_pipe *p,
     809             :                                               void *data)
     810             : {
     811           0 :         struct dcerpc_binding_handle *b = p->binding_handle;
     812           0 :         struct witness_AsyncNotify r;
     813           0 :         struct witness_notifyResponse *response;
     814           0 :         struct torture_test_witness_state *state =
     815             :                 (struct torture_test_witness_state *)data;
     816           0 :         int i;
     817             : 
     818           0 :         init_witness_test_state(tctx, p, state);
     819             : 
     820           0 :         setup_clusapi_connection(tctx, state);
     821             : 
     822           0 :         for (i=0; state->list && i < state->list->num_interfaces; i++) {
     823             : 
     824           0 :                 const char *ip_address;
     825           0 :                 struct witness_interfaceInfo interface = state->list->interfaces[i];
     826           0 :                 uint32_t timeouts[] = {
     827             :                         0, 1, 10, 100, 120
     828             :                 };
     829           0 :                 int t;
     830           0 :                 uint32_t old_timeout;
     831             : 
     832           0 :                 if (!check_valid_interface(tctx, &interface)) {
     833           0 :                         continue;
     834             :                 }
     835             : 
     836           0 :                 torture_assert(tctx,
     837             :                         get_ip_address_from_interface(tctx, &interface, &ip_address),
     838             :                         "failed to get ip_address from interface");
     839             : 
     840           0 :                 for (t=0; t < ARRAY_SIZE(timeouts); t++) {
     841             : 
     842           0 :                         torture_comment(tctx, "Testing Async Notify with timeout of %d milliseconds", timeouts[t]);
     843             : 
     844           0 :                         torture_assert(tctx,
     845             :                                 test_do_witness_RegisterEx(tctx, b,
     846             :                                                            WITNESS_V2,
     847             :                                                            state->net_name,
     848             :                                                            NULL,
     849             :                                                            ip_address,
     850             :                                                            lpcfg_netbios_name(tctx->lp_ctx),
     851             :                                                            0,
     852             :                                                            timeouts[t],
     853             :                                                            &state->context_handle),
     854             :                                 "failed to RegisterEx");
     855             : 
     856           0 :                         r.in.context_handle = state->context_handle;
     857           0 :                         r.out.response = &response;
     858             : 
     859           0 :                         old_timeout = dcerpc_binding_handle_set_timeout(b, UINT_MAX);
     860             : 
     861           0 :                         torture_subunit_report_time(tctx);
     862             : 
     863           0 :                         torture_assert_ntstatus_ok(tctx,
     864             :                                 dcerpc_witness_AsyncNotify_r(b, tctx, &r),
     865             :                                 "AsyncNotify failed");
     866           0 :                         torture_assert_werr_equal(tctx,
     867             :                                 r.out.result,
     868             :                                 WERR_TIMEOUT,
     869             :                                 "AsyncNotify failed");
     870             : 
     871           0 :                         torture_subunit_report_time(tctx);
     872             : 
     873           0 :                         dcerpc_binding_handle_set_timeout(b, old_timeout);
     874             : 
     875           0 :                         torture_assert(tctx,
     876             :                                 test_witness_UnRegister_with_handle(tctx, p, &state->context_handle),
     877             :                                 "Failed to unregister");
     878             : 
     879           0 :                         ZERO_STRUCT(state->context_handle);
     880             :                 }
     881             :         }
     882             : 
     883           0 :         return true;
     884             : }
     885             : 
     886        2358 : struct torture_suite *torture_rpc_witness(TALLOC_CTX *mem_ctx)
     887             : {
     888         125 :         struct torture_rpc_tcase *tcase;
     889        2358 :         struct torture_suite *suite = torture_suite_create(mem_ctx, "witness");
     890         125 :         struct torture_test_witness_state *state;
     891             : 
     892        2358 :         tcase = torture_suite_add_rpc_iface_tcase(suite, "witness",
     893             :                                                   &ndr_table_witness);
     894             : 
     895        2358 :         state = talloc_zero(tcase, struct torture_test_witness_state);
     896             : 
     897        2358 :         torture_rpc_tcase_add_test_ex(tcase, "GetInterfaceList",
     898             :                                       test_witness_GetInterfaceList, state);
     899        2358 :         torture_rpc_tcase_add_test_ex(tcase, "Register",
     900             :                                       test_witness_Register, state);
     901        2358 :         torture_rpc_tcase_add_test_ex(tcase, "UnRegister",
     902             :                                       test_witness_UnRegister, state);
     903        2358 :         torture_rpc_tcase_add_test_ex(tcase, "RegisterEx",
     904             :                                       test_witness_RegisterEx, state);
     905        2358 :         torture_rpc_tcase_add_test_ex(tcase, "AsyncNotify",
     906             :                                       test_witness_AsyncNotify, state);
     907        2358 :         torture_rpc_tcase_add_test_ex(tcase, "AsyncNotify_timeouts",
     908             :                                       test_witness_AsyncNotify_timeouts, state);
     909             : 
     910        2358 :         return suite;
     911             : }

Generated by: LCOV version 1.14