s3: Fix Coverity ID 242724 Uninitialized scalar variable
[Samba/gebeck_regimport.git] / source3 / rpcclient / cmd_spoolss.c
blob418ad3bcfbb0444d6a0f5994b7e245d1107705ae
1 /*
2 Unix SMB/CIFS implementation.
3 RPC pipe client
5 Copyright (C) Gerald Carter 2001-2005
6 Copyright (C) Tim Potter 2000
7 Copyright (C) Andrew Tridgell 1992-1999
8 Copyright (C) Luke Kenneth Casson Leighton 1996-1999
9 Copyright (C) Guenther Deschner 2009
11 This program is free software; you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation; either version 3 of the License, or
14 (at your option) any later version.
16 This program is distributed in the hope that it will be useful,
17 but WITHOUT ANY WARRANTY; without even the implied warranty of
18 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 GNU General Public License for more details.
21 You should have received a copy of the GNU General Public License
22 along with this program. If not, see <http://www.gnu.org/licenses/>.
25 #include "includes.h"
26 #include "rpcclient.h"
27 #include "../librpc/gen_ndr/ndr_spoolss_c.h"
28 #include "rpc_client/cli_spoolss.h"
29 #include "rpc_client/init_spoolss.h"
30 #include "nt_printing.h"
31 #include "../libcli/security/display_sec.h"
32 #include "../libcli/security/security_descriptor.h"
33 #include "../libcli/registry/util_reg.h"
34 #include "libsmb/libsmb.h"
36 #define RPCCLIENT_PRINTERNAME(_printername, _cli, _arg) \
37 { \
38 _printername = talloc_asprintf_strupper_m(mem_ctx, "%s\\%s", \
39 _cli->srv_name_slash, _arg); \
40 W_ERROR_HAVE_NO_MEMORY(_printername); \
43 /* The version int is used by getdrivers. Note that
44 all architecture strings that support mutliple
45 versions must be grouped together since enumdrivers
46 uses this property to prevent issuing multiple
47 enumdriver calls for the same arch */
50 static const struct print_architecture_table_node archi_table[]= {
52 {"Windows 4.0", "WIN40", 0 },
53 {"Windows NT x86", "W32X86", 2 },
54 {"Windows NT x86", "W32X86", 3 },
55 {"Windows NT R4000", "W32MIPS", 2 },
56 {"Windows NT Alpha_AXP", "W32ALPHA", 2 },
57 {"Windows NT PowerPC", "W32PPC", 2 },
58 {"Windows IA64", "IA64", 3 },
59 {"Windows x64", "x64", 3 },
60 {NULL, "", -1 }
63 /**
64 * @file
66 * rpcclient module for SPOOLSS rpc pipe.
68 * This generally just parses and checks command lines, and then calls
69 * a cli_spoolss function.
70 **/
72 /****************************************************************************
73 function to do the mapping between the long architecture name and
74 the short one.
75 ****************************************************************************/
77 static const char *cmd_spoolss_get_short_archi(const char *long_archi)
79 int i=-1;
81 DEBUG(107,("Getting architecture dependent directory\n"));
82 do {
83 i++;
84 } while ( (archi_table[i].long_archi!=NULL ) &&
85 strcasecmp_m(long_archi, archi_table[i].long_archi) );
87 if (archi_table[i].long_archi==NULL) {
88 DEBUGADD(10,("Unknown architecture [%s] !\n", long_archi));
89 return NULL;
92 /* this might be client code - but shouldn't this be an fstrcpy etc? */
95 DEBUGADD(108,("index: [%d]\n", i));
96 DEBUGADD(108,("long architecture: [%s]\n", archi_table[i].long_archi));
97 DEBUGADD(108,("short architecture: [%s]\n", archi_table[i].short_archi));
99 return archi_table[i].short_archi;
102 /****************************************************************************
103 ****************************************************************************/
105 static WERROR cmd_spoolss_open_printer_ex(struct rpc_pipe_client *cli,
106 TALLOC_CTX *mem_ctx,
107 int argc, const char **argv)
109 WERROR werror;
110 struct policy_handle hnd;
111 uint32_t access_mask = PRINTER_ALL_ACCESS;
112 struct dcerpc_binding_handle *b = cli->binding_handle;
114 if (argc < 2) {
115 printf("Usage: %s <printername> [access_mask]\n", argv[0]);
116 return WERR_OK;
119 if (argc >= 3) {
120 sscanf(argv[2], "%x", &access_mask);
123 /* Open the printer handle */
125 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
126 argv[1],
127 access_mask,
128 &hnd);
129 if (W_ERROR_IS_OK(werror)) {
130 printf("Printer %s opened successfully\n", argv[1]);
131 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &werror);
133 if (!W_ERROR_IS_OK(werror)) {
134 printf("Error closing printer handle! (%s)\n",
135 get_dos_error_msg(werror));
139 return werror;
142 /****************************************************************************
143 ****************************************************************************/
145 static WERROR cmd_spoolss_open_printer(struct rpc_pipe_client *cli,
146 TALLOC_CTX *mem_ctx,
147 int argc, const char **argv)
149 WERROR werror;
150 struct policy_handle hnd;
151 uint32_t access_mask = PRINTER_ALL_ACCESS;
152 NTSTATUS status;
153 struct spoolss_DevmodeContainer devmode_ctr;
154 struct dcerpc_binding_handle *b = cli->binding_handle;
156 ZERO_STRUCT(devmode_ctr);
158 if (argc < 2) {
159 printf("Usage: %s <printername> [access_mask]\n", argv[0]);
160 return WERR_OK;
163 if (argc >= 3) {
164 sscanf(argv[2], "%x", &access_mask);
167 /* Open the printer handle */
169 status = dcerpc_spoolss_OpenPrinter(b, mem_ctx,
170 argv[1],
171 NULL,
172 devmode_ctr,
173 access_mask,
174 &hnd,
175 &werror);
176 if (!NT_STATUS_IS_OK(status)) {
177 return ntstatus_to_werror(status);
179 if (W_ERROR_IS_OK(werror)) {
180 printf("Printer %s opened successfully\n", argv[1]);
181 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &werror);
183 if (!W_ERROR_IS_OK(werror)) {
184 printf("Error closing printer handle! (%s)\n",
185 get_dos_error_msg(werror));
189 return werror;
192 /****************************************************************************
193 ****************************************************************************/
195 static void display_print_info0(struct spoolss_PrinterInfo0 *r)
197 if (!r)
198 return;
200 printf("\tprintername:[%s]\n", r->printername);
201 printf("\tservername:[%s]\n", r->servername);
202 printf("\tcjobs:[0x%x]\n", r->cjobs);
203 printf("\ttotal_jobs:[0x%x]\n", r->total_jobs);
204 printf("\ttotal_bytes:[0x%x]\n", r->total_bytes);
205 printf("\t:date: [%d]-[%d]-[%d] (%d)\n", r->time.year, r->time.month,
206 r->time.day, r->time.day_of_week);
207 printf("\t:time: [%d]-[%d]-[%d]-[%d]\n", r->time.hour, r->time.minute,
208 r->time.second, r->time.millisecond);
210 printf("\tglobal_counter:[0x%x]\n", r->global_counter);
211 printf("\ttotal_pages:[0x%x]\n", r->total_pages);
213 printf("\tversion:[0x%x]\n", r->version);
214 printf("\tfree_build:[0x%x]\n", r->free_build);
215 printf("\tspooling:[0x%x]\n", r->spooling);
216 printf("\tmax_spooling:[0x%x]\n", r->max_spooling);
217 printf("\tsession_counter:[0x%x]\n", r->session_counter);
218 printf("\tnum_error_out_of_paper:[0x%x]\n", r->num_error_out_of_paper);
219 printf("\tnum_error_not_ready:[0x%x]\n", r->num_error_not_ready);
220 printf("\tjob_error:[0x%x]\n", r->job_error);
221 printf("\tnumber_of_processors:[0x%x]\n", r->number_of_processors);
222 printf("\tprocessor_type:[0x%x]\n", r->processor_type);
223 printf("\thigh_part_total_bytes:[0x%x]\n", r->high_part_total_bytes);
224 printf("\tchange_id:[0x%x]\n", r->change_id);
225 printf("\tlast_error: %s\n", win_errstr(r->last_error));
226 printf("\tstatus:[0x%x]\n", r->status);
227 printf("\tenumerate_network_printers:[0x%x]\n", r->enumerate_network_printers);
228 printf("\tc_setprinter:[0x%x]\n", r->c_setprinter);
229 printf("\tprocessor_architecture:[0x%x]\n", r->processor_architecture);
230 printf("\tprocessor_level:[0x%x]\n", r->processor_level);
231 printf("\tref_ic:[0x%x]\n", r->ref_ic);
232 printf("\treserved2:[0x%x]\n", r->reserved2);
233 printf("\treserved3:[0x%x]\n", r->reserved3);
235 printf("\n");
238 /****************************************************************************
239 ****************************************************************************/
241 static void display_print_info1(struct spoolss_PrinterInfo1 *r)
243 printf("\tflags:[0x%x]\n", r->flags);
244 printf("\tname:[%s]\n", r->name);
245 printf("\tdescription:[%s]\n", r->description);
246 printf("\tcomment:[%s]\n", r->comment);
248 printf("\n");
251 /****************************************************************************
252 ****************************************************************************/
254 static void display_print_info2(struct spoolss_PrinterInfo2 *r)
256 printf("\tservername:[%s]\n", r->servername);
257 printf("\tprintername:[%s]\n", r->printername);
258 printf("\tsharename:[%s]\n", r->sharename);
259 printf("\tportname:[%s]\n", r->portname);
260 printf("\tdrivername:[%s]\n", r->drivername);
261 printf("\tcomment:[%s]\n", r->comment);
262 printf("\tlocation:[%s]\n", r->location);
263 printf("\tsepfile:[%s]\n", r->sepfile);
264 printf("\tprintprocessor:[%s]\n", r->printprocessor);
265 printf("\tdatatype:[%s]\n", r->datatype);
266 printf("\tparameters:[%s]\n", r->parameters);
267 printf("\tattributes:[0x%x]\n", r->attributes);
268 printf("\tpriority:[0x%x]\n", r->priority);
269 printf("\tdefaultpriority:[0x%x]\n", r->defaultpriority);
270 printf("\tstarttime:[0x%x]\n", r->starttime);
271 printf("\tuntiltime:[0x%x]\n", r->untiltime);
272 printf("\tstatus:[0x%x]\n", r->status);
273 printf("\tcjobs:[0x%x]\n", r->cjobs);
274 printf("\taverageppm:[0x%x]\n", r->averageppm);
276 if (r->secdesc)
277 display_sec_desc(r->secdesc);
279 printf("\n");
282 /****************************************************************************
283 ****************************************************************************/
285 static void display_print_info3(struct spoolss_PrinterInfo3 *r)
287 display_sec_desc(r->secdesc);
289 printf("\n");
292 /****************************************************************************
293 ****************************************************************************/
295 static void display_print_info4(struct spoolss_PrinterInfo4 *r)
297 printf("\tservername:[%s]\n", r->servername);
298 printf("\tprintername:[%s]\n", r->printername);
299 printf("\tattributes:[0x%x]\n", r->attributes);
300 printf("\n");
303 /****************************************************************************
304 ****************************************************************************/
306 static void display_print_info5(struct spoolss_PrinterInfo5 *r)
308 printf("\tprintername:[%s]\n", r->printername);
309 printf("\tportname:[%s]\n", r->portname);
310 printf("\tattributes:[0x%x]\n", r->attributes);
311 printf("\tdevice_not_selected_timeout:[0x%x]\n", r->device_not_selected_timeout);
312 printf("\ttransmission_retry_timeout:[0x%x]\n", r->transmission_retry_timeout);
313 printf("\n");
316 /****************************************************************************
317 ****************************************************************************/
319 static void display_print_info6(struct spoolss_PrinterInfo6 *r)
321 printf("\tstatus:[0x%x]\n", r->status);
322 printf("\n");
325 /****************************************************************************
326 ****************************************************************************/
328 static void display_print_info7(struct spoolss_PrinterInfo7 *r)
330 printf("\tguid:[%s]\n", r->guid);
331 printf("\taction:[0x%x]\n", r->action);
332 printf("\n");
335 /****************************************************************************
336 ****************************************************************************/
338 static WERROR cmd_spoolss_enum_printers(struct rpc_pipe_client *cli,
339 TALLOC_CTX *mem_ctx,
340 int argc, const char **argv)
342 WERROR result;
343 uint32_t level = 1;
344 union spoolss_PrinterInfo *info;
345 uint32_t i, count;
346 const char *name;
347 uint32_t flags = PRINTER_ENUM_LOCAL;
349 if (argc > 4) {
350 printf("Usage: %s [level] [name] [flags]\n", argv[0]);
351 return WERR_OK;
354 if (argc >= 2) {
355 level = atoi(argv[1]);
358 if (argc >= 3) {
359 name = argv[2];
360 } else {
361 name = cli->srv_name_slash;
364 if (argc == 4) {
365 flags = atoi(argv[3]);
368 result = rpccli_spoolss_enumprinters(cli, mem_ctx,
369 flags,
370 name,
371 level,
373 &count,
374 &info);
375 if (W_ERROR_IS_OK(result)) {
377 if (!count) {
378 printf ("No printers returned.\n");
379 goto done;
382 for (i = 0; i < count; i++) {
383 switch (level) {
384 case 0:
385 display_print_info0(&info[i].info0);
386 break;
387 case 1:
388 display_print_info1(&info[i].info1);
389 break;
390 case 2:
391 display_print_info2(&info[i].info2);
392 break;
393 case 3:
394 display_print_info3(&info[i].info3);
395 break;
396 case 4:
397 display_print_info4(&info[i].info4);
398 break;
399 case 5:
400 display_print_info5(&info[i].info5);
401 break;
402 case 6:
403 display_print_info6(&info[i].info6);
404 break;
405 default:
406 printf("unknown info level %d\n", level);
407 goto done;
411 done:
413 return result;
416 /****************************************************************************
417 ****************************************************************************/
419 static void display_port_info_1(struct spoolss_PortInfo1 *r)
421 printf("\tPort Name:\t[%s]\n", r->port_name);
424 /****************************************************************************
425 ****************************************************************************/
427 static void display_port_info_2(struct spoolss_PortInfo2 *r)
429 printf("\tPort Name:\t[%s]\n", r->port_name);
430 printf("\tMonitor Name:\t[%s]\n", r->monitor_name);
431 printf("\tDescription:\t[%s]\n", r->description);
432 printf("\tPort Type:\t" );
433 if (r->port_type) {
434 int comma = 0; /* hack */
435 printf( "[" );
436 if (r->port_type & SPOOLSS_PORT_TYPE_READ) {
437 printf( "Read" );
438 comma = 1;
440 if (r->port_type & SPOOLSS_PORT_TYPE_WRITE) {
441 printf( "%sWrite", comma ? ", " : "" );
442 comma = 1;
444 /* These two have slightly different interpretations
445 on 95/98/ME but I'm disregarding that for now */
446 if (r->port_type & SPOOLSS_PORT_TYPE_REDIRECTED) {
447 printf( "%sRedirected", comma ? ", " : "" );
448 comma = 1;
450 if (r->port_type & SPOOLSS_PORT_TYPE_NET_ATTACHED) {
451 printf( "%sNet-Attached", comma ? ", " : "" );
453 printf( "]\n" );
454 } else {
455 printf( "[Unset]\n" );
457 printf("\tReserved:\t[%d]\n", r->reserved);
458 printf("\n");
461 /****************************************************************************
462 ****************************************************************************/
464 static WERROR cmd_spoolss_enum_ports(struct rpc_pipe_client *cli,
465 TALLOC_CTX *mem_ctx, int argc,
466 const char **argv)
468 WERROR result;
469 uint32_t level = 1;
470 uint32_t count;
471 union spoolss_PortInfo *info;
473 if (argc > 2) {
474 printf("Usage: %s [level]\n", argv[0]);
475 return WERR_OK;
478 if (argc == 2) {
479 level = atoi(argv[1]);
482 /* Enumerate ports */
484 result = rpccli_spoolss_enumports(cli, mem_ctx,
485 cli->srv_name_slash,
486 level,
488 &count,
489 &info);
490 if (W_ERROR_IS_OK(result)) {
491 int i;
493 for (i = 0; i < count; i++) {
494 switch (level) {
495 case 1:
496 display_port_info_1(&info[i].info1);
497 break;
498 case 2:
499 display_port_info_2(&info[i].info2);
500 break;
501 default:
502 printf("unknown info level %d\n", level);
503 break;
508 return result;
511 /****************************************************************************
512 ****************************************************************************/
514 static WERROR cmd_spoolss_setprinter(struct rpc_pipe_client *cli,
515 TALLOC_CTX *mem_ctx,
516 int argc, const char **argv)
518 struct policy_handle pol;
519 WERROR result;
520 NTSTATUS status;
521 uint32_t info_level = 2;
522 union spoolss_PrinterInfo info;
523 struct spoolss_SetPrinterInfoCtr info_ctr;
524 struct spoolss_SetPrinterInfo2 info2;
525 const char *printername, *comment = NULL;
526 struct spoolss_DevmodeContainer devmode_ctr;
527 struct sec_desc_buf secdesc_ctr;
528 struct dcerpc_binding_handle *b = cli->binding_handle;
530 if (argc == 1 || argc > 3) {
531 printf("Usage: %s printername comment\n", argv[0]);
533 return WERR_OK;
536 /* Open a printer handle */
537 if (argc == 3) {
538 comment = argv[2];
541 ZERO_STRUCT(devmode_ctr);
542 ZERO_STRUCT(secdesc_ctr);
544 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
546 /* get a printer handle */
547 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
548 printername,
549 PRINTER_ALL_ACCESS,
550 &pol);
551 if (!W_ERROR_IS_OK(result))
552 goto done;
554 /* Get printer info */
555 result = rpccli_spoolss_getprinter(cli, mem_ctx,
556 &pol,
557 info_level,
559 &info);
560 if (!W_ERROR_IS_OK(result))
561 goto done;
564 /* Modify the comment. */
565 spoolss_printerinfo2_to_setprinterinfo2(&info.info2, &info2);
566 info2.comment = comment;
568 info_ctr.level = 2;
569 info_ctr.info.info2 = &info2;
571 status = dcerpc_spoolss_SetPrinter(b, mem_ctx,
572 &pol,
573 &info_ctr,
574 &devmode_ctr,
575 &secdesc_ctr,
576 0, /* command */
577 &result);
578 if (!NT_STATUS_IS_OK(status)) {
579 result = ntstatus_to_werror(status);
580 goto done;
582 if (W_ERROR_IS_OK(result))
583 printf("Success in setting comment.\n");
585 done:
586 if (is_valid_policy_hnd(&pol)) {
587 WERROR _result;
588 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
591 return result;
594 /****************************************************************************
595 ****************************************************************************/
597 static WERROR cmd_spoolss_setprintername(struct rpc_pipe_client *cli,
598 TALLOC_CTX *mem_ctx,
599 int argc, const char **argv)
601 struct policy_handle pol;
602 WERROR result;
603 NTSTATUS status;
604 uint32_t info_level = 2;
605 union spoolss_PrinterInfo info;
606 const char *printername,
607 *new_printername = NULL;
608 struct spoolss_SetPrinterInfoCtr info_ctr;
609 struct spoolss_SetPrinterInfo2 info2;
610 struct spoolss_DevmodeContainer devmode_ctr;
611 struct sec_desc_buf secdesc_ctr;
612 struct dcerpc_binding_handle *b = cli->binding_handle;
614 ZERO_STRUCT(devmode_ctr);
615 ZERO_STRUCT(secdesc_ctr);
617 if (argc == 1 || argc > 3) {
618 printf("Usage: %s printername new_printername\n", argv[0]);
620 return WERR_OK;
623 /* Open a printer handle */
624 if (argc == 3) {
625 new_printername = argv[2];
628 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
630 /* get a printer handle */
631 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
632 printername,
633 PRINTER_ALL_ACCESS,
634 &pol);
635 if (!W_ERROR_IS_OK(result))
636 goto done;
638 /* Get printer info */
639 result = rpccli_spoolss_getprinter(cli, mem_ctx,
640 &pol,
641 info_level,
643 &info);
644 if (!W_ERROR_IS_OK(result))
645 goto done;
647 /* Modify the printername. */
648 spoolss_printerinfo2_to_setprinterinfo2(&info.info2, &info2);
649 info2.printername = new_printername;
651 info_ctr.level = 2;
652 info_ctr.info.info2 = &info2;
654 status = dcerpc_spoolss_SetPrinter(b, mem_ctx,
655 &pol,
656 &info_ctr,
657 &devmode_ctr,
658 &secdesc_ctr,
659 0, /* command */
660 &result);
661 if (!NT_STATUS_IS_OK(status)) {
662 result = ntstatus_to_werror(status);
663 goto done;
665 if (W_ERROR_IS_OK(result))
666 printf("Success in setting printername.\n");
668 done:
669 if (is_valid_policy_hnd(&pol)) {
670 WERROR _result;
671 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
674 return result;
677 /****************************************************************************
678 ****************************************************************************/
680 static WERROR cmd_spoolss_getprinter(struct rpc_pipe_client *cli,
681 TALLOC_CTX *mem_ctx,
682 int argc, const char **argv)
684 struct policy_handle pol;
685 WERROR result;
686 uint32_t level = 1;
687 const char *printername;
688 union spoolss_PrinterInfo info;
689 struct dcerpc_binding_handle *b = cli->binding_handle;
691 if (argc == 1 || argc > 3) {
692 printf("Usage: %s <printername> [level]\n", argv[0]);
693 return WERR_OK;
696 /* Open a printer handle */
697 if (argc == 3) {
698 level = atoi(argv[2]);
701 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
703 /* get a printer handle */
705 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
706 printername,
707 SEC_FLAG_MAXIMUM_ALLOWED,
708 &pol);
709 if (!W_ERROR_IS_OK(result)) {
710 goto done;
713 /* Get printer info */
715 result = rpccli_spoolss_getprinter(cli, mem_ctx,
716 &pol,
717 level,
719 &info);
720 if (!W_ERROR_IS_OK(result)) {
721 goto done;
724 /* Display printer info */
725 switch (level) {
726 case 0:
727 display_print_info0(&info.info0);
728 break;
729 case 1:
730 display_print_info1(&info.info1);
731 break;
732 case 2:
733 display_print_info2(&info.info2);
734 break;
735 case 3:
736 display_print_info3(&info.info3);
737 break;
738 case 4:
739 display_print_info4(&info.info4);
740 break;
741 case 5:
742 display_print_info5(&info.info5);
743 break;
744 case 6:
745 display_print_info6(&info.info6);
746 break;
747 case 7:
748 display_print_info7(&info.info7);
749 break;
750 default:
751 printf("unknown info level %d\n", level);
752 break;
754 done:
755 if (is_valid_policy_hnd(&pol)) {
756 WERROR _result;
757 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
760 return result;
763 /****************************************************************************
764 ****************************************************************************/
766 static void display_reg_value(const char *name, enum winreg_Type type, DATA_BLOB blob)
768 const char *text = NULL;
770 switch(type) {
771 case REG_DWORD:
772 if (blob.length == sizeof(uint32)) {
773 printf("%s: REG_DWORD: 0x%08x\n", name, IVAL(blob.data,0));
774 } else {
775 printf("%s: REG_DWORD: <invalid>\n", name);
777 break;
778 case REG_SZ:
779 pull_reg_sz(talloc_tos(), &blob, &text);
780 printf("%s: REG_SZ: %s\n", name, text ? text : "");
781 break;
782 case REG_BINARY: {
783 char *hex = hex_encode_talloc(NULL, blob.data, blob.length);
784 size_t i, len;
785 printf("%s: REG_BINARY:", name);
786 len = strlen(hex);
787 for (i=0; i<len; i++) {
788 if (hex[i] == '\0') {
789 break;
791 if (i%40 == 0) {
792 putchar('\n');
794 putchar(hex[i]);
796 TALLOC_FREE(hex);
797 putchar('\n');
798 break;
800 case REG_MULTI_SZ: {
801 uint32_t i;
802 const char **values;
804 if (!pull_reg_multi_sz(NULL, &blob, &values)) {
805 d_printf("pull_reg_multi_sz failed\n");
806 break;
809 printf("%s: REG_MULTI_SZ: \n", name);
810 for (i=0; values[i] != NULL; i++) {
811 d_printf("%s\n", values[i]);
813 TALLOC_FREE(values);
814 break;
816 default:
817 printf("%s: unknown type %d\n", name, type);
822 /****************************************************************************
823 ****************************************************************************/
825 static void display_printer_data(const char *v,
826 enum winreg_Type type,
827 uint8_t *data,
828 uint32_t length)
830 int i;
831 union spoolss_PrinterData r;
832 DATA_BLOB blob = data_blob_const(data, length);
833 WERROR result;
835 result = pull_spoolss_PrinterData(talloc_tos(), &blob, &r, type);
836 if (!W_ERROR_IS_OK(result)) {
837 return;
840 switch (type) {
841 case REG_DWORD:
842 printf("%s: REG_DWORD: 0x%08x\n", v, r.value);
843 break;
844 case REG_SZ:
845 printf("%s: REG_SZ: %s\n", v, r.string);
846 break;
847 case REG_BINARY: {
848 char *hex = hex_encode_talloc(NULL,
849 r.binary.data, r.binary.length);
850 size_t len;
851 printf("%s: REG_BINARY:", v);
852 len = strlen(hex);
853 for (i=0; i<len; i++) {
854 if (hex[i] == '\0') {
855 break;
857 if (i%40 == 0) {
858 putchar('\n');
860 putchar(hex[i]);
862 TALLOC_FREE(hex);
863 putchar('\n');
864 break;
866 case REG_MULTI_SZ:
867 printf("%s: REG_MULTI_SZ: ", v);
868 for (i=0; r.string_array[i] != NULL; i++) {
869 printf("%s ", r.string_array[i]);
871 printf("\n");
872 break;
873 default:
874 printf("%s: unknown type 0x%02x:\n", v, type);
875 break;
879 /****************************************************************************
880 ****************************************************************************/
882 static WERROR cmd_spoolss_getprinterdata(struct rpc_pipe_client *cli,
883 TALLOC_CTX *mem_ctx,
884 int argc, const char **argv)
886 struct policy_handle pol;
887 WERROR result;
888 fstring printername;
889 const char *valuename;
890 enum winreg_Type type;
891 uint8_t *data;
892 uint32_t needed;
893 struct dcerpc_binding_handle *b = cli->binding_handle;
895 if (argc != 3) {
896 printf("Usage: %s <printername> <valuename>\n", argv[0]);
897 printf("<printername> of . queries print server\n");
898 return WERR_OK;
900 valuename = argv[2];
902 /* Open a printer handle */
904 if (strncmp(argv[1], ".", sizeof(".")) == 0)
905 fstrcpy(printername, cli->srv_name_slash);
906 else
907 slprintf(printername, sizeof(printername)-1, "%s\\%s",
908 cli->srv_name_slash, argv[1]);
910 /* get a printer handle */
912 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
913 printername,
914 SEC_FLAG_MAXIMUM_ALLOWED,
915 &pol);
916 if (!W_ERROR_IS_OK(result))
917 goto done;
919 /* Get printer info */
921 result = rpccli_spoolss_getprinterdata(cli, mem_ctx,
922 &pol,
923 valuename,
925 &type,
926 &needed,
927 &data);
928 if (!W_ERROR_IS_OK(result))
929 goto done;
931 /* Display printer data */
933 display_printer_data(valuename, type, data, needed);
935 done:
936 if (is_valid_policy_hnd(&pol)) {
937 WERROR _result;
938 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
941 return result;
944 /****************************************************************************
945 ****************************************************************************/
947 static WERROR cmd_spoolss_getprinterdataex(struct rpc_pipe_client *cli,
948 TALLOC_CTX *mem_ctx,
949 int argc, const char **argv)
951 struct policy_handle pol;
952 WERROR result;
953 NTSTATUS status;
954 fstring printername;
955 const char *valuename, *keyname;
957 enum winreg_Type type;
958 uint8_t *data = NULL;
959 uint32_t offered = 0;
960 uint32_t needed;
961 struct dcerpc_binding_handle *b = cli->binding_handle;
963 if (argc != 4) {
964 printf("Usage: %s <printername> <keyname> <valuename>\n",
965 argv[0]);
966 printf("<printername> of . queries print server\n");
967 return WERR_OK;
969 valuename = argv[3];
970 keyname = argv[2];
972 /* Open a printer handle */
974 if (strncmp(argv[1], ".", sizeof(".")) == 0)
975 fstrcpy(printername, cli->srv_name_slash);
976 else
977 slprintf(printername, sizeof(printername)-1, "%s\\%s",
978 cli->srv_name_slash, argv[1]);
980 /* get a printer handle */
982 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
983 printername,
984 SEC_FLAG_MAXIMUM_ALLOWED,
985 &pol);
986 if (!W_ERROR_IS_OK(result))
987 goto done;
989 /* Get printer info */
991 data = talloc_zero_array(mem_ctx, uint8_t, offered);
992 if (!data) {
993 goto done;
996 status = dcerpc_spoolss_GetPrinterDataEx(b, mem_ctx,
997 &pol,
998 keyname,
999 valuename,
1000 &type,
1001 data,
1002 offered,
1003 &needed,
1004 &result);
1005 if (!NT_STATUS_IS_OK(status)) {
1006 result = ntstatus_to_werror(status);
1007 goto done;
1009 if (W_ERROR_EQUAL(result, WERR_MORE_DATA)) {
1010 offered = needed;
1011 data = talloc_zero_array(mem_ctx, uint8_t, offered);
1012 if (!data) {
1013 goto done;
1015 status = dcerpc_spoolss_GetPrinterDataEx(b, mem_ctx,
1016 &pol,
1017 keyname,
1018 valuename,
1019 &type,
1020 data,
1021 offered,
1022 &needed,
1023 &result);
1026 if (!NT_STATUS_IS_OK(status)) {
1027 result = ntstatus_to_werror(status);
1028 goto done;
1031 if (!W_ERROR_IS_OK(result))
1032 goto done;
1034 /* Display printer data */
1036 display_printer_data(valuename, type, data, needed);
1039 done:
1040 if (is_valid_policy_hnd(&pol)) {
1041 WERROR _result;
1042 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
1045 return result;
1048 /****************************************************************************
1049 ****************************************************************************/
1051 static void display_print_driver1(struct spoolss_DriverInfo1 *r)
1053 if (!r) {
1054 return;
1057 printf("Printer Driver Info 1:\n");
1058 printf("\tDriver Name: [%s]\n", r->driver_name);
1059 printf("\n");
1062 /****************************************************************************
1063 ****************************************************************************/
1065 static void display_print_driver2(struct spoolss_DriverInfo2 *r)
1067 if (!r) {
1068 return;
1071 printf("Printer Driver Info 2:\n");
1072 printf("\tVersion: [%x]\n", r->version);
1073 printf("\tDriver Name: [%s]\n", r->driver_name);
1074 printf("\tArchitecture: [%s]\n", r->architecture);
1075 printf("\tDriver Path: [%s]\n", r->driver_path);
1076 printf("\tDatafile: [%s]\n", r->data_file);
1077 printf("\tConfigfile: [%s]\n", r->config_file);
1078 printf("\n");
1081 /****************************************************************************
1082 ****************************************************************************/
1084 static void display_print_driver3(struct spoolss_DriverInfo3 *r)
1086 int i;
1088 if (!r) {
1089 return;
1092 printf("Printer Driver Info 3:\n");
1093 printf("\tVersion: [%x]\n", r->version);
1094 printf("\tDriver Name: [%s]\n", r->driver_name);
1095 printf("\tArchitecture: [%s]\n", r->architecture);
1096 printf("\tDriver Path: [%s]\n", r->driver_path);
1097 printf("\tDatafile: [%s]\n", r->data_file);
1098 printf("\tConfigfile: [%s]\n", r->config_file);
1099 printf("\tHelpfile: [%s]\n", r->help_file);
1101 for (i=0; r->dependent_files && r->dependent_files[i] != NULL; i++) {
1102 printf("\tDependentfiles: [%s]\n", r->dependent_files[i]);
1105 printf("\tMonitorname: [%s]\n", r->monitor_name);
1106 printf("\tDefaultdatatype: [%s]\n", r->default_datatype);
1107 printf("\n");
1110 /****************************************************************************
1111 ****************************************************************************/
1113 static void display_print_driver4(struct spoolss_DriverInfo4 *r)
1115 int i;
1117 if (!r) {
1118 return;
1121 printf("Printer Driver Info 4:\n");
1122 printf("\tVersion: [%x]\n", r->version);
1123 printf("\tDriver Name: [%s]\n", r->driver_name);
1124 printf("\tArchitecture: [%s]\n", r->architecture);
1125 printf("\tDriver Path: [%s]\n", r->driver_path);
1126 printf("\tDatafile: [%s]\n", r->data_file);
1127 printf("\tConfigfile: [%s]\n", r->config_file);
1128 printf("\tHelpfile: [%s]\n", r->help_file);
1130 for (i=0; r->dependent_files && r->dependent_files[i] != NULL; i++) {
1131 printf("\tDependentfiles: [%s]\n", r->dependent_files[i]);
1134 printf("\tMonitorname: [%s]\n", r->monitor_name);
1135 printf("\tDefaultdatatype: [%s]\n", r->default_datatype);
1137 for (i=0; r->previous_names && r->previous_names[i] != NULL; i++) {
1138 printf("\tPrevious Names: [%s]\n", r->previous_names[i]);
1140 printf("\n");
1143 /****************************************************************************
1144 ****************************************************************************/
1146 static void display_print_driver5(struct spoolss_DriverInfo5 *r)
1148 if (!r) {
1149 return;
1152 printf("Printer Driver Info 5:\n");
1153 printf("\tVersion: [%x]\n", r->version);
1154 printf("\tDriver Name: [%s]\n", r->driver_name);
1155 printf("\tArchitecture: [%s]\n", r->architecture);
1156 printf("\tDriver Path: [%s]\n", r->driver_path);
1157 printf("\tDatafile: [%s]\n", r->data_file);
1158 printf("\tConfigfile: [%s]\n", r->config_file);
1159 printf("\tDriver Attributes: [0x%x]\n", r->driver_attributes);
1160 printf("\tConfig Version: [0x%x]\n", r->config_version);
1161 printf("\tDriver Version: [0x%x]\n", r->driver_version);
1162 printf("\n");
1165 /****************************************************************************
1166 ****************************************************************************/
1168 static void display_print_driver6(struct spoolss_DriverInfo6 *r)
1170 int i;
1172 if (!r) {
1173 return;
1176 printf("Printer Driver Info 6:\n");
1177 printf("\tVersion: [%x]\n", r->version);
1178 printf("\tDriver Name: [%s]\n", r->driver_name);
1179 printf("\tArchitecture: [%s]\n", r->architecture);
1180 printf("\tDriver Path: [%s]\n", r->driver_path);
1181 printf("\tDatafile: [%s]\n", r->data_file);
1182 printf("\tConfigfile: [%s]\n", r->config_file);
1183 printf("\tHelpfile: [%s]\n", r->help_file);
1185 for (i=0; r->dependent_files && r->dependent_files[i] != NULL; i++) {
1186 printf("\tDependentfiles: [%s]\n", r->dependent_files[i]);
1189 printf("\tMonitorname: [%s]\n", r->monitor_name);
1190 printf("\tDefaultdatatype: [%s]\n", r->default_datatype);
1192 for (i=0; r->previous_names && r->previous_names[i] != NULL; i++) {
1193 printf("\tPrevious Names: [%s]\n", r->previous_names[i]);
1196 printf("\tDriver Date: [%s]\n", nt_time_string(talloc_tos(), r->driver_date));
1197 printf("\tDriver Version: [0x%016llx]\n", (long long unsigned int)r->driver_version);
1198 printf("\tManufacturer Name: [%s]\n", r->manufacturer_name);
1199 printf("\tManufacturer Url: [%s]\n", r->manufacturer_url);
1200 printf("\tHardware ID: [%s]\n", r->hardware_id);
1201 printf("\tProvider: [%s]\n", r->provider);
1203 printf("\n");
1206 /****************************************************************************
1207 ****************************************************************************/
1209 static void display_print_driver8(struct spoolss_DriverInfo8 *r)
1211 int i;
1213 if (!r) {
1214 return;
1217 printf("Printer Driver Info 8:\n");
1218 printf("\tVersion: [%x]\n", r->version);
1219 printf("\tDriver Name: [%s]\n", r->driver_name);
1220 printf("\tArchitecture: [%s]\n", r->architecture);
1221 printf("\tDriver Path: [%s]\n", r->driver_path);
1222 printf("\tDatafile: [%s]\n", r->data_file);
1223 printf("\tConfigfile: [%s]\n", r->config_file);
1224 printf("\tHelpfile: [%s]\n", r->help_file);
1225 printf("\tMonitorname: [%s]\n", r->monitor_name);
1226 printf("\tDefaultdatatype: [%s]\n", r->default_datatype);
1228 for (i=0; r->dependent_files && r->dependent_files[i] != NULL; i++) {
1229 printf("\tDependentfiles: [%s]\n", r->dependent_files[i]);
1232 for (i=0; r->previous_names && r->previous_names[i] != NULL; i++) {
1233 printf("\tPrevious Names: [%s]\n", r->previous_names[i]);
1236 printf("\tDriver Date: [%s]\n", nt_time_string(talloc_tos(), r->driver_date));
1237 printf("\tDriver Version: [0x%016llx]\n", (long long unsigned int)r->driver_version);
1238 printf("\tManufacturer Name: [%s]\n", r->manufacturer_name);
1239 printf("\tManufacturer Url: [%s]\n", r->manufacturer_url);
1240 printf("\tHardware ID: [%s]\n", r->hardware_id);
1241 printf("\tProvider: [%s]\n", r->provider);
1242 printf("\tPrint Processor: [%s]\n", r->print_processor);
1243 printf("\tVendor Setup: [%s]\n", r->vendor_setup);
1244 for (i=0; r->color_profiles && r->color_profiles[i] != NULL; i++) {
1245 printf("\tColor Profiles: [%s]\n", r->color_profiles[i]);
1247 printf("\tInf Path: [%s]\n", r->inf_path);
1248 printf("\tPrinter Driver Attributes: [0x%x]\n", r->printer_driver_attributes);
1249 for (i=0; r->core_driver_dependencies && r->core_driver_dependencies[i] != NULL; i++) {
1250 printf("\tCore Driver Dependencies: [%s]\n", r->core_driver_dependencies[i]);
1252 printf("\tMin Driver Inbox Driver Version Date: [%s]\n", nt_time_string(talloc_tos(), r->min_inbox_driver_ver_date));
1253 printf("\tMin Driver Inbox Driver Version Version: [0x%016llx]\n",
1254 (long long unsigned int)r->min_inbox_driver_ver_version);
1256 printf("\n");
1259 /****************************************************************************
1260 ****************************************************************************/
1262 static WERROR cmd_spoolss_getdriver(struct rpc_pipe_client *cli,
1263 TALLOC_CTX *mem_ctx,
1264 int argc, const char **argv)
1266 struct policy_handle pol;
1267 WERROR werror;
1268 uint32_t level = 3;
1269 const char *printername;
1270 uint32_t i;
1271 bool success = false;
1272 union spoolss_DriverInfo info;
1273 uint32_t server_major_version;
1274 uint32_t server_minor_version;
1275 struct dcerpc_binding_handle *b = cli->binding_handle;
1277 if ((argc == 1) || (argc > 3)) {
1278 printf("Usage: %s <printername> [level]\n", argv[0]);
1279 return WERR_OK;
1282 /* get the arguments need to open the printer handle */
1284 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
1286 if (argc == 3) {
1287 level = atoi(argv[2]);
1290 /* Open a printer handle */
1292 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
1293 printername,
1294 PRINTER_ACCESS_USE,
1295 &pol);
1296 if (!W_ERROR_IS_OK(werror)) {
1297 printf("Error opening printer handle for %s!\n", printername);
1298 return werror;
1301 /* loop through and print driver info level for each architecture */
1303 for (i=0; archi_table[i].long_archi!=NULL; i++) {
1305 werror = rpccli_spoolss_getprinterdriver2(cli, mem_ctx,
1306 &pol,
1307 archi_table[i].long_archi,
1308 level,
1309 0, /* offered */
1310 archi_table[i].version,
1312 &info,
1313 &server_major_version,
1314 &server_minor_version);
1315 if (!W_ERROR_IS_OK(werror)) {
1316 continue;
1319 /* need at least one success */
1321 success = true;
1323 printf("\n[%s]\n", archi_table[i].long_archi);
1325 switch (level) {
1326 case 1:
1327 display_print_driver1(&info.info1);
1328 break;
1329 case 2:
1330 display_print_driver2(&info.info2);
1331 break;
1332 case 3:
1333 display_print_driver3(&info.info3);
1334 break;
1335 case 4:
1336 display_print_driver4(&info.info4);
1337 break;
1338 case 5:
1339 display_print_driver5(&info.info5);
1340 break;
1341 case 6:
1342 display_print_driver6(&info.info6);
1343 break;
1344 case 8:
1345 display_print_driver8(&info.info8);
1346 break;
1347 default:
1348 printf("unknown info level %d\n", level);
1349 break;
1353 /* Cleanup */
1355 if (is_valid_policy_hnd(&pol)) {
1356 WERROR _result;
1357 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
1360 if (success) {
1361 werror = WERR_OK;
1364 return werror;
1367 /****************************************************************************
1368 ****************************************************************************/
1370 static WERROR enum_driver_by_architecture(struct rpc_pipe_client *cli,
1371 TALLOC_CTX *mem_ctx,
1372 const char *architecture,
1373 uint32_t level)
1375 WERROR werror;
1376 uint32_t count = 0;
1377 union spoolss_DriverInfo *info = NULL;
1378 uint32_t j;
1380 werror = rpccli_spoolss_enumprinterdrivers(cli, mem_ctx,
1381 cli->srv_name_slash,
1382 architecture,
1383 level,
1385 &count,
1386 &info);
1388 if (W_ERROR_EQUAL(werror, WERR_INVALID_ENVIRONMENT)) {
1389 printf("Server does not support environment [%s]\n",
1390 architecture);
1391 return WERR_OK;
1394 if (count == 0) {
1395 return WERR_OK;
1398 if (!W_ERROR_IS_OK(werror)) {
1399 printf("Error getting driver for environment [%s] - %s\n",
1400 architecture, win_errstr(werror));
1401 return werror;
1404 printf("\n[%s]\n", architecture);
1406 switch (level) {
1407 case 1:
1408 for (j=0; j < count; j++) {
1409 display_print_driver1(&info[j].info1);
1411 break;
1412 case 2:
1413 for (j=0; j < count; j++) {
1414 display_print_driver2(&info[j].info2);
1416 break;
1417 case 3:
1418 for (j=0; j < count; j++) {
1419 display_print_driver3(&info[j].info3);
1421 break;
1422 case 4:
1423 for (j=0; j < count; j++) {
1424 display_print_driver4(&info[j].info4);
1426 break;
1427 case 5:
1428 for (j=0; j < count; j++) {
1429 display_print_driver5(&info[j].info5);
1431 break;
1432 case 6:
1433 for (j=0; j < count; j++) {
1434 display_print_driver6(&info[j].info6);
1436 break;
1437 case 8:
1438 for (j=0; j < count; j++) {
1439 display_print_driver8(&info[j].info8);
1441 break;
1442 default:
1443 printf("unknown info level %d\n", level);
1444 return WERR_UNKNOWN_LEVEL;
1447 return werror;
1450 static WERROR cmd_spoolss_enum_drivers(struct rpc_pipe_client *cli,
1451 TALLOC_CTX *mem_ctx,
1452 int argc, const char **argv)
1454 WERROR werror = WERR_OK;
1455 uint32_t level = 1;
1456 uint32_t i;
1457 const char *architecture = NULL;
1459 if (argc > 3) {
1460 printf("Usage: enumdrivers [level] [architecture]\n");
1461 return WERR_OK;
1464 if (argc >= 2) {
1465 level = atoi(argv[1]);
1468 if (argc == 3) {
1469 architecture = argv[2];
1472 if (architecture) {
1473 return enum_driver_by_architecture(cli, mem_ctx,
1474 architecture,
1475 level);
1478 /* loop through and print driver info level for each architecture */
1479 for (i=0; archi_table[i].long_archi!=NULL; i++) {
1480 /* check to see if we already asked for this architecture string */
1482 if (i>0 && strequal(archi_table[i].long_archi, archi_table[i-1].long_archi)) {
1483 continue;
1486 werror = enum_driver_by_architecture(cli, mem_ctx,
1487 archi_table[i].long_archi,
1488 level);
1489 if (!W_ERROR_IS_OK(werror)) {
1490 break;
1494 return werror;
1497 /****************************************************************************
1498 ****************************************************************************/
1500 static void display_printdriverdir_1(struct spoolss_DriverDirectoryInfo1 *r)
1502 printf("\tDirectory Name:[%s]\n", r->directory_name);
1505 /****************************************************************************
1506 ****************************************************************************/
1508 static WERROR cmd_spoolss_getdriverdir(struct rpc_pipe_client *cli,
1509 TALLOC_CTX *mem_ctx,
1510 int argc, const char **argv)
1512 WERROR result;
1513 NTSTATUS status;
1514 const char *env = SPOOLSS_ARCHITECTURE_NT_X86;
1515 DATA_BLOB buffer;
1516 uint32_t offered;
1517 union spoolss_DriverDirectoryInfo info;
1518 uint32_t needed;
1519 struct dcerpc_binding_handle *b = cli->binding_handle;
1521 if (argc > 2) {
1522 printf("Usage: %s [environment]\n", argv[0]);
1523 return WERR_OK;
1526 /* Get the arguments need to open the printer handle */
1528 if (argc == 2) {
1529 env = argv[1];
1532 /* Get the directory. Only use Info level 1 */
1534 status = dcerpc_spoolss_GetPrinterDriverDirectory(b, mem_ctx,
1535 cli->srv_name_slash,
1536 env,
1538 NULL, /* buffer */
1539 0, /* offered */
1540 NULL, /* info */
1541 &needed,
1542 &result);
1543 if (!NT_STATUS_IS_OK(status)) {
1544 return ntstatus_to_werror(status);
1546 if (W_ERROR_EQUAL(result, WERR_INSUFFICIENT_BUFFER)) {
1547 offered = needed;
1548 buffer = data_blob_talloc_zero(mem_ctx, needed);
1550 status = dcerpc_spoolss_GetPrinterDriverDirectory(b, mem_ctx,
1551 cli->srv_name_slash,
1552 env,
1554 &buffer,
1555 offered,
1556 &info,
1557 &needed,
1558 &result);
1559 if (!NT_STATUS_IS_OK(status)) {
1560 return ntstatus_to_werror(status);
1564 if (W_ERROR_IS_OK(result)) {
1565 display_printdriverdir_1(&info.info1);
1568 return result;
1571 /****************************************************************************
1572 ****************************************************************************/
1574 static void set_drv_info_3_env(TALLOC_CTX *mem_ctx,
1575 struct spoolss_AddDriverInfo3 *info,
1576 const char *arch)
1579 int i;
1581 for (i=0; archi_table[i].long_archi != NULL; i++)
1583 if (strcmp(arch, archi_table[i].short_archi) == 0)
1585 info->version = archi_table[i].version;
1586 info->architecture = talloc_strdup(mem_ctx, archi_table[i].long_archi);
1587 break;
1591 if (archi_table[i].long_archi == NULL)
1593 DEBUG(0, ("set_drv_info_3_env: Unknown arch [%s]\n", arch));
1596 return;
1600 /**************************************************************************
1601 wrapper for strtok to get the next parameter from a delimited list.
1602 Needed to handle the empty parameter string denoted by "NULL"
1603 *************************************************************************/
1605 static char *get_driver_3_param(TALLOC_CTX *mem_ctx, char *str,
1606 const char *delim, const char **dest,
1607 char **saveptr)
1609 char *ptr;
1611 /* get the next token */
1612 ptr = strtok_r(str, delim, saveptr);
1614 /* a string of 'NULL' is used to represent an empty
1615 parameter because two consecutive delimiters
1616 will not return an empty string. See man strtok(3)
1617 for details */
1618 if (ptr && (strcasecmp_m(ptr, "NULL") == 0)) {
1619 ptr = NULL;
1622 if (dest != NULL) {
1623 *dest = talloc_strdup(mem_ctx, ptr);
1626 return ptr;
1629 /********************************************************************************
1630 fill in the members of a spoolss_AddDriverInfo3 struct using a character
1631 string in the form of
1632 <Long Printer Name>:<Driver File Name>:<Data File Name>:\
1633 <Config File Name>:<Help File Name>:<Language Monitor Name>:\
1634 <Default Data Type>:<Comma Separated list of Files>
1635 *******************************************************************************/
1637 static bool init_drv_info_3_members(TALLOC_CTX *mem_ctx, struct spoolss_AddDriverInfo3 *r,
1638 char *args)
1640 char *str, *str2;
1641 int count = 0;
1642 char *saveptr = NULL;
1643 struct spoolss_StringArray *deps;
1644 const char **file_array = NULL;
1645 int i;
1647 /* fill in the UNISTR fields */
1648 str = get_driver_3_param(mem_ctx, args, ":", &r->driver_name, &saveptr);
1649 str = get_driver_3_param(mem_ctx, NULL, ":", &r->driver_path, &saveptr);
1650 str = get_driver_3_param(mem_ctx, NULL, ":", &r->data_file, &saveptr);
1651 str = get_driver_3_param(mem_ctx, NULL, ":", &r->config_file, &saveptr);
1652 str = get_driver_3_param(mem_ctx, NULL, ":", &r->help_file, &saveptr);
1653 str = get_driver_3_param(mem_ctx, NULL, ":", &r->monitor_name, &saveptr);
1654 str = get_driver_3_param(mem_ctx, NULL, ":", &r->default_datatype, &saveptr);
1656 /* <Comma Separated List of Dependent Files> */
1657 /* save the beginning of the string */
1658 str2 = get_driver_3_param(mem_ctx, NULL, ":", NULL, &saveptr);
1659 str = str2;
1661 /* begin to strip out each filename */
1662 str = strtok_r(str, ",", &saveptr);
1664 /* no dependent files, we are done */
1665 if (!str) {
1666 return true;
1669 deps = talloc_zero(mem_ctx, struct spoolss_StringArray);
1670 if (!deps) {
1671 return false;
1674 while (str != NULL) {
1675 add_string_to_array(deps, str, &file_array, &count);
1676 str = strtok_r(NULL, ",", &saveptr);
1679 deps->string = talloc_zero_array(deps, const char *, count + 1);
1680 if (!deps->string) {
1681 return false;
1684 for (i=0; i < count; i++) {
1685 deps->string[i] = file_array[i];
1688 r->dependent_files = deps;
1690 return true;
1693 /****************************************************************************
1694 ****************************************************************************/
1696 static WERROR cmd_spoolss_addprinterdriver(struct rpc_pipe_client *cli,
1697 TALLOC_CTX *mem_ctx,
1698 int argc, const char **argv)
1700 WERROR result;
1701 NTSTATUS status;
1702 uint32_t level = 3;
1703 struct spoolss_AddDriverInfoCtr info_ctr;
1704 struct spoolss_AddDriverInfo3 info3;
1705 const char *arch;
1706 char *driver_args;
1707 struct dcerpc_binding_handle *b = cli->binding_handle;
1709 /* parse the command arguments */
1710 if (argc != 3 && argc != 4)
1712 printf ("Usage: %s <Environment> \\\n", argv[0]);
1713 printf ("\t<Long Printer Name>:<Driver File Name>:<Data File Name>:\\\n");
1714 printf ("\t<Config File Name>:<Help File Name>:<Language Monitor Name>:\\\n");
1715 printf ("\t<Default Data Type>:<Comma Separated list of Files> \\\n");
1716 printf ("\t[version]\n");
1718 return WERR_OK;
1721 /* Fill in the spoolss_AddDriverInfo3 struct */
1722 ZERO_STRUCT(info3);
1724 arch = cmd_spoolss_get_short_archi(argv[1]);
1725 if (!arch) {
1726 printf ("Error Unknown architecture [%s]\n", argv[1]);
1727 return WERR_INVALID_PARAM;
1730 set_drv_info_3_env(mem_ctx, &info3, arch);
1732 driver_args = talloc_strdup( mem_ctx, argv[2] );
1733 if (!init_drv_info_3_members(mem_ctx, &info3, driver_args ))
1735 printf ("Error Invalid parameter list - %s.\n", argv[2]);
1736 return WERR_INVALID_PARAM;
1739 /* if printer driver version specified, override the default version
1740 * used by the architecture. This allows installation of Windows
1741 * 2000 (version 3) printer drivers. */
1742 if (argc == 4)
1744 info3.version = atoi(argv[3]);
1748 info_ctr.level = level;
1749 info_ctr.info.info3 = &info3;
1751 status = dcerpc_spoolss_AddPrinterDriver(b, mem_ctx,
1752 cli->srv_name_slash,
1753 &info_ctr,
1754 &result);
1755 if (!NT_STATUS_IS_OK(status)) {
1756 return ntstatus_to_werror(status);
1758 if (W_ERROR_IS_OK(result)) {
1759 printf ("Printer Driver %s successfully installed.\n",
1760 info3.driver_name);
1763 return result;
1767 /****************************************************************************
1768 ****************************************************************************/
1770 static WERROR cmd_spoolss_addprinterex(struct rpc_pipe_client *cli,
1771 TALLOC_CTX *mem_ctx,
1772 int argc, const char **argv)
1774 WERROR result;
1775 struct spoolss_SetPrinterInfoCtr info_ctr;
1776 struct spoolss_SetPrinterInfo2 info2;
1778 /* parse the command arguments */
1779 if (argc != 5)
1781 printf ("Usage: %s <name> <shared name> <driver> <port>\n", argv[0]);
1782 return WERR_OK;
1785 /* Fill in the DRIVER_INFO_2 struct */
1786 ZERO_STRUCT(info2);
1788 info2.printername = argv[1];
1789 info2.drivername = argv[3];
1790 info2.sharename = argv[2];
1791 info2.portname = argv[4];
1792 info2.comment = "Created by rpcclient";
1793 info2.printprocessor = "winprint";
1794 info2.datatype = "RAW";
1795 info2.devmode_ptr = 0;
1796 info2.secdesc_ptr = 0;
1797 info2.attributes = PRINTER_ATTRIBUTE_SHARED;
1798 info2.priority = 0;
1799 info2.defaultpriority = 0;
1800 info2.starttime = 0;
1801 info2.untiltime = 0;
1803 /* These three fields must not be used by AddPrinter()
1804 as defined in the MS Platform SDK documentation..
1805 --jerry
1806 info2.status = 0;
1807 info2.cjobs = 0;
1808 info2.averageppm = 0;
1811 info_ctr.level = 2;
1812 info_ctr.info.info2 = &info2;
1814 result = rpccli_spoolss_addprinterex(cli, mem_ctx,
1815 &info_ctr);
1816 if (W_ERROR_IS_OK(result))
1817 printf ("Printer %s successfully installed.\n", argv[1]);
1819 return result;
1822 /****************************************************************************
1823 ****************************************************************************/
1825 static WERROR cmd_spoolss_setdriver(struct rpc_pipe_client *cli,
1826 TALLOC_CTX *mem_ctx,
1827 int argc, const char **argv)
1829 struct policy_handle pol;
1830 WERROR result;
1831 NTSTATUS status;
1832 uint32_t level = 2;
1833 const char *printername;
1834 union spoolss_PrinterInfo info;
1835 struct spoolss_SetPrinterInfoCtr info_ctr;
1836 struct spoolss_SetPrinterInfo2 info2;
1837 struct spoolss_DevmodeContainer devmode_ctr;
1838 struct sec_desc_buf secdesc_ctr;
1839 struct dcerpc_binding_handle *b = cli->binding_handle;
1841 ZERO_STRUCT(devmode_ctr);
1842 ZERO_STRUCT(secdesc_ctr);
1844 /* parse the command arguments */
1845 if (argc != 3)
1847 printf ("Usage: %s <printer> <driver>\n", argv[0]);
1848 return WERR_OK;
1851 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
1853 /* Get a printer handle */
1855 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
1856 printername,
1857 PRINTER_ALL_ACCESS,
1858 &pol);
1859 if (!W_ERROR_IS_OK(result))
1860 goto done;
1862 /* Get printer info */
1864 result = rpccli_spoolss_getprinter(cli, mem_ctx,
1865 &pol,
1866 level,
1868 &info);
1869 if (!W_ERROR_IS_OK(result)) {
1870 printf ("Unable to retrieve printer information!\n");
1871 goto done;
1874 /* Set the printer driver */
1876 spoolss_printerinfo2_to_setprinterinfo2(&info.info2, &info2);
1877 info2.drivername = argv[2];
1879 info_ctr.level = 2;
1880 info_ctr.info.info2 = &info2;
1882 status = dcerpc_spoolss_SetPrinter(b, mem_ctx,
1883 &pol,
1884 &info_ctr,
1885 &devmode_ctr,
1886 &secdesc_ctr,
1887 0, /* command */
1888 &result);
1889 if (!NT_STATUS_IS_OK(status)) {
1890 result = ntstatus_to_werror(status);
1891 goto done;
1893 if (!W_ERROR_IS_OK(result)) {
1894 printf("SetPrinter call failed!\n");
1895 goto done;
1898 printf("Successfully set %s to driver %s.\n", argv[1], argv[2]);
1900 done:
1901 /* Cleanup */
1903 if (is_valid_policy_hnd(&pol)) {
1904 WERROR _result;
1905 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
1908 return result;
1912 /****************************************************************************
1913 ****************************************************************************/
1915 static WERROR cmd_spoolss_deletedriverex(struct rpc_pipe_client *cli,
1916 TALLOC_CTX *mem_ctx,
1917 int argc, const char **argv)
1919 WERROR result, ret = WERR_UNKNOWN_PRINTER_DRIVER;
1920 NTSTATUS status;
1921 struct dcerpc_binding_handle *b = cli->binding_handle;
1923 int i;
1924 int vers = -1;
1926 const char *arch = NULL;
1927 uint32_t delete_flags = 0;
1929 /* parse the command arguments */
1930 if (argc < 2 || argc > 5) {
1931 printf("Usage: %s <driver> [arch] [version] [flags]\n", argv[0]);
1932 return WERR_OK;
1935 if (argc >= 3)
1936 arch = argv[2];
1937 if (argc >= 4) {
1938 vers = atoi(argv[3]);
1939 delete_flags |= DPD_DELETE_SPECIFIC_VERSION;
1941 if (argc == 5)
1942 delete_flags = atoi(argv[4]);
1944 /* delete the driver for all architectures */
1945 for (i=0; archi_table[i].long_archi; i++) {
1947 if (arch && !strequal(archi_table[i].long_archi, arch))
1948 continue;
1950 if (vers >= 0 && archi_table[i].version != vers)
1951 continue;
1953 /* make the call to remove the driver */
1954 status = dcerpc_spoolss_DeletePrinterDriverEx(b, mem_ctx,
1955 cli->srv_name_slash,
1956 archi_table[i].long_archi,
1957 argv[1],
1958 delete_flags,
1959 archi_table[i].version,
1960 &result);
1961 if (!NT_STATUS_IS_OK(status)) {
1962 return ntstatus_to_werror(status);
1964 if ( !W_ERROR_IS_OK(result) )
1966 if ( !W_ERROR_EQUAL(result, WERR_UNKNOWN_PRINTER_DRIVER) ) {
1967 printf ("Failed to remove driver %s for arch [%s] (version: %d): %s\n",
1968 argv[1], archi_table[i].long_archi, archi_table[i].version, win_errstr(result));
1971 else
1973 printf ("Driver %s and files removed for arch [%s] (version: %d).\n", argv[1],
1974 archi_table[i].long_archi, archi_table[i].version);
1975 ret = WERR_OK;
1979 return ret;
1983 /****************************************************************************
1984 ****************************************************************************/
1986 static WERROR cmd_spoolss_deletedriver(struct rpc_pipe_client *cli,
1987 TALLOC_CTX *mem_ctx,
1988 int argc, const char **argv)
1990 WERROR result = WERR_OK;
1991 NTSTATUS status;
1992 int i;
1993 struct dcerpc_binding_handle *b = cli->binding_handle;
1995 /* parse the command arguments */
1996 if (argc != 2) {
1997 printf ("Usage: %s <driver>\n", argv[0]);
1998 return WERR_OK;
2001 /* delete the driver for all architectures */
2002 for (i=0; archi_table[i].long_archi; i++) {
2003 result = WERR_OK;
2005 /* make the call to remove the driver */
2006 status = dcerpc_spoolss_DeletePrinterDriver(b, mem_ctx,
2007 cli->srv_name_slash,
2008 archi_table[i].long_archi,
2009 argv[1],
2010 &result);
2011 if (!NT_STATUS_IS_OK(status)) {
2012 result = ntstatus_to_werror(status);
2013 continue;
2015 if ( !W_ERROR_IS_OK(result) ) {
2016 if ( !W_ERROR_EQUAL(result, WERR_UNKNOWN_PRINTER_DRIVER) ) {
2017 printf ("Failed to remove driver %s for arch [%s] - error %s!\n",
2018 argv[1], archi_table[i].long_archi,
2019 win_errstr(result));
2021 } else {
2022 printf ("Driver %s removed for arch [%s].\n", argv[1],
2023 archi_table[i].long_archi);
2027 return result;
2030 /****************************************************************************
2031 ****************************************************************************/
2033 static WERROR cmd_spoolss_getprintprocdir(struct rpc_pipe_client *cli,
2034 TALLOC_CTX *mem_ctx,
2035 int argc, const char **argv)
2037 WERROR result;
2038 NTSTATUS status;
2039 const char *environment = SPOOLSS_ARCHITECTURE_NT_X86;
2040 DATA_BLOB buffer;
2041 uint32_t offered;
2042 union spoolss_PrintProcessorDirectoryInfo info;
2043 uint32_t needed;
2044 struct dcerpc_binding_handle *b = cli->binding_handle;
2046 /* parse the command arguments */
2047 if (argc > 2) {
2048 printf ("Usage: %s [environment]\n", argv[0]);
2049 return WERR_OK;
2052 if (argc == 2) {
2053 environment = argv[1];
2056 status = dcerpc_spoolss_GetPrintProcessorDirectory(b, mem_ctx,
2057 cli->srv_name_slash,
2058 environment,
2060 NULL, /* buffer */
2061 0, /* offered */
2062 NULL, /* info */
2063 &needed,
2064 &result);
2065 if (!NT_STATUS_IS_OK(status)) {
2066 return ntstatus_to_werror(status);
2068 if (W_ERROR_EQUAL(result, WERR_INSUFFICIENT_BUFFER)) {
2069 offered = needed;
2070 buffer = data_blob_talloc_zero(mem_ctx, needed);
2072 status = dcerpc_spoolss_GetPrintProcessorDirectory(b, mem_ctx,
2073 cli->srv_name_slash,
2074 environment,
2076 &buffer,
2077 offered,
2078 &info,
2079 &needed,
2080 &result);
2081 if (!NT_STATUS_IS_OK(status)) {
2082 return ntstatus_to_werror(status);
2086 if (W_ERROR_IS_OK(result)) {
2087 printf("%s\n", info.info1.directory_name);
2090 return result;
2093 /****************************************************************************
2094 ****************************************************************************/
2096 static WERROR cmd_spoolss_addform(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
2097 int argc, const char **argv)
2099 struct policy_handle handle;
2100 WERROR werror;
2101 NTSTATUS status;
2102 const char *printername;
2103 union spoolss_AddFormInfo info;
2104 struct spoolss_AddFormInfo1 info1;
2105 struct spoolss_AddFormInfo2 info2;
2106 uint32_t level = 1;
2107 struct dcerpc_binding_handle *b = cli->binding_handle;
2109 /* Parse the command arguments */
2111 if (argc < 3 || argc > 5) {
2112 printf ("Usage: %s <printer> <formname> [level]\n", argv[0]);
2113 return WERR_OK;
2116 /* Get a printer handle */
2118 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2120 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2121 printername,
2122 PRINTER_ALL_ACCESS,
2123 &handle);
2124 if (!W_ERROR_IS_OK(werror))
2125 goto done;
2127 /* Dummy up some values for the form data */
2129 if (argc == 4) {
2130 level = atoi(argv[3]);
2133 switch (level) {
2134 case 1:
2135 info1.flags = SPOOLSS_FORM_USER;
2136 info1.form_name = argv[2];
2137 info1.size.width = 100;
2138 info1.size.height = 100;
2139 info1.area.left = 0;
2140 info1.area.top = 10;
2141 info1.area.right = 20;
2142 info1.area.bottom = 30;
2144 info.info1 = &info1;
2146 break;
2147 case 2:
2148 info2.flags = SPOOLSS_FORM_USER;
2149 info2.form_name = argv[2];
2150 info2.size.width = 100;
2151 info2.size.height = 100;
2152 info2.area.left = 0;
2153 info2.area.top = 10;
2154 info2.area.right = 20;
2155 info2.area.bottom = 30;
2156 info2.keyword = argv[2];
2157 info2.string_type = SPOOLSS_FORM_STRING_TYPE_NONE;
2158 info2.mui_dll = NULL;
2159 info2.ressource_id = 0;
2160 info2.display_name = argv[2];
2161 info2.lang_id = 0;
2163 info.info2 = &info2;
2165 break;
2166 default:
2167 werror = WERR_INVALID_PARAM;
2168 goto done;
2171 /* Add the form */
2174 status = dcerpc_spoolss_AddForm(b, mem_ctx,
2175 &handle,
2176 level,
2177 info,
2178 &werror);
2179 if (!NT_STATUS_IS_OK(status)) {
2180 werror = ntstatus_to_werror(status);
2181 goto done;
2183 done:
2184 if (is_valid_policy_hnd(&handle)) {
2185 WERROR _result;
2186 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &handle, &_result);
2189 return werror;
2192 /****************************************************************************
2193 ****************************************************************************/
2195 static WERROR cmd_spoolss_setform(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
2196 int argc, const char **argv)
2198 struct policy_handle handle;
2199 WERROR werror;
2200 NTSTATUS status;
2201 const char *printername;
2202 union spoolss_AddFormInfo info;
2203 struct spoolss_AddFormInfo1 info1;
2204 struct dcerpc_binding_handle *b = cli->binding_handle;
2206 /* Parse the command arguments */
2208 if (argc != 3) {
2209 printf ("Usage: %s <printer> <formname>\n", argv[0]);
2210 return WERR_OK;
2213 /* Get a printer handle */
2215 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2217 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2218 printername,
2219 SEC_FLAG_MAXIMUM_ALLOWED,
2220 &handle);
2221 if (!W_ERROR_IS_OK(werror))
2222 goto done;
2224 /* Dummy up some values for the form data */
2226 info1.flags = SPOOLSS_FORM_PRINTER;
2227 info1.size.width = 100;
2228 info1.size.height = 100;
2229 info1.area.left = 0;
2230 info1.area.top = 1000;
2231 info1.area.right = 2000;
2232 info1.area.bottom = 3000;
2233 info1.form_name = argv[2];
2235 info.info1 = &info1;
2237 /* Set the form */
2239 status = dcerpc_spoolss_SetForm(b, mem_ctx,
2240 &handle,
2241 argv[2],
2243 info,
2244 &werror);
2245 if (!NT_STATUS_IS_OK(status)) {
2246 werror = ntstatus_to_werror(status);
2247 goto done;
2249 done:
2250 if (is_valid_policy_hnd(&handle)) {
2251 WERROR _result;
2252 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &handle, &_result);
2255 return werror;
2258 /****************************************************************************
2259 ****************************************************************************/
2261 static const char *get_form_flag(int form_flag)
2263 switch (form_flag) {
2264 case SPOOLSS_FORM_USER:
2265 return "FORM_USER";
2266 case SPOOLSS_FORM_BUILTIN:
2267 return "FORM_BUILTIN";
2268 case SPOOLSS_FORM_PRINTER:
2269 return "FORM_PRINTER";
2270 default:
2271 return "unknown";
2275 /****************************************************************************
2276 ****************************************************************************/
2278 static void display_form_info1(struct spoolss_FormInfo1 *r)
2280 printf("%s\n" \
2281 "\tflag: %s (%d)\n" \
2282 "\twidth: %d, length: %d\n" \
2283 "\tleft: %d, right: %d, top: %d, bottom: %d\n\n",
2284 r->form_name, get_form_flag(r->flags), r->flags,
2285 r->size.width, r->size.height,
2286 r->area.left, r->area.right,
2287 r->area.top, r->area.bottom);
2290 /****************************************************************************
2291 ****************************************************************************/
2293 static void display_form_info2(struct spoolss_FormInfo2 *r)
2295 printf("%s\n" \
2296 "\tflag: %s (%d)\n" \
2297 "\twidth: %d, length: %d\n" \
2298 "\tleft: %d, right: %d, top: %d, bottom: %d\n",
2299 r->form_name, get_form_flag(r->flags), r->flags,
2300 r->size.width, r->size.height,
2301 r->area.left, r->area.right,
2302 r->area.top, r->area.bottom);
2303 printf("\tkeyword: %s\n", r->keyword);
2304 printf("\tstring_type: 0x%08x\n", r->string_type);
2305 printf("\tmui_dll: %s\n", r->mui_dll);
2306 printf("\tressource_id: 0x%08x\n", r->ressource_id);
2307 printf("\tdisplay_name: %s\n", r->display_name);
2308 printf("\tlang_id: %d\n", r->lang_id);
2309 printf("\n");
2312 /****************************************************************************
2313 ****************************************************************************/
2315 static WERROR cmd_spoolss_getform(struct rpc_pipe_client *cli, TALLOC_CTX *mem_ctx,
2316 int argc, const char **argv)
2318 struct policy_handle handle;
2319 WERROR werror;
2320 NTSTATUS status;
2321 const char *printername;
2322 DATA_BLOB buffer;
2323 uint32_t offered = 0;
2324 union spoolss_FormInfo info;
2325 uint32_t needed;
2326 uint32_t level = 1;
2327 struct dcerpc_binding_handle *b = cli->binding_handle;
2329 /* Parse the command arguments */
2331 if (argc < 3 || argc > 5) {
2332 printf ("Usage: %s <printer> <formname> [level]\n", argv[0]);
2333 return WERR_OK;
2336 /* Get a printer handle */
2338 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2340 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2341 printername,
2342 SEC_FLAG_MAXIMUM_ALLOWED,
2343 &handle);
2344 if (!W_ERROR_IS_OK(werror))
2345 goto done;
2347 if (argc == 4) {
2348 level = atoi(argv[3]);
2351 /* Get the form */
2353 status = dcerpc_spoolss_GetForm(b, mem_ctx,
2354 &handle,
2355 argv[2],
2356 level,
2357 NULL,
2358 offered,
2359 &info,
2360 &needed,
2361 &werror);
2362 if (!NT_STATUS_IS_OK(status)) {
2363 werror = ntstatus_to_werror(status);
2364 goto done;
2366 if (W_ERROR_EQUAL(werror, WERR_INSUFFICIENT_BUFFER)) {
2367 buffer = data_blob_talloc_zero(mem_ctx, needed);
2368 offered = needed;
2369 status = dcerpc_spoolss_GetForm(b, mem_ctx,
2370 &handle,
2371 argv[2],
2372 level,
2373 &buffer,
2374 offered,
2375 &info,
2376 &needed,
2377 &werror);
2378 if (!NT_STATUS_IS_OK(status)) {
2379 werror = ntstatus_to_werror(status);
2380 goto done;
2384 if (!W_ERROR_IS_OK(werror)) {
2385 goto done;
2388 switch (level) {
2389 case 1:
2390 display_form_info1(&info.info1);
2391 break;
2392 case 2:
2393 display_form_info2(&info.info2);
2394 break;
2397 done:
2398 if (is_valid_policy_hnd(&handle)) {
2399 WERROR _result;
2400 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &handle, &_result);
2403 return werror;
2406 /****************************************************************************
2407 ****************************************************************************/
2409 static WERROR cmd_spoolss_deleteform(struct rpc_pipe_client *cli,
2410 TALLOC_CTX *mem_ctx, int argc,
2411 const char **argv)
2413 struct policy_handle handle;
2414 WERROR werror;
2415 NTSTATUS status;
2416 const char *printername;
2417 struct dcerpc_binding_handle *b = cli->binding_handle;
2419 /* Parse the command arguments */
2421 if (argc != 3) {
2422 printf ("Usage: %s <printer> <formname>\n", argv[0]);
2423 return WERR_OK;
2426 /* Get a printer handle */
2428 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2430 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2431 printername,
2432 SEC_FLAG_MAXIMUM_ALLOWED,
2433 &handle);
2434 if (!W_ERROR_IS_OK(werror))
2435 goto done;
2437 /* Delete the form */
2439 status = dcerpc_spoolss_DeleteForm(b, mem_ctx,
2440 &handle,
2441 argv[2],
2442 &werror);
2443 if (!NT_STATUS_IS_OK(status)) {
2444 werror = ntstatus_to_werror(status);
2445 goto done;
2448 done:
2449 if (is_valid_policy_hnd(&handle)) {
2450 WERROR _result;
2451 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &handle, &_result);
2454 return werror;
2457 /****************************************************************************
2458 ****************************************************************************/
2460 static WERROR cmd_spoolss_enum_forms(struct rpc_pipe_client *cli,
2461 TALLOC_CTX *mem_ctx, int argc,
2462 const char **argv)
2464 struct policy_handle handle;
2465 WERROR werror;
2466 const char *printername;
2467 uint32_t num_forms, level = 1, i;
2468 union spoolss_FormInfo *forms;
2469 struct dcerpc_binding_handle *b = cli->binding_handle;
2471 /* Parse the command arguments */
2473 if (argc < 2 || argc > 4) {
2474 printf ("Usage: %s <printer> [level]\n", argv[0]);
2475 return WERR_OK;
2478 /* Get a printer handle */
2480 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2482 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2483 printername,
2484 SEC_FLAG_MAXIMUM_ALLOWED,
2485 &handle);
2486 if (!W_ERROR_IS_OK(werror))
2487 goto done;
2489 if (argc == 3) {
2490 level = atoi(argv[2]);
2493 /* Enumerate forms */
2495 werror = rpccli_spoolss_enumforms(cli, mem_ctx,
2496 &handle,
2497 level,
2499 &num_forms,
2500 &forms);
2502 if (!W_ERROR_IS_OK(werror))
2503 goto done;
2505 /* Display output */
2507 for (i = 0; i < num_forms; i++) {
2508 switch (level) {
2509 case 1:
2510 display_form_info1(&forms[i].info1);
2511 break;
2512 case 2:
2513 display_form_info2(&forms[i].info2);
2514 break;
2518 done:
2519 if (is_valid_policy_hnd(&handle)) {
2520 WERROR _result;
2521 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &handle, &_result);
2524 return werror;
2527 /****************************************************************************
2528 ****************************************************************************/
2530 static WERROR cmd_spoolss_setprinterdata(struct rpc_pipe_client *cli,
2531 TALLOC_CTX *mem_ctx,
2532 int argc, const char **argv)
2534 WERROR result;
2535 NTSTATUS status;
2536 const char *printername;
2537 struct policy_handle pol = { 0, };
2538 union spoolss_PrinterInfo info;
2539 enum winreg_Type type;
2540 union spoolss_PrinterData data;
2541 DATA_BLOB blob;
2542 struct dcerpc_binding_handle *b = cli->binding_handle;
2544 /* parse the command arguments */
2545 if (argc < 5) {
2546 printf ("Usage: %s <printer> <string|binary|dword|multistring>"
2547 " <value> <data>\n",
2548 argv[0]);
2549 return WERR_OK;
2552 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2554 type = REG_NONE;
2556 if (strequal(argv[2], "string")) {
2557 type = REG_SZ;
2560 if (strequal(argv[2], "binary")) {
2561 type = REG_BINARY;
2564 if (strequal(argv[2], "dword")) {
2565 type = REG_DWORD;
2568 if (strequal(argv[2], "multistring")) {
2569 type = REG_MULTI_SZ;
2572 if (type == REG_NONE) {
2573 printf("Unknown data type: %s\n", argv[2]);
2574 result = WERR_INVALID_PARAM;
2575 goto done;
2578 /* get a printer handle */
2580 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2581 printername,
2582 SEC_FLAG_MAXIMUM_ALLOWED,
2583 &pol);
2584 if (!W_ERROR_IS_OK(result)) {
2585 goto done;
2588 result = rpccli_spoolss_getprinter(cli, mem_ctx,
2589 &pol,
2592 &info);
2593 if (!W_ERROR_IS_OK(result)) {
2594 goto done;
2597 printf("%s\n", current_timestring(mem_ctx, true));
2598 printf("\tchange_id (before set)\t:[0x%x]\n", info.info0.change_id);
2600 /* Set the printer data */
2602 switch (type) {
2603 case REG_SZ:
2604 data.string = talloc_strdup(mem_ctx, argv[4]);
2605 W_ERROR_HAVE_NO_MEMORY(data.string);
2606 break;
2607 case REG_DWORD:
2608 data.value = strtoul(argv[4], NULL, 10);
2609 break;
2610 case REG_BINARY:
2611 data.binary = strhex_to_data_blob(mem_ctx, argv[4]);
2612 break;
2613 case REG_MULTI_SZ: {
2614 int i, num_strings;
2615 const char **strings = NULL;
2617 for (i=4; i<argc; i++) {
2618 if (strcmp(argv[i], "NULL") == 0) {
2619 argv[i] = "";
2621 if (!add_string_to_array(mem_ctx, argv[i],
2622 &strings,
2623 &num_strings)) {
2624 result = WERR_NOMEM;
2625 goto done;
2628 data.string_array = talloc_zero_array(mem_ctx, const char *, num_strings + 1);
2629 if (!data.string_array) {
2630 result = WERR_NOMEM;
2631 goto done;
2633 for (i=0; i < num_strings; i++) {
2634 data.string_array[i] = strings[i];
2636 break;
2638 default:
2639 printf("Unknown data type: %s\n", argv[2]);
2640 result = WERR_INVALID_PARAM;
2641 goto done;
2644 result = push_spoolss_PrinterData(mem_ctx, &blob, type, &data);
2645 if (!W_ERROR_IS_OK(result)) {
2646 goto done;
2649 status = dcerpc_spoolss_SetPrinterData(b, mem_ctx,
2650 &pol,
2651 argv[3], /* value_name */
2652 type,
2653 blob.data,
2654 blob.length,
2655 &result);
2656 if (!NT_STATUS_IS_OK(status)) {
2657 printf ("Unable to set [%s=%s]!\n", argv[3], argv[4]);
2658 result = ntstatus_to_werror(status);
2659 goto done;
2661 if (!W_ERROR_IS_OK(result)) {
2662 printf ("Unable to set [%s=%s]!\n", argv[3], argv[4]);
2663 goto done;
2665 printf("\tSetPrinterData succeeded [%s: %s]\n", argv[3], argv[4]);
2667 result = rpccli_spoolss_getprinter(cli, mem_ctx,
2668 &pol,
2671 &info);
2672 if (!W_ERROR_IS_OK(result)) {
2673 goto done;
2676 printf("%s\n", current_timestring(mem_ctx, true));
2677 printf("\tchange_id (after set)\t:[0x%x]\n", info.info0.change_id);
2679 done:
2680 /* cleanup */
2681 if (is_valid_policy_hnd(&pol)) {
2682 WERROR _result;
2683 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &pol, &_result);
2686 return result;
2689 /****************************************************************************
2690 ****************************************************************************/
2692 static void display_job_info1(struct spoolss_JobInfo1 *r)
2694 printf("%d: jobid[%d]: %s %s %s %d/%d pages\n", r->position, r->job_id,
2695 r->user_name, r->document_name, r->text_status, r->pages_printed,
2696 r->total_pages);
2699 /****************************************************************************
2700 ****************************************************************************/
2702 static void display_job_info2(struct spoolss_JobInfo2 *r)
2704 printf("%d: jobid[%d]: %s %s %s %d/%d pages, %d bytes\n",
2705 r->position, r->job_id,
2706 r->user_name, r->document_name, r->text_status, r->pages_printed,
2707 r->total_pages, r->size);
2710 /****************************************************************************
2711 ****************************************************************************/
2713 static void display_job_info3(struct spoolss_JobInfo3 *r)
2715 printf("jobid[%d], next_jobid[%d]\n",
2716 r->job_id, r->next_job_id);
2719 /****************************************************************************
2720 ****************************************************************************/
2722 static void display_job_info4(struct spoolss_JobInfo4 *r)
2724 printf("%d: jobid[%d]: %s %s %s %d/%d pages, %d/%d bytes\n",
2725 r->position, r->job_id,
2726 r->user_name, r->document_name, r->text_status, r->pages_printed,
2727 r->total_pages, r->size, r->size_high);
2730 /****************************************************************************
2731 ****************************************************************************/
2733 static WERROR cmd_spoolss_enum_jobs(struct rpc_pipe_client *cli,
2734 TALLOC_CTX *mem_ctx, int argc,
2735 const char **argv)
2737 WERROR result;
2738 uint32_t level = 1, count, i;
2739 const char *printername;
2740 struct policy_handle hnd;
2741 union spoolss_JobInfo *info;
2742 struct dcerpc_binding_handle *b = cli->binding_handle;
2744 if (argc < 2 || argc > 3) {
2745 printf("Usage: %s printername [level]\n", argv[0]);
2746 return WERR_OK;
2749 if (argc == 3) {
2750 level = atoi(argv[2]);
2753 /* Open printer handle */
2755 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2757 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2758 printername,
2759 SEC_FLAG_MAXIMUM_ALLOWED,
2760 &hnd);
2761 if (!W_ERROR_IS_OK(result))
2762 goto done;
2764 /* Enumerate ports */
2766 result = rpccli_spoolss_enumjobs(cli, mem_ctx,
2767 &hnd,
2768 0, /* firstjob */
2769 1000, /* numjobs */
2770 level,
2772 &count,
2773 &info);
2774 if (!W_ERROR_IS_OK(result)) {
2775 goto done;
2778 for (i = 0; i < count; i++) {
2779 switch (level) {
2780 case 1:
2781 display_job_info1(&info[i].info1);
2782 break;
2783 case 2:
2784 display_job_info2(&info[i].info2);
2785 break;
2786 default:
2787 d_printf("unknown info level %d\n", level);
2788 break;
2792 done:
2793 if (is_valid_policy_hnd(&hnd)) {
2794 WERROR _result;
2795 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
2798 return result;
2801 /****************************************************************************
2802 ****************************************************************************/
2804 static WERROR cmd_spoolss_get_job(struct rpc_pipe_client *cli,
2805 TALLOC_CTX *mem_ctx, int argc,
2806 const char **argv)
2808 WERROR result;
2809 const char *printername;
2810 struct policy_handle hnd;
2811 uint32_t job_id;
2812 uint32_t level = 1;
2813 union spoolss_JobInfo info;
2814 struct dcerpc_binding_handle *b = cli->binding_handle;
2816 if (argc < 3 || argc > 4) {
2817 printf("Usage: %s printername job_id [level]\n", argv[0]);
2818 return WERR_OK;
2821 job_id = atoi(argv[2]);
2823 if (argc == 4) {
2824 level = atoi(argv[3]);
2827 /* Open printer handle */
2829 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2831 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2832 printername,
2833 SEC_FLAG_MAXIMUM_ALLOWED,
2834 &hnd);
2835 if (!W_ERROR_IS_OK(result)) {
2836 goto done;
2839 /* Enumerate ports */
2841 result = rpccli_spoolss_getjob(cli, mem_ctx,
2842 &hnd,
2843 job_id,
2844 level,
2846 &info);
2848 if (!W_ERROR_IS_OK(result)) {
2849 goto done;
2852 switch (level) {
2853 case 1:
2854 display_job_info1(&info.info1);
2855 break;
2856 case 2:
2857 display_job_info2(&info.info2);
2858 break;
2859 case 3:
2860 display_job_info3(&info.info3);
2861 break;
2862 case 4:
2863 display_job_info4(&info.info4);
2864 break;
2865 default:
2866 d_printf("unknown info level %d\n", level);
2867 break;
2870 done:
2871 if (is_valid_policy_hnd(&hnd)) {
2872 WERROR _result;
2873 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
2876 return result;
2879 /****************************************************************************
2880 ****************************************************************************/
2882 static struct {
2883 const char *name;
2884 enum spoolss_JobControl val;
2885 } cmdvals[] = {
2886 {"PAUSE", SPOOLSS_JOB_CONTROL_PAUSE},
2887 {"RESUME", SPOOLSS_JOB_CONTROL_RESUME},
2888 {"CANCEL", SPOOLSS_JOB_CONTROL_CANCEL},
2889 {"RESTART", SPOOLSS_JOB_CONTROL_RESTART},
2890 {"DELETE", SPOOLSS_JOB_CONTROL_DELETE},
2891 {"SEND_TO_PRINTER", SPOOLSS_JOB_CONTROL_SEND_TO_PRINTER},
2892 {"EJECTED", SPOOLSS_JOB_CONTROL_LAST_PAGE_EJECTED},
2893 {"RETAIN", SPOOLSS_JOB_CONTROL_RETAIN},
2894 {"RELEASE", SPOOLSS_JOB_CONTROL_RELEASE}
2897 static enum spoolss_JobControl parse_setjob_command(const char *cmd)
2899 int i;
2901 for (i = 0; i < sizeof(cmdvals)/sizeof(cmdvals[0]); i++) {
2902 if (strequal(cmdvals[i].name, cmd)) {
2903 return cmdvals[i].val;
2906 return (enum spoolss_JobControl)atoi(cmd);
2909 static WERROR cmd_spoolss_set_job(struct rpc_pipe_client *cli,
2910 TALLOC_CTX *mem_ctx, int argc,
2911 const char **argv)
2913 WERROR result;
2914 NTSTATUS status;
2915 const char *printername;
2916 struct policy_handle hnd;
2917 uint32_t job_id;
2918 enum spoolss_JobControl command;
2919 struct dcerpc_binding_handle *b = cli->binding_handle;
2921 if (argc != 4) {
2922 printf("Usage: %s printername job_id command\n", argv[0]);
2923 printf("command = [PAUSE|RESUME|CANCEL|RESTART|DELETE|"
2924 "SEND_TO_PRINTER|EJECTED|RETAIN|RELEASE]\n");
2925 return WERR_OK;
2928 job_id = atoi(argv[2]);
2929 command = parse_setjob_command(argv[3]);
2931 /* Open printer handle */
2933 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2935 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2936 printername,
2937 SEC_FLAG_MAXIMUM_ALLOWED,
2938 &hnd);
2939 if (!W_ERROR_IS_OK(result)) {
2940 goto done;
2943 /* Set Job */
2945 status = dcerpc_spoolss_SetJob(b, mem_ctx,
2946 &hnd,
2947 job_id,
2948 NULL,
2949 command,
2950 &result);
2951 if (!NT_STATUS_IS_OK(status)) {
2952 result = ntstatus_to_werror(status);
2953 goto done;
2955 if (!W_ERROR_IS_OK(result)) {
2956 goto done;
2959 done:
2960 if (is_valid_policy_hnd(&hnd)) {
2961 WERROR _result;
2962 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
2965 return result;
2968 /****************************************************************************
2969 ****************************************************************************/
2971 static WERROR cmd_spoolss_enum_data(struct rpc_pipe_client *cli,
2972 TALLOC_CTX *mem_ctx, int argc,
2973 const char **argv)
2975 WERROR result;
2976 NTSTATUS status;
2977 const char *printername;
2978 struct policy_handle hnd;
2979 uint32_t value_needed;
2980 enum winreg_Type type;
2981 uint32_t data_needed;
2982 struct dcerpc_binding_handle *b = cli->binding_handle;
2983 struct spoolss_EnumPrinterData r;
2985 if (argc != 2) {
2986 printf("Usage: %s printername\n", argv[0]);
2987 return WERR_OK;
2990 /* Open printer handle */
2992 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
2994 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
2995 printername,
2996 SEC_FLAG_MAXIMUM_ALLOWED,
2997 &hnd);
2998 if (!W_ERROR_IS_OK(result)) {
2999 goto done;
3002 /* Enumerate data */
3004 r.in.handle = &hnd;
3005 r.in.enum_index = 0;
3006 r.in.value_offered = 0;
3007 r.in.data_offered = 0;
3008 r.out.value_name = NULL;
3009 r.out.value_needed = &value_needed;
3010 r.out.type = &type;
3011 r.out.data = NULL;
3012 r.out.data_needed = &data_needed;
3014 status = dcerpc_spoolss_EnumPrinterData_r(b, mem_ctx, &r);
3015 if (!NT_STATUS_IS_OK(status)) {
3016 result = ntstatus_to_werror(status);
3017 goto done;
3020 if (!W_ERROR_IS_OK(r.out.result)) {
3021 result = r.out.result;
3022 goto done;
3025 r.in.data_offered = *r.out.data_needed;
3026 r.in.value_offered = *r.out.value_needed;
3027 r.out.data = talloc_zero_array(mem_ctx, uint8_t, r.in.data_offered);
3028 r.out.value_name = talloc_zero_array(mem_ctx, char, r.in.value_offered);
3030 do {
3032 status = dcerpc_spoolss_EnumPrinterData_r(b, mem_ctx, &r);
3033 if (!NT_STATUS_IS_OK(status)) {
3034 result = ntstatus_to_werror(status);
3035 goto done;
3038 if (W_ERROR_EQUAL(r.out.result, WERR_NO_MORE_ITEMS)) {
3039 result = WERR_OK;
3040 break;
3043 r.in.enum_index++;
3045 display_reg_value(r.out.value_name, *r.out.type,
3046 data_blob_const(r.out.data, r.in.data_offered));
3048 } while (W_ERROR_IS_OK(r.out.result));
3050 done:
3051 if (is_valid_policy_hnd(&hnd)) {
3052 WERROR _result;
3053 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
3056 return result;
3059 /****************************************************************************
3060 ****************************************************************************/
3062 static WERROR cmd_spoolss_enum_data_ex( struct rpc_pipe_client *cli,
3063 TALLOC_CTX *mem_ctx, int argc,
3064 const char **argv)
3066 WERROR result;
3067 uint32_t i;
3068 const char *printername;
3069 struct policy_handle hnd;
3070 uint32_t count;
3071 struct spoolss_PrinterEnumValues *info;
3072 struct dcerpc_binding_handle *b = cli->binding_handle;
3074 if (argc != 3) {
3075 printf("Usage: %s printername <keyname>\n", argv[0]);
3076 return WERR_OK;
3079 /* Open printer handle */
3081 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
3083 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
3084 printername,
3085 SEC_FLAG_MAXIMUM_ALLOWED,
3086 &hnd);
3087 if (!W_ERROR_IS_OK(result)) {
3088 goto done;
3091 /* Enumerate subkeys */
3093 result = rpccli_spoolss_enumprinterdataex(cli, mem_ctx,
3094 &hnd,
3095 argv[2],
3097 &count,
3098 &info);
3099 if (!W_ERROR_IS_OK(result)) {
3100 goto done;
3103 for (i=0; i < count; i++) {
3104 display_printer_data(info[i].value_name,
3105 info[i].type,
3106 info[i].data->data,
3107 info[i].data->length);
3110 done:
3111 if (is_valid_policy_hnd(&hnd)) {
3112 WERROR _result;
3113 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
3116 return result;
3119 /****************************************************************************
3120 ****************************************************************************/
3122 static WERROR cmd_spoolss_enum_printerkey(struct rpc_pipe_client *cli,
3123 TALLOC_CTX *mem_ctx, int argc,
3124 const char **argv)
3126 WERROR result;
3127 const char *printername;
3128 const char *keyname = NULL;
3129 struct policy_handle hnd;
3130 const char **key_buffer = NULL;
3131 int i;
3132 uint32_t offered = 0;
3133 struct dcerpc_binding_handle *b = cli->binding_handle;
3135 if (argc < 2 || argc > 4) {
3136 printf("Usage: %s printername [keyname] [offered]\n", argv[0]);
3137 return WERR_OK;
3140 if (argc >= 3) {
3141 keyname = argv[2];
3142 } else {
3143 keyname = "";
3146 if (argc == 4) {
3147 offered = atoi(argv[3]);
3150 /* Open printer handle */
3152 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
3154 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
3155 printername,
3156 SEC_FLAG_MAXIMUM_ALLOWED,
3157 &hnd);
3158 if (!W_ERROR_IS_OK(result)) {
3159 goto done;
3162 /* Enumerate subkeys */
3164 result = rpccli_spoolss_enumprinterkey(cli, mem_ctx,
3165 &hnd,
3166 keyname,
3167 &key_buffer,
3168 offered);
3170 if (!W_ERROR_IS_OK(result)) {
3171 goto done;
3174 for (i=0; key_buffer && key_buffer[i]; i++) {
3175 printf("%s\n", key_buffer[i]);
3178 done:
3180 if (is_valid_policy_hnd(&hnd)) {
3181 WERROR _result;
3182 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
3185 return result;
3188 /****************************************************************************
3189 ****************************************************************************/
3191 static WERROR cmd_spoolss_rffpcnex(struct rpc_pipe_client *cli,
3192 TALLOC_CTX *mem_ctx, int argc,
3193 const char **argv)
3195 const char *printername;
3196 const char *clientname;
3197 struct policy_handle hnd;
3198 WERROR result;
3199 NTSTATUS status;
3200 struct spoolss_NotifyOption option;
3201 struct dcerpc_binding_handle *b = cli->binding_handle;
3203 if (argc != 2) {
3204 printf("Usage: %s printername\n", argv[0]);
3205 result = WERR_OK;
3206 goto done;
3209 /* Open printer */
3211 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
3213 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
3214 printername,
3215 SEC_FLAG_MAXIMUM_ALLOWED,
3216 &hnd);
3217 if (!W_ERROR_IS_OK(result)) {
3218 printf("Error opening %s\n", argv[1]);
3219 goto done;
3222 /* Create spool options */
3224 option.version = 2;
3225 option.count = 2;
3227 option.types = talloc_array(mem_ctx, struct spoolss_NotifyOptionType, 2);
3228 if (option.types == NULL) {
3229 result = WERR_NOMEM;
3230 goto done;
3233 option.types[0].type = PRINTER_NOTIFY_TYPE;
3234 option.types[0].count = 1;
3235 option.types[0].fields = talloc_array(mem_ctx, union spoolss_Field, 1);
3236 if (option.types[0].fields == NULL) {
3237 result = WERR_NOMEM;
3238 goto done;
3240 option.types[0].fields[0].field = PRINTER_NOTIFY_FIELD_SERVER_NAME;
3242 option.types[1].type = JOB_NOTIFY_TYPE;
3243 option.types[1].count = 1;
3244 option.types[1].fields = talloc_array(mem_ctx, union spoolss_Field, 1);
3245 if (option.types[1].fields == NULL) {
3246 result = WERR_NOMEM;
3247 goto done;
3249 option.types[1].fields[0].field = JOB_NOTIFY_FIELD_PRINTER_NAME;
3251 clientname = talloc_asprintf(mem_ctx, "\\\\%s", lp_netbios_name());
3252 if (!clientname) {
3253 result = WERR_NOMEM;
3254 goto done;
3257 /* Send rffpcnex */
3259 status = dcerpc_spoolss_RemoteFindFirstPrinterChangeNotifyEx(b, mem_ctx,
3260 &hnd,
3263 clientname,
3264 123,
3265 &option,
3266 &result);
3267 if (!NT_STATUS_IS_OK(status)) {
3268 result = ntstatus_to_werror(status);
3269 goto done;
3271 if (!W_ERROR_IS_OK(result)) {
3272 printf("Error rffpcnex %s\n", argv[1]);
3273 goto done;
3276 done:
3277 if (is_valid_policy_hnd(&hnd)) {
3278 WERROR _result;
3279 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &hnd, &_result);
3282 return result;
3285 /****************************************************************************
3286 ****************************************************************************/
3288 static bool compare_printer( struct rpc_pipe_client *cli1, struct policy_handle *hnd1,
3289 struct rpc_pipe_client *cli2, struct policy_handle *hnd2 )
3291 union spoolss_PrinterInfo info1, info2;
3292 WERROR werror;
3293 TALLOC_CTX *mem_ctx = talloc_init("compare_printer");
3295 printf("Retrieving printer propertiesfor %s...", cli1->desthost);
3296 werror = rpccli_spoolss_getprinter(cli1, mem_ctx,
3297 hnd1,
3300 &info1);
3301 if ( !W_ERROR_IS_OK(werror) ) {
3302 printf("failed (%s)\n", win_errstr(werror));
3303 talloc_destroy(mem_ctx);
3304 return false;
3306 printf("ok\n");
3308 printf("Retrieving printer properties for %s...", cli2->desthost);
3309 werror = rpccli_spoolss_getprinter(cli2, mem_ctx,
3310 hnd2,
3313 &info2);
3314 if ( !W_ERROR_IS_OK(werror) ) {
3315 printf("failed (%s)\n", win_errstr(werror));
3316 talloc_destroy(mem_ctx);
3317 return false;
3319 printf("ok\n");
3321 talloc_destroy(mem_ctx);
3323 return true;
3326 /****************************************************************************
3327 ****************************************************************************/
3329 static bool compare_printer_secdesc( struct rpc_pipe_client *cli1, struct policy_handle *hnd1,
3330 struct rpc_pipe_client *cli2, struct policy_handle *hnd2 )
3332 union spoolss_PrinterInfo info1, info2;
3333 WERROR werror;
3334 TALLOC_CTX *mem_ctx = talloc_init("compare_printer_secdesc");
3335 struct security_descriptor *sd1, *sd2;
3336 bool result = true;
3339 printf("Retrieving printer security for %s...", cli1->desthost);
3340 werror = rpccli_spoolss_getprinter(cli1, mem_ctx,
3341 hnd1,
3344 &info1);
3345 if ( !W_ERROR_IS_OK(werror) ) {
3346 printf("failed (%s)\n", win_errstr(werror));
3347 result = false;
3348 goto done;
3350 printf("ok\n");
3352 printf("Retrieving printer security for %s...", cli2->desthost);
3353 werror = rpccli_spoolss_getprinter(cli2, mem_ctx,
3354 hnd2,
3357 &info2);
3358 if ( !W_ERROR_IS_OK(werror) ) {
3359 printf("failed (%s)\n", win_errstr(werror));
3360 result = false;
3361 goto done;
3363 printf("ok\n");
3366 printf("++ ");
3368 sd1 = info1.info3.secdesc;
3369 sd2 = info2.info3.secdesc;
3371 if ( (sd1 != sd2) && ( !sd1 || !sd2 ) ) {
3372 printf("NULL secdesc!\n");
3373 result = false;
3374 goto done;
3377 if (!security_descriptor_equal( sd1, sd2 ) ) {
3378 printf("Security Descriptors *not* equal!\n");
3379 result = false;
3380 goto done;
3383 printf("Security descriptors match\n");
3385 done:
3386 talloc_destroy(mem_ctx);
3387 return result;
3391 /****************************************************************************
3392 ****************************************************************************/
3394 extern struct user_auth_info *rpcclient_auth_info;
3396 static WERROR cmd_spoolss_printercmp(struct rpc_pipe_client *cli,
3397 TALLOC_CTX *mem_ctx, int argc,
3398 const char **argv)
3400 const char *printername;
3401 char *printername_path = NULL;
3402 struct cli_state *cli_server2 = NULL;
3403 struct rpc_pipe_client *cli2 = NULL;
3404 struct policy_handle hPrinter1, hPrinter2;
3405 NTSTATUS nt_status;
3406 WERROR werror;
3408 if ( argc != 3 ) {
3409 printf("Usage: %s <printer> <server>\n", argv[0]);
3410 return WERR_OK;
3413 printername = argv[1];
3415 /* first get the connection to the remote server */
3417 nt_status = cli_full_connection(&cli_server2, lp_netbios_name(), argv[2],
3418 NULL, 0,
3419 "IPC$", "IPC",
3420 get_cmdline_auth_info_username(rpcclient_auth_info),
3421 lp_workgroup(),
3422 get_cmdline_auth_info_password(rpcclient_auth_info),
3423 get_cmdline_auth_info_use_kerberos(rpcclient_auth_info) ? CLI_FULL_CONNECTION_USE_KERBEROS : 0,
3424 get_cmdline_auth_info_signing_state(rpcclient_auth_info));
3426 if ( !NT_STATUS_IS_OK(nt_status) )
3427 return WERR_GENERAL_FAILURE;
3429 nt_status = cli_rpc_pipe_open_noauth(cli_server2, &ndr_table_spoolss.syntax_id,
3430 &cli2);
3431 if (!NT_STATUS_IS_OK(nt_status)) {
3432 printf("failed to open spoolss pipe on server %s (%s)\n",
3433 argv[2], nt_errstr(nt_status));
3434 return WERR_GENERAL_FAILURE;
3437 /* now open up both printers */
3439 RPCCLIENT_PRINTERNAME(printername_path, cli, printername);
3441 printf("Opening %s...", printername_path);
3443 werror = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
3444 printername_path,
3445 PRINTER_ALL_ACCESS,
3446 &hPrinter1);
3447 if ( !W_ERROR_IS_OK(werror) ) {
3448 printf("failed (%s)\n", win_errstr(werror));
3449 goto done;
3451 printf("ok\n");
3453 RPCCLIENT_PRINTERNAME(printername_path, cli2, printername);
3455 printf("Opening %s...", printername_path);
3456 werror = rpccli_spoolss_openprinter_ex(cli2, mem_ctx,
3457 printername_path,
3458 PRINTER_ALL_ACCESS,
3459 &hPrinter2);
3460 if ( !W_ERROR_IS_OK(werror) ) {
3461 printf("failed (%s)\n", win_errstr(werror));
3462 goto done;
3464 printf("ok\n");
3466 compare_printer( cli, &hPrinter1, cli2, &hPrinter2 );
3467 compare_printer_secdesc( cli, &hPrinter1, cli2, &hPrinter2 );
3468 #if 0
3469 compare_printerdata( cli_server1, &hPrinter1, cli_server2, &hPrinter2 );
3470 #endif
3473 done:
3474 /* cleanup */
3476 printf("Closing printers...");
3478 WERROR _result;
3479 dcerpc_spoolss_ClosePrinter(cli->binding_handle, mem_ctx, &hPrinter1, &_result);
3480 dcerpc_spoolss_ClosePrinter(cli2->binding_handle, mem_ctx, &hPrinter2, &_result);
3482 printf("ok\n");
3484 /* close the second remote connection */
3486 cli_shutdown( cli_server2 );
3487 return WERR_OK;
3490 static void display_proc_info1(struct spoolss_PrintProcessorInfo1 *r)
3492 printf("print_processor_name: %s\n", r->print_processor_name);
3495 static WERROR cmd_spoolss_enum_procs(struct rpc_pipe_client *cli,
3496 TALLOC_CTX *mem_ctx, int argc,
3497 const char **argv)
3499 WERROR werror;
3500 const char *environment = SPOOLSS_ARCHITECTURE_NT_X86;
3501 uint32_t num_procs, level = 1, i;
3502 union spoolss_PrintProcessorInfo *procs;
3504 /* Parse the command arguments */
3506 if (argc < 1 || argc > 4) {
3507 printf ("Usage: %s [environment] [level]\n", argv[0]);
3508 return WERR_OK;
3511 if (argc >= 2) {
3512 environment = argv[1];
3515 if (argc == 3) {
3516 level = atoi(argv[2]);
3519 /* Enumerate Print Processors */
3521 werror = rpccli_spoolss_enumprintprocessors(cli, mem_ctx,
3522 cli->srv_name_slash,
3523 environment,
3524 level,
3526 &num_procs,
3527 &procs);
3528 if (!W_ERROR_IS_OK(werror))
3529 goto done;
3531 /* Display output */
3533 for (i = 0; i < num_procs; i++) {
3534 switch (level) {
3535 case 1:
3536 display_proc_info1(&procs[i].info1);
3537 break;
3541 done:
3542 return werror;
3545 static void display_proc_data_types_info1(struct spoolss_PrintProcDataTypesInfo1 *r)
3547 printf("name_array: %s\n", r->name_array);
3550 static WERROR cmd_spoolss_enum_proc_data_types(struct rpc_pipe_client *cli,
3551 TALLOC_CTX *mem_ctx, int argc,
3552 const char **argv)
3554 WERROR werror;
3555 const char *print_processor_name = "winprint";
3556 uint32_t num_procs, level = 1, i;
3557 union spoolss_PrintProcDataTypesInfo *procs;
3559 /* Parse the command arguments */
3561 if (argc < 1 || argc > 4) {
3562 printf ("Usage: %s [environment] [level]\n", argv[0]);
3563 return WERR_OK;
3566 if (argc >= 2) {
3567 print_processor_name = argv[1];
3570 if (argc == 3) {
3571 level = atoi(argv[2]);
3574 /* Enumerate Print Processor Data Types */
3576 werror = rpccli_spoolss_enumprintprocessordatatypes(cli, mem_ctx,
3577 cli->srv_name_slash,
3578 print_processor_name,
3579 level,
3581 &num_procs,
3582 &procs);
3583 if (!W_ERROR_IS_OK(werror))
3584 goto done;
3586 /* Display output */
3588 for (i = 0; i < num_procs; i++) {
3589 switch (level) {
3590 case 1:
3591 display_proc_data_types_info1(&procs[i].info1);
3592 break;
3596 done:
3597 return werror;
3600 static void display_monitor1(const struct spoolss_MonitorInfo1 *r)
3602 printf("monitor_name: %s\n", r->monitor_name);
3605 static void display_monitor2(const struct spoolss_MonitorInfo2 *r)
3607 printf("monitor_name: %s\n", r->monitor_name);
3608 printf("environment: %s\n", r->environment);
3609 printf("dll_name: %s\n", r->dll_name);
3612 static WERROR cmd_spoolss_enum_monitors(struct rpc_pipe_client *cli,
3613 TALLOC_CTX *mem_ctx, int argc,
3614 const char **argv)
3616 WERROR werror;
3617 uint32_t count, level = 1, i;
3618 union spoolss_MonitorInfo *info;
3620 /* Parse the command arguments */
3622 if (argc > 2) {
3623 printf("Usage: %s [level]\n", argv[0]);
3624 return WERR_OK;
3627 if (argc == 2) {
3628 level = atoi(argv[1]);
3631 /* Enumerate Print Monitors */
3633 werror = rpccli_spoolss_enummonitors(cli, mem_ctx,
3634 cli->srv_name_slash,
3635 level,
3637 &count,
3638 &info);
3639 if (!W_ERROR_IS_OK(werror)) {
3640 goto done;
3643 /* Display output */
3645 for (i = 0; i < count; i++) {
3646 switch (level) {
3647 case 1:
3648 display_monitor1(&info[i].info1);
3649 break;
3650 case 2:
3651 display_monitor2(&info[i].info2);
3652 break;
3656 done:
3657 return werror;
3660 static WERROR cmd_spoolss_create_printer_ic(struct rpc_pipe_client *cli,
3661 TALLOC_CTX *mem_ctx, int argc,
3662 const char **argv)
3664 WERROR result;
3665 NTSTATUS status;
3666 struct policy_handle handle, gdi_handle;
3667 const char *printername;
3668 struct spoolss_DevmodeContainer devmode_ctr;
3669 struct dcerpc_binding_handle *b = cli->binding_handle;
3671 RPCCLIENT_PRINTERNAME(printername, cli, argv[1]);
3673 result = rpccli_spoolss_openprinter_ex(cli, mem_ctx,
3674 printername,
3675 SEC_FLAG_MAXIMUM_ALLOWED,
3676 &handle);
3677 if (!W_ERROR_IS_OK(result)) {
3678 return result;
3681 ZERO_STRUCT(devmode_ctr);
3683 status = dcerpc_spoolss_CreatePrinterIC(b, mem_ctx,
3684 &handle,
3685 &gdi_handle,
3686 &devmode_ctr,
3687 &result);
3688 if (!NT_STATUS_IS_OK(status)) {
3689 result = ntstatus_to_werror(status);
3690 goto done;
3692 if (!W_ERROR_IS_OK(result)) {
3693 goto done;
3696 done:
3697 if (is_valid_policy_hnd(&gdi_handle)) {
3698 WERROR _result;
3699 dcerpc_spoolss_DeletePrinterIC(b, mem_ctx, &gdi_handle, &_result);
3701 if (is_valid_policy_hnd(&handle)) {
3702 WERROR _result;
3703 dcerpc_spoolss_ClosePrinter(b, mem_ctx, &handle, &_result);
3706 return result;
3709 /* List of commands exported by this module */
3710 struct cmd_set spoolss_commands[] = {
3712 { "SPOOLSS" },
3714 { "adddriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addprinterdriver, &ndr_table_spoolss, NULL, "Add a print driver", "" },
3715 { "addprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addprinterex, &ndr_table_spoolss, NULL, "Add a printer", "" },
3716 { "deldriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deletedriver, &ndr_table_spoolss, NULL, "Delete a printer driver", "" },
3717 { "deldriverex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deletedriverex, &ndr_table_spoolss, NULL, "Delete a printer driver with files", "" },
3718 { "enumdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_data, &ndr_table_spoolss, NULL, "Enumerate printer data", "" },
3719 { "enumdataex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_data_ex, &ndr_table_spoolss, NULL, "Enumerate printer data for a key", "" },
3720 { "enumkey", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_printerkey, &ndr_table_spoolss, NULL, "Enumerate printer keys", "" },
3721 { "enumjobs", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_jobs, &ndr_table_spoolss, NULL, "Enumerate print jobs", "" },
3722 { "getjob", RPC_RTYPE_WERROR, NULL, cmd_spoolss_get_job, &ndr_table_spoolss, NULL, "Get print job", "" },
3723 { "setjob", RPC_RTYPE_WERROR, NULL, cmd_spoolss_set_job, &ndr_table_spoolss, NULL, "Set print job", "" },
3724 { "enumports", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_ports, &ndr_table_spoolss, NULL, "Enumerate printer ports", "" },
3725 { "enumdrivers", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_drivers, &ndr_table_spoolss, NULL, "Enumerate installed printer drivers", "" },
3726 { "enumprinters", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_printers, &ndr_table_spoolss, NULL, "Enumerate printers", "" },
3727 { "getdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinterdata, &ndr_table_spoolss, NULL, "Get print driver data", "" },
3728 { "getdataex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinterdataex, &ndr_table_spoolss, NULL, "Get printer driver data with keyname", ""},
3729 { "getdriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getdriver, &ndr_table_spoolss, NULL, "Get print driver information", "" },
3730 { "getdriverdir", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getdriverdir, &ndr_table_spoolss, NULL, "Get print driver upload directory", "" },
3731 { "getprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinter, &ndr_table_spoolss, NULL, "Get printer info", "" },
3732 { "openprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_open_printer, &ndr_table_spoolss, NULL, "Open printer handle", "" },
3733 { "openprinter_ex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_open_printer_ex, &ndr_table_spoolss, NULL, "Open printer handle", "" },
3734 { "setdriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setdriver, &ndr_table_spoolss, NULL, "Set printer driver", "" },
3735 { "getprintprocdir", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprintprocdir, &ndr_table_spoolss, NULL, "Get print processor directory", "" },
3736 { "addform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addform, &ndr_table_spoolss, NULL, "Add form", "" },
3737 { "setform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setform, &ndr_table_spoolss, NULL, "Set form", "" },
3738 { "getform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getform, &ndr_table_spoolss, NULL, "Get form", "" },
3739 { "deleteform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deleteform, &ndr_table_spoolss, NULL, "Delete form", "" },
3740 { "enumforms", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_forms, &ndr_table_spoolss, NULL, "Enumerate forms", "" },
3741 { "setprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprinter, &ndr_table_spoolss, NULL, "Set printer comment", "" },
3742 { "setprintername", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprintername, &ndr_table_spoolss, NULL, "Set printername", "" },
3743 { "setprinterdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprinterdata, &ndr_table_spoolss, NULL, "Set REG_SZ printer data", "" },
3744 { "rffpcnex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_rffpcnex, &ndr_table_spoolss, NULL, "Rffpcnex test", "" },
3745 { "printercmp", RPC_RTYPE_WERROR, NULL, cmd_spoolss_printercmp, &ndr_table_spoolss, NULL, "Printer comparison test", "" },
3746 { "enumprocs", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_procs, &ndr_table_spoolss, NULL, "Enumerate Print Processors", "" },
3747 { "enumprocdatatypes", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_proc_data_types, &ndr_table_spoolss, NULL, "Enumerate Print Processor Data Types", "" },
3748 { "enummonitors", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_monitors, &ndr_table_spoolss, NULL, "Enumerate Print Monitors", "" },
3749 { "createprinteric", RPC_RTYPE_WERROR, NULL, cmd_spoolss_create_printer_ic, &ndr_table_spoolss, NULL, "Create Printer IC", "" },
3751 { NULL }