Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 : RPC pipe client
4 :
5 : Copyright (C) Tim Potter 2000
6 : Copyright (C) Jelmer Vernooij 2005.
7 :
8 : This program is free software; you can redistribute it and/or modify
9 : it under the terms of the GNU General Public License as published by
10 : the Free Software Foundation; either version 3 of the License, or
11 : (at your option) any later version.
12 :
13 : This program is distributed in the hope that it will be useful,
14 : but WITHOUT ANY WARRANTY; without even the implied warranty of
15 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 : GNU General Public License for more details.
17 :
18 : You should have received a copy of the GNU General Public License
19 : along with this program. If not, see <http://www.gnu.org/licenses/>.
20 : */
21 :
22 : #include "includes.h"
23 : #include "rpcclient.h"
24 : #include "../librpc/gen_ndr/ndr_dfs_c.h"
25 :
26 : /* Check DFS is supported by the remote server */
27 :
28 0 : static WERROR cmd_dfs_version(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
29 : int argc, const char **argv)
30 : {
31 : enum dfs_ManagerVersion version;
32 : NTSTATUS result;
33 0 : struct dcerpc_binding_handle *b = cli->binding_handle;
34 :
35 0 : if (argc != 1) {
36 0 : printf("Usage: %s\n", argv[0]);
37 0 : return WERR_OK;
38 : }
39 :
40 0 : result = dcerpc_dfs_GetManagerVersion(b, mem_ctx, &version);
41 :
42 0 : if (!NT_STATUS_IS_OK(result)) {
43 0 : return ntstatus_to_werror(result);
44 : }
45 :
46 0 : if (version > 0) {
47 0 : printf("dfs is present (%d)\n", version);
48 : } else {
49 0 : printf("dfs is not present\n");
50 : }
51 :
52 0 : return WERR_OK;
53 : }
54 :
55 0 : static WERROR cmd_dfs_add(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
56 : int argc, const char **argv)
57 : {
58 : NTSTATUS result;
59 : WERROR werr;
60 : const char *path, *servername, *sharename, *comment;
61 0 : uint32_t flags = 0;
62 0 : struct dcerpc_binding_handle *b = cli->binding_handle;
63 :
64 0 : if (argc != 5) {
65 0 : printf("Usage: %s path servername sharename comment\n",
66 : argv[0]);
67 0 : return WERR_OK;
68 : }
69 :
70 0 : path = argv[1];
71 0 : servername = argv[2];
72 0 : sharename = argv[3];
73 0 : comment = argv[4];
74 :
75 0 : result = dcerpc_dfs_Add(b, mem_ctx, path, servername,
76 : sharename, comment, flags, &werr);
77 0 : if (!NT_STATUS_IS_OK(result)) {
78 0 : return ntstatus_to_werror(result);
79 : }
80 :
81 0 : return werr;
82 : }
83 :
84 0 : static WERROR cmd_dfs_remove(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
85 : int argc, const char **argv)
86 : {
87 : NTSTATUS result;
88 : WERROR werr;
89 : const char *path, *servername, *sharename;
90 0 : struct dcerpc_binding_handle *b = cli->binding_handle;
91 :
92 0 : if (argc != 4) {
93 0 : printf("Usage: %s path servername sharename\n", argv[0]);
94 0 : return WERR_OK;
95 : }
96 :
97 0 : path = argv[1];
98 0 : servername = argv[2];
99 0 : sharename = argv[3];
100 :
101 0 : result = dcerpc_dfs_Remove(b, mem_ctx, path, servername,
102 : sharename, &werr);
103 0 : if (!NT_STATUS_IS_OK(result)) {
104 0 : return ntstatus_to_werror(result);
105 : }
106 :
107 0 : return werr;
108 : }
109 :
110 : /* Display a DFS_INFO_1 structure */
111 :
112 0 : static void display_dfs_info_1(struct dfs_Info1 *info1)
113 : {
114 0 : printf("path: %s\n", info1->path);
115 0 : }
116 :
117 : /* Display a DFS_INFO_2 structure */
118 :
119 0 : static void display_dfs_info_2(struct dfs_Info2 *info2)
120 : {
121 0 : printf("path: %s\n", info2->path);
122 0 : printf("\tcomment: %s\n", info2->comment);
123 :
124 0 : printf("\tstate: %d\n", info2->state);
125 0 : printf("\tnum_stores: %d\n", info2->num_stores);
126 0 : }
127 :
128 : /* Display a DFS_INFO_3 structure */
129 :
130 0 : static void display_dfs_info_3(struct dfs_Info3 *info3)
131 : {
132 : int i;
133 :
134 0 : printf("path: %s\n", info3->path);
135 :
136 0 : printf("\tcomment: %s\n", info3->comment);
137 :
138 0 : printf("\tstate: %d\n", info3->state);
139 0 : printf("\tnum_stores: %d\n", info3->num_stores);
140 :
141 0 : for (i = 0; i < info3->num_stores; i++) {
142 0 : struct dfs_StorageInfo *dsi = &info3->stores[i];
143 :
144 0 : printf("\t\tstorage[%d] server: %s\n", i, dsi->server);
145 :
146 0 : printf("\t\tstorage[%d] share: %s\n", i, dsi->share);
147 : }
148 0 : }
149 :
150 :
151 : /* Display a DFS_INFO_CTR structure */
152 0 : static void display_dfs_info(uint32_t level, union dfs_Info *ctr)
153 : {
154 0 : switch (level) {
155 0 : case 0x01:
156 0 : display_dfs_info_1(ctr->info1);
157 0 : break;
158 0 : case 0x02:
159 0 : display_dfs_info_2(ctr->info2);
160 0 : break;
161 0 : case 0x03:
162 0 : display_dfs_info_3(ctr->info3);
163 0 : break;
164 0 : default:
165 0 : printf("unsupported info level %d\n",
166 : level);
167 0 : break;
168 : }
169 0 : }
170 :
171 0 : static void display_dfs_enumstruct(struct dfs_EnumStruct *ctr)
172 : {
173 : int i;
174 :
175 : /* count is always the first element, so we can just use info1 here */
176 0 : for (i = 0; i < ctr->e.info1->count; i++) {
177 0 : switch (ctr->level) {
178 0 : case 1: display_dfs_info_1(&ctr->e.info1->s[i]); break;
179 0 : case 2: display_dfs_info_2(&ctr->e.info2->s[i]); break;
180 0 : case 3: display_dfs_info_3(&ctr->e.info3->s[i]); break;
181 0 : default:
182 0 : printf("unsupported info level %d\n",
183 : ctr->level);
184 0 : return;
185 : }
186 : }
187 : }
188 :
189 : /* Enumerate dfs shares */
190 :
191 0 : static WERROR cmd_dfs_enum(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
192 : int argc, const char **argv)
193 : {
194 : struct dfs_EnumStruct str;
195 : struct dfs_EnumArray1 info1;
196 : struct dfs_EnumArray2 info2;
197 : struct dfs_EnumArray3 info3;
198 : struct dfs_EnumArray4 info4;
199 : struct dfs_EnumArray200 info200;
200 : struct dfs_EnumArray300 info300;
201 0 : struct dcerpc_binding_handle *b = cli->binding_handle;
202 :
203 : NTSTATUS result;
204 : WERROR werr;
205 0 : uint32_t total = 0;
206 :
207 0 : if (argc > 2) {
208 0 : printf("Usage: %s [info_level]\n", argv[0]);
209 0 : return WERR_OK;
210 : }
211 :
212 0 : str.level = 1;
213 0 : if (argc == 2)
214 0 : str.level = atoi(argv[1]);
215 :
216 0 : switch (str.level) {
217 0 : case 1: str.e.info1 = &info1; ZERO_STRUCT(info1); break;
218 0 : case 2: str.e.info2 = &info2; ZERO_STRUCT(info2); break;
219 0 : case 3: str.e.info3 = &info3; ZERO_STRUCT(info3); break;
220 0 : case 4: str.e.info4 = &info4; ZERO_STRUCT(info4); break;
221 0 : case 200: str.e.info200 = &info200; ZERO_STRUCT(info200); break;
222 0 : case 300: str.e.info300 = &info300; ZERO_STRUCT(info300); break;
223 0 : default:
224 0 : printf("Unknown info level %d\n", str.level);
225 0 : return WERR_OK;
226 : }
227 :
228 0 : result = dcerpc_dfs_Enum(b, mem_ctx, str.level, 0xFFFFFFFF, &str,
229 : &total, &werr);
230 0 : if (!NT_STATUS_IS_OK(result)) {
231 0 : return ntstatus_to_werror(result);
232 : }
233 0 : if (W_ERROR_IS_OK(werr)) {
234 0 : display_dfs_enumstruct(&str);
235 : }
236 :
237 0 : return werr;
238 : }
239 :
240 : /* Enumerate dfs shares */
241 :
242 0 : static WERROR cmd_dfs_enumex(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
243 : int argc, const char **argv)
244 : {
245 : struct dfs_EnumStruct str;
246 : struct dfs_EnumArray1 info1;
247 : struct dfs_EnumArray2 info2;
248 : struct dfs_EnumArray3 info3;
249 : struct dfs_EnumArray4 info4;
250 : struct dfs_EnumArray200 info200;
251 : struct dfs_EnumArray300 info300;
252 0 : struct dcerpc_binding_handle *b = cli->binding_handle;
253 :
254 : NTSTATUS result;
255 : WERROR werr;
256 0 : uint32_t total = 0;
257 :
258 0 : if (argc < 2 || argc > 3) {
259 0 : printf("Usage: %s dfs_name [info_level]\n", argv[0]);
260 0 : return WERR_OK;
261 : }
262 :
263 0 : str.level = 1;
264 :
265 0 : if (argc == 3)
266 0 : str.level = atoi(argv[2]);
267 :
268 0 : switch (str.level) {
269 0 : case 1: str.e.info1 = &info1; ZERO_STRUCT(info1); break;
270 0 : case 2: str.e.info2 = &info2; ZERO_STRUCT(info2); break;
271 0 : case 3: str.e.info3 = &info3; ZERO_STRUCT(info3); break;
272 0 : case 4: str.e.info4 = &info4; ZERO_STRUCT(info4); break;
273 0 : case 200: str.e.info200 = &info200; ZERO_STRUCT(info200); break;
274 0 : case 300: str.e.info300 = &info300; ZERO_STRUCT(info300); break;
275 0 : default:
276 0 : printf("Unknown info level %d\n", str.level);
277 0 : return WERR_OK;
278 : }
279 :
280 0 : result = dcerpc_dfs_EnumEx(b, mem_ctx, argv[1], str.level,
281 : 0xFFFFFFFF, &str, &total, &werr);
282 0 : if (!NT_STATUS_IS_OK(result)) {
283 0 : return ntstatus_to_werror(result);
284 : }
285 0 : if (W_ERROR_IS_OK(werr)) {
286 0 : display_dfs_enumstruct(&str);
287 : }
288 :
289 0 : return werr;
290 : }
291 :
292 :
293 0 : static WERROR cmd_dfs_getinfo(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
294 : int argc, const char **argv)
295 : {
296 : NTSTATUS result;
297 : WERROR werr;
298 : const char *path, *servername, *sharename;
299 0 : uint32_t info_level = 1;
300 : union dfs_Info ctr;
301 0 : struct dcerpc_binding_handle *b = cli->binding_handle;
302 :
303 0 : if (argc < 4 || argc > 5) {
304 0 : printf("Usage: %s path servername sharename "
305 : "[info_level]\n", argv[0]);
306 0 : return WERR_OK;
307 : }
308 :
309 0 : path = argv[1];
310 0 : servername = argv[2];
311 0 : sharename = argv[3];
312 :
313 0 : if (argc == 5)
314 0 : info_level = atoi(argv[4]);
315 :
316 0 : result = dcerpc_dfs_GetInfo(b, mem_ctx, path, servername,
317 : sharename, info_level, &ctr, &werr);
318 0 : if (!NT_STATUS_IS_OK(result)) {
319 0 : return ntstatus_to_werror(result);
320 : }
321 0 : if (W_ERROR_IS_OK(werr)) {
322 0 : display_dfs_info(info_level, &ctr);
323 : }
324 :
325 0 : return werr;
326 : }
327 :
328 : /* List of commands exported by this module */
329 :
330 : struct cmd_set dfs_commands[] = {
331 :
332 : { .name = "DFS" },
333 :
334 : {
335 : .name = "dfsversion",
336 : .returntype = RPC_RTYPE_WERROR,
337 : .ntfn = NULL,
338 : .wfn = cmd_dfs_version,
339 : .table = &ndr_table_netdfs,
340 : .rpc_pipe = NULL,
341 : .description = "Query DFS support",
342 : .usage = "",
343 : },
344 : {
345 : .name = "dfsadd",
346 : .returntype = RPC_RTYPE_WERROR,
347 : .ntfn = NULL,
348 : .wfn = cmd_dfs_add,
349 : .table = &ndr_table_netdfs,
350 : .rpc_pipe = NULL,
351 : .description = "Add a DFS share",
352 : .usage = "",
353 : },
354 : {
355 : .name = "dfsremove",
356 : .returntype = RPC_RTYPE_WERROR,
357 : .ntfn = NULL,
358 : .wfn = cmd_dfs_remove,
359 : .table = &ndr_table_netdfs,
360 : .rpc_pipe = NULL,
361 : .description = "Remove a DFS share",
362 : .usage = "",
363 : },
364 : {
365 : .name = "dfsgetinfo",
366 : .returntype = RPC_RTYPE_WERROR,
367 : .ntfn = NULL,
368 : .wfn = cmd_dfs_getinfo,
369 : .table = &ndr_table_netdfs,
370 : .rpc_pipe = NULL,
371 : .description = "Query DFS share info",
372 : .usage = "",
373 : },
374 : {
375 : .name = "dfsenum",
376 : .returntype = RPC_RTYPE_WERROR,
377 : .ntfn = NULL,
378 : .wfn = cmd_dfs_enum,
379 : .table = &ndr_table_netdfs,
380 : .rpc_pipe = NULL,
381 : .description = "Enumerate dfs shares",
382 : .usage = "",
383 : },
384 : {
385 : .name = "dfsenumex",
386 : .returntype = RPC_RTYPE_WERROR,
387 : .ntfn = NULL,
388 : .wfn = cmd_dfs_enumex,
389 : .table = &ndr_table_netdfs,
390 : .rpc_pipe = NULL,
391 : .description = "Enumerate dfs shares",
392 : .usage = "",
393 : },
394 :
395 : { .name = NULL }
396 : };
|