/* Unix SMB/CIFS implementation. Test suite for libnet calls. Copyright (C) Gregory LEOCADIE 2005 Copyright (C) Rafal Szczesniak 2005 This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "includes.h" #include "torture/torture.h" #include "libnet/libnet.h" #include "lib/cmdline/popt_common.h" #define TEST_SHARENAME "libnetsharetest" static void test_displayshares(struct libnet_ListShares s) { int i, j; struct share_type { enum srvsvc_ShareType type; const char *desc; } share_types[] = { { STYPE_DISKTREE, "STYPE_DISKTREE" }, { STYPE_DISKTREE_TEMPORARY, "STYPE_DISKTREE_TEMPORARY" }, { STYPE_DISKTREE_HIDDEN, "STYPE_DISKTREE_HIDDEN" }, { STYPE_PRINTQ, "STYPE_PRINTQ" }, { STYPE_PRINTQ_TEMPORARY, "STYPE_PRINTQ_TEMPORARY" }, { STYPE_PRINTQ_HIDDEN, "STYPE_PRINTQ_HIDDEN" }, { STYPE_DEVICE, "STYPE_DEVICE" }, { STYPE_DEVICE_TEMPORARY, "STYPE_DEVICE_TEMPORARY" }, { STYPE_DEVICE_HIDDEN, "STYPE_DEVICE_HIDDEN" }, { STYPE_IPC, "STYPE_IPC" }, { STYPE_IPC_TEMPORARY, "STYPE_IPC_TEMPORARY" }, { STYPE_IPC_HIDDEN, "STYPE_IPC_HIDDEN" } }; switch (s.in.level) { case 0: for (i = 0; i < s.out.ctr.ctr0->count; i++) { printf("\t[%d] %s\n", i, s.out.ctr.ctr0->array[i].name); } break; case 1: for (i = 0; i < s.out.ctr.ctr1->count; i++) { for (j = 0; j < ARRAY_SIZE(share_types); j++) { if (share_types[j].type == s.out.ctr.ctr1->array[i].type) break; } printf("\t[%d] %s (%s)\t%s\n", i, s.out.ctr.ctr1->array[i].name, s.out.ctr.ctr1->array[i].comment, share_types[j].desc); } break; case 2: for (i = 0; i < s.out.ctr.ctr2->count; i++) { for (j = 0; j < ARRAY_SIZE(share_types); j++) { if (share_types[j].type == s.out.ctr.ctr2->array[i].type) break; } printf("\t[%d] %s\t%s\n\t %s\n\t [perms=0x%08x, max_usr=%d, cur_usr=%d, path=%s, pass=%s]\n", i, s.out.ctr.ctr2->array[i].name, share_types[j].desc, s.out.ctr.ctr2->array[i].comment, s.out.ctr.ctr2->array[i].permissions, s.out.ctr.ctr2->array[i].max_users, s.out.ctr.ctr2->array[i].current_users, s.out.ctr.ctr2->array[i].path, s.out.ctr.ctr2->array[i].password); } break; case 501: for (i = 0; i < s.out.ctr.ctr501->count; i++) { for (j = 0; j < ARRAY_SIZE(share_types); j++) { if (share_types[j].type == s.out.ctr.ctr501->array[i].type) break; } printf("\t[%d] %s\t%s [csc_policy=0x%08x]\n\t %s\n", i, s.out.ctr.ctr501->array[i].name, share_types[j].desc, s.out.ctr.ctr501->array[i].csc_policy, s.out.ctr.ctr501->array[i].comment); } break; case 502: for (i = 0; i < s.out.ctr.ctr502->count; i++) { for (j = 0; j < ARRAY_SIZE(share_types); j++) { if (share_types[j].type == s.out.ctr.ctr502->array[i].type) break; } printf("\t[%d] %s\t%s\n\t %s\n\t [perms=0x%08x, max_usr=%d, cur_usr=%d, path=%s, pass=%s, unknown=0x%08x]\n", i, s.out.ctr.ctr502->array[i].name, share_types[j].desc, s.out.ctr.ctr502->array[i].comment, s.out.ctr.ctr502->array[i].permissions, s.out.ctr.ctr502->array[i].max_users, s.out.ctr.ctr502->array[i].current_users, s.out.ctr.ctr502->array[i].path, s.out.ctr.ctr502->array[i].password, s.out.ctr.ctr502->array[i].unknown); } break; } } BOOL torture_listshares(void) { struct libnet_ListShares share; NTSTATUS status; uint32_t levels[] = { 0, 1, 2, 501, 502 }; int i; BOOL ret = True; struct libnet_context* libnetctx; const char *binding; struct dcerpc_binding *bind; TALLOC_CTX *mem_ctx; mem_ctx = talloc_init("test_listshares"); binding = lp_parm_string(-1, "torture", "binding"); status = dcerpc_parse_binding(mem_ctx, binding, &bind); if (!NT_STATUS_IS_OK(status)) { printf("Error while parsing the binding string\n"); ret = False; goto done; } libnetctx = libnet_context_init(NULL); if (!libnetctx) { printf("Couldn't allocate libnet context\n"); ret = False; goto done; } libnetctx->cred = cmdline_credentials; printf("Testing libnet_ListShare\n"); share.in.server_name = talloc_asprintf(mem_ctx, "%s", bind->host); for (i = 0; i < ARRAY_SIZE(levels); i++) { share.in.level = levels[i]; printf("testing libnet_ListShare level %u\n", share.in.level); status = libnet_ListShares(libnetctx, mem_ctx, &share); if (!NT_STATUS_IS_OK(status)) { printf("libnet_ListShare level %u failed - %s\n", share.in.level, share.out.error_string); ret = False; goto done; } printf("listing shares:\n"); test_displayshares(share); } done: talloc_free(mem_ctx); return ret; } static BOOL test_addshare(struct dcerpc_pipe *pipe, TALLOC_CTX *mem_ctx, const char *host, const char* share) { NTSTATUS status; struct srvsvc_NetShareAdd add; struct srvsvc_NetShareInfo2 i; i.name = share; i.type = STYPE_DISKTREE; i.path = "C:\\WINDOWS\\TEMP"; i.max_users = 5; i.comment = "Comment to the test share"; i.password = NULL; i.permissions = 0x0; add.in.server_unc = host; add.in.level = 2; add.in.info.info2 = &i; status = dcerpc_srvsvc_NetShareAdd(pipe, mem_ctx, &add); if (!NT_STATUS_IS_OK(status)) { printf("Failed to add a new share\n"); return False; } printf("share added\n"); return True; } BOOL torture_delshare(void) { struct dcerpc_pipe *p; struct dcerpc_binding *bind; struct libnet_context* libnetctx; const char *host, *binding; TALLOC_CTX *mem_ctx; NTSTATUS status; BOOL ret = True; struct libnet_DelShare share; mem_ctx = talloc_init("test_listshares"); host = lp_parm_string(-1, "torture", "host"); binding = lp_parm_string(-1, "torture", "binding"); status = dcerpc_parse_binding(mem_ctx, binding, &bind); if (!NT_STATUS_IS_OK(status)) { printf("Error while parsing the binding string\n"); ret = False; goto done; } libnetctx = libnet_context_init(NULL); libnetctx->cred = cmdline_credentials; status = torture_rpc_connection(mem_ctx, &p, &dcerpc_table_srvsvc); if (!test_addshare(p, mem_ctx, host, TEST_SHARENAME)) { ret = False; goto done; } share.in.server_name = bind->host; share.in.share_name = TEST_SHARENAME; status = libnet_DelShare(libnetctx, mem_ctx, &share); if (!NT_STATUS_IS_OK(status)) { ret = False; goto done; } done: talloc_free(mem_ctx); return ret; }