r2176: syncing necessary changes for 3.0.7
[Samba.git] / source / rpcclient / cmd_spoolss.c
blob5fa59e1903b52396cd760e4f983a86775ec00b98
1 /*
2 Unix SMB/CIFS implementation.
3 RPC pipe client
5 Copyright (C) Gerald Carter 2001
6 Copyright (C) Tim Potter 2000
7 Copyright (C) Andrew Tridgell 1992-1999
8 Copyright (C) Luke Kenneth Casson Leighton 1996-1999
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 2 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program; if not, write to the Free Software
22 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
25 #include "includes.h"
26 #include "rpcclient.h"
28 struct table_node {
29 const char *long_archi;
30 const char *short_archi;
31 int version;
34 static const struct table_node archi_table[]= {
36 {"Windows 4.0", "WIN40", 0 },
37 {"Windows NT x86", "W32X86", 2 },
38 {"Windows NT x86", "W32X86", 3 },
39 {"Windows NT R4000", "W32MIPS", 2 },
40 {"Windows NT Alpha_AXP", "W32ALPHA", 2 },
41 {"Windows NT PowerPC", "W32PPC", 2 },
42 {NULL, "", -1 }
45 /**
46 * @file
48 * rpcclient module for SPOOLSS rpc pipe.
50 * This generally just parses and checks command lines, and then calls
51 * a cli_spoolss function.
52 **/
54 /****************************************************************************
55 function to do the mapping between the long architecture name and
56 the short one.
57 ****************************************************************************/
58 static const char *cmd_spoolss_get_short_archi(const char *long_archi)
60 int i=-1;
62 DEBUG(107,("Getting architecture dependant directory\n"));
63 do {
64 i++;
65 } while ( (archi_table[i].long_archi!=NULL ) &&
66 StrCaseCmp(long_archi, archi_table[i].long_archi) );
68 if (archi_table[i].long_archi==NULL) {
69 DEBUGADD(10,("Unknown architecture [%s] !\n", long_archi));
70 return NULL;
73 /* this might be client code - but shouldn't this be an fstrcpy etc? */
76 DEBUGADD(108,("index: [%d]\n", i));
77 DEBUGADD(108,("long architecture: [%s]\n", archi_table[i].long_archi));
78 DEBUGADD(108,("short architecture: [%s]\n", archi_table[i].short_archi));
80 return archi_table[i].short_archi;
83 #if 0
84 /**********************************************************************
85 * dummy function -- placeholder
87 static WERROR cmd_spoolss_not_implemented(struct cli_state *cli,
88 TALLOC_CTX *mem_ctx,
89 int argc, const char **argv)
91 printf ("(*) This command is not currently implemented.\n");
92 return WERR_OK;
94 #endif
96 /***********************************************************************
97 * Get printer information
99 static WERROR cmd_spoolss_open_printer_ex(struct cli_state *cli,
100 TALLOC_CTX *mem_ctx,
101 int argc, const char **argv)
103 WERROR werror;
104 fstring printername;
105 fstring servername, user;
106 POLICY_HND hnd;
108 if (argc != 2) {
109 printf("Usage: %s <printername>\n", argv[0]);
110 return WERR_OK;
113 if (!cli)
114 return WERR_GENERAL_FAILURE;
116 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
117 strupper_m(servername);
118 fstrcpy(user, cli->user_name);
119 fstrcpy(printername, argv[1]);
121 /* Open the printer handle */
123 werror = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
124 "", PRINTER_ALL_ACCESS,
125 servername, user, &hnd);
127 if (W_ERROR_IS_OK(werror)) {
128 printf("Printer %s opened successfully\n", printername);
129 werror = cli_spoolss_close_printer(cli, mem_ctx, &hnd);
131 if (!W_ERROR_IS_OK(werror)) {
132 printf("Error closing printer handle! (%s)\n",
133 get_dos_error_msg(werror));
137 return werror;
141 /****************************************************************************
142 printer info level 0 display function
143 ****************************************************************************/
144 static void display_print_info_0(PRINTER_INFO_0 *i0)
146 fstring name = "";
147 fstring servername = "";
149 if (!i0)
150 return;
152 rpcstr_pull(name, i0->printername.buffer, sizeof(name), -1, STR_TERMINATE);
154 rpcstr_pull(servername, i0->servername.buffer, sizeof(servername), -1,STR_TERMINATE);
156 printf("\tprintername:[%s]\n", name);
157 printf("\tservername:[%s]\n", servername);
158 printf("\tcjobs:[0x%x]\n", i0->cjobs);
159 printf("\ttotal_jobs:[0x%x]\n", i0->total_jobs);
161 printf("\t:date: [%d]-[%d]-[%d] (%d)\n", i0->year, i0->month,
162 i0->day, i0->dayofweek);
163 printf("\t:time: [%d]-[%d]-[%d]-[%d]\n", i0->hour, i0->minute,
164 i0->second, i0->milliseconds);
166 printf("\tglobal_counter:[0x%x]\n", i0->global_counter);
167 printf("\ttotal_pages:[0x%x]\n", i0->total_pages);
169 printf("\tmajorversion:[0x%x]\n", i0->major_version);
170 printf("\tbuildversion:[0x%x]\n", i0->build_version);
172 printf("\tunknown7:[0x%x]\n", i0->unknown7);
173 printf("\tunknown8:[0x%x]\n", i0->unknown8);
174 printf("\tunknown9:[0x%x]\n", i0->unknown9);
175 printf("\tsession_counter:[0x%x]\n", i0->session_counter);
176 printf("\tunknown11:[0x%x]\n", i0->unknown11);
177 printf("\tprinter_errors:[0x%x]\n", i0->printer_errors);
178 printf("\tunknown13:[0x%x]\n", i0->unknown13);
179 printf("\tunknown14:[0x%x]\n", i0->unknown14);
180 printf("\tunknown15:[0x%x]\n", i0->unknown15);
181 printf("\tunknown16:[0x%x]\n", i0->unknown16);
182 printf("\tchange_id:[0x%x]\n", i0->change_id);
183 printf("\tunknown18:[0x%x]\n", i0->unknown18);
184 printf("\tstatus:[0x%x]\n", i0->status);
185 printf("\tunknown20:[0x%x]\n", i0->unknown20);
186 printf("\tc_setprinter:[0x%x]\n", i0->c_setprinter);
187 printf("\tunknown22:[0x%x]\n", i0->unknown22);
188 printf("\tunknown23:[0x%x]\n", i0->unknown23);
189 printf("\tunknown24:[0x%x]\n", i0->unknown24);
190 printf("\tunknown25:[0x%x]\n", i0->unknown25);
191 printf("\tunknown26:[0x%x]\n", i0->unknown26);
192 printf("\tunknown27:[0x%x]\n", i0->unknown27);
193 printf("\tunknown28:[0x%x]\n", i0->unknown28);
194 printf("\tunknown29:[0x%x]\n", i0->unknown29);
196 printf("\n");
199 /****************************************************************************
200 printer info level 1 display function
201 ****************************************************************************/
202 static void display_print_info_1(PRINTER_INFO_1 *i1)
204 fstring desc = "";
205 fstring name = "";
206 fstring comm = "";
208 rpcstr_pull(desc, i1->description.buffer, sizeof(desc), -1,
209 STR_TERMINATE);
211 rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
212 rpcstr_pull(comm, i1->comment.buffer, sizeof(comm), -1, STR_TERMINATE);
214 printf("\tflags:[0x%x]\n", i1->flags);
215 printf("\tname:[%s]\n", name);
216 printf("\tdescription:[%s]\n", desc);
217 printf("\tcomment:[%s]\n", comm);
219 printf("\n");
222 /****************************************************************************
223 printer info level 2 display function
224 ****************************************************************************/
225 static void display_print_info_2(PRINTER_INFO_2 *i2)
227 fstring servername = "";
228 fstring printername = "";
229 fstring sharename = "";
230 fstring portname = "";
231 fstring drivername = "";
232 fstring comment = "";
233 fstring location = "";
234 fstring sepfile = "";
235 fstring printprocessor = "";
236 fstring datatype = "";
237 fstring parameters = "";
239 rpcstr_pull(servername, i2->servername.buffer,sizeof(servername), -1, STR_TERMINATE);
241 rpcstr_pull(printername, i2->printername.buffer,sizeof(printername), -1, STR_TERMINATE);
243 rpcstr_pull(sharename, i2->sharename.buffer,sizeof(sharename), -1, STR_TERMINATE);
245 rpcstr_pull(portname, i2->portname.buffer,sizeof(portname), -1, STR_TERMINATE);
247 rpcstr_pull(drivername, i2->drivername.buffer,sizeof(drivername), -1, STR_TERMINATE);
249 rpcstr_pull(comment, i2->comment.buffer,sizeof(comment), -1, STR_TERMINATE);
251 rpcstr_pull(location, i2->location.buffer,sizeof(location), -1, STR_TERMINATE);
253 rpcstr_pull(sepfile, i2->sepfile.buffer,sizeof(sepfile), -1, STR_TERMINATE);
255 rpcstr_pull(printprocessor, i2->printprocessor.buffer,sizeof(printprocessor), -1, STR_TERMINATE);
257 rpcstr_pull(datatype, i2->datatype.buffer,sizeof(datatype), -1, STR_TERMINATE);
259 rpcstr_pull(parameters, i2->parameters.buffer,sizeof(parameters), -1, STR_TERMINATE);
261 printf("\tservername:[%s]\n", servername);
262 printf("\tprintername:[%s]\n", printername);
263 printf("\tsharename:[%s]\n", sharename);
264 printf("\tportname:[%s]\n", portname);
265 printf("\tdrivername:[%s]\n", drivername);
266 printf("\tcomment:[%s]\n", comment);
267 printf("\tlocation:[%s]\n", location);
268 printf("\tsepfile:[%s]\n", sepfile);
269 printf("\tprintprocessor:[%s]\n", printprocessor);
270 printf("\tdatatype:[%s]\n", datatype);
271 printf("\tparameters:[%s]\n", parameters);
272 printf("\tattributes:[0x%x]\n", i2->attributes);
273 printf("\tpriority:[0x%x]\n", i2->priority);
274 printf("\tdefaultpriority:[0x%x]\n", i2->defaultpriority);
275 printf("\tstarttime:[0x%x]\n", i2->starttime);
276 printf("\tuntiltime:[0x%x]\n", i2->untiltime);
277 printf("\tstatus:[0x%x]\n", i2->status);
278 printf("\tcjobs:[0x%x]\n", i2->cjobs);
279 printf("\taverageppm:[0x%x]\n", i2->averageppm);
281 if (i2->secdesc)
282 display_sec_desc(i2->secdesc);
284 printf("\n");
287 /****************************************************************************
288 printer info level 3 display function
289 ****************************************************************************/
290 static void display_print_info_3(PRINTER_INFO_3 *i3)
292 printf("\tflags:[0x%x]\n", i3->flags);
294 display_sec_desc(i3->secdesc);
296 printf("\n");
299 /* Enumerate printers */
301 static WERROR cmd_spoolss_enum_printers(struct cli_state *cli,
302 TALLOC_CTX *mem_ctx,
303 int argc, const char **argv)
305 WERROR result;
306 uint32 info_level = 1;
307 PRINTER_INFO_CTR ctr;
308 uint32 i = 0, num_printers, needed;
309 fstring name;
311 if (argc > 3)
313 printf("Usage: %s [level] [name]\n", argv[0]);
314 return WERR_OK;
317 if (argc >= 2)
318 info_level = atoi(argv[1]);
320 if (argc == 3)
321 fstrcpy(name, argv[2]);
322 else {
323 slprintf(name, sizeof(name)-1, "\\\\%s", cli->desthost);
324 strupper_m(name);
327 /* Enumerate printers -- Should we enumerate types other
328 than PRINTER_ENUM_LOCAL? Maybe accept as a parameter? --jerry */
330 ZERO_STRUCT(ctr);
332 result = cli_spoolss_enum_printers(
333 cli, mem_ctx, 0, &needed, name, PRINTER_ENUM_LOCAL,
334 info_level, &num_printers, &ctr);
336 if (W_ERROR_V(result) == ERRinsufficientbuffer)
337 result = cli_spoolss_enum_printers(
338 cli, mem_ctx, needed, NULL, name, PRINTER_ENUM_LOCAL,
339 info_level, &num_printers, &ctr);
341 if (W_ERROR_IS_OK(result)) {
343 if (!num_printers) {
344 printf ("No printers returned.\n");
345 goto done;
348 for (i = 0; i < num_printers; i++) {
349 switch(info_level) {
350 case 0:
351 display_print_info_0(&ctr.printers_0[i]);
352 break;
353 case 1:
354 display_print_info_1(&ctr.printers_1[i]);
355 break;
356 case 2:
357 display_print_info_2(&ctr.printers_2[i]);
358 break;
359 case 3:
360 display_print_info_3(&ctr.printers_3[i]);
361 break;
362 default:
363 printf("unknown info level %d\n", info_level);
364 goto done;
368 done:
370 return result;
373 /****************************************************************************
374 port info level 1 display function
375 ****************************************************************************/
376 static void display_port_info_1(PORT_INFO_1 *i1)
378 fstring buffer;
380 rpcstr_pull(buffer, i1->port_name.buffer, sizeof(buffer), -1, STR_TERMINATE);
381 printf("\tPort Name:\t[%s]\n", buffer);
384 /****************************************************************************
385 port info level 2 display function
386 ****************************************************************************/
387 static void display_port_info_2(PORT_INFO_2 *i2)
389 fstring buffer;
391 rpcstr_pull(buffer, i2->port_name.buffer, sizeof(buffer), -1, STR_TERMINATE);
392 printf("\tPort Name:\t[%s]\n", buffer);
393 rpcstr_pull(buffer, i2->monitor_name.buffer, sizeof(buffer), -1, STR_TERMINATE);
395 printf("\tMonitor Name:\t[%s]\n", buffer);
396 rpcstr_pull(buffer, i2->description.buffer, sizeof(buffer), -1, STR_TERMINATE);
398 printf("\tDescription:\t[%s]\n", buffer);
399 printf("\tPort Type:\t" );
400 if ( i2->port_type ) {
401 int comma = 0; /* hack */
402 printf( "[" );
403 if ( i2->port_type & PORT_TYPE_READ ) {
404 printf( "Read" );
405 comma = 1;
407 if ( i2->port_type & PORT_TYPE_WRITE ) {
408 printf( "%sWrite", comma ? ", " : "" );
409 comma = 1;
411 /* These two have slightly different interpretations
412 on 95/98/ME but I'm disregarding that for now */
413 if ( i2->port_type & PORT_TYPE_REDIRECTED ) {
414 printf( "%sRedirected", comma ? ", " : "" );
415 comma = 1;
417 if ( i2->port_type & PORT_TYPE_NET_ATTACHED ) {
418 printf( "%sNet-Attached", comma ? ", " : "" );
420 printf( "]\n" );
421 } else {
422 printf( "[Unset]\n" );
424 printf("\tReserved:\t[%d]\n", i2->reserved);
425 printf("\n");
428 /* Enumerate ports */
430 static WERROR cmd_spoolss_enum_ports(struct cli_state *cli,
431 TALLOC_CTX *mem_ctx, int argc,
432 const char **argv)
434 WERROR result;
435 uint32 needed, info_level = 1;
436 PORT_INFO_CTR ctr;
437 uint32 returned;
439 if (argc > 2) {
440 printf("Usage: %s [level]\n", argv[0]);
441 return WERR_OK;
444 if (argc == 2)
445 info_level = atoi(argv[1]);
447 /* Enumerate ports */
449 ZERO_STRUCT(ctr);
451 result = cli_spoolss_enum_ports(cli, mem_ctx, 0, &needed, info_level,
452 &returned, &ctr);
454 if (W_ERROR_V(result) == ERRinsufficientbuffer)
455 result = cli_spoolss_enum_ports(cli, mem_ctx, needed, NULL,
456 info_level, &returned, &ctr);
458 if (W_ERROR_IS_OK(result)) {
459 int i;
461 for (i = 0; i < returned; i++) {
462 switch (info_level) {
463 case 1:
464 display_port_info_1(&ctr.port.info_1[i]);
465 break;
466 case 2:
467 display_port_info_2(&ctr.port.info_2[i]);
468 break;
469 default:
470 printf("unknown info level %d\n", info_level);
471 break;
476 return result;
479 /***********************************************************************
480 * Set printer comment - use a level2 set.
482 static WERROR cmd_spoolss_setprinter(struct cli_state *cli,
483 TALLOC_CTX *mem_ctx,
484 int argc, const char **argv)
486 POLICY_HND pol;
487 WERROR result;
488 uint32 needed;
489 uint32 info_level = 2;
490 BOOL opened_hnd = False;
491 PRINTER_INFO_CTR ctr;
492 fstring printername,
493 servername,
494 user,
495 comment;
497 if (argc == 1 || argc > 3) {
498 printf("Usage: %s printername comment\n", argv[0]);
500 return WERR_OK;
503 /* Open a printer handle */
504 if (argc == 3) {
505 fstrcpy(comment, argv[2]);
508 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
509 strupper_m(servername);
510 fstrcpy(printername, argv[1]);
511 fstrcpy(user, cli->user_name);
513 /* get a printer handle */
514 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
515 PRINTER_ALL_ACCESS, servername,
516 user, &pol);
518 if (!W_ERROR_IS_OK(result))
519 goto done;
521 opened_hnd = True;
523 /* Get printer info */
524 result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed, &pol, info_level, &ctr);
526 if (W_ERROR_V(result) == ERRinsufficientbuffer)
527 result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, info_level, &ctr);
529 if (!W_ERROR_IS_OK(result))
530 goto done;
533 /* Modify the comment. */
534 init_unistr(&ctr.printers_2->comment, comment);
535 ctr.printers_2->devmode = NULL;
536 ctr.printers_2->secdesc = NULL;
538 result = cli_spoolss_setprinter(cli, mem_ctx, &pol, info_level, &ctr, 0);
539 if (W_ERROR_IS_OK(result))
540 printf("Success in setting comment.\n");
542 done:
543 if (opened_hnd)
544 cli_spoolss_close_printer(cli, mem_ctx, &pol);
546 return result;
549 /***********************************************************************
550 * Set printer name - use a level2 set.
552 static WERROR cmd_spoolss_setprintername(struct cli_state *cli,
553 TALLOC_CTX *mem_ctx,
554 int argc, const char **argv)
556 POLICY_HND pol;
557 WERROR result;
558 uint32 needed;
559 uint32 info_level = 2;
560 BOOL opened_hnd = False;
561 PRINTER_INFO_CTR ctr;
562 fstring printername,
563 servername,
564 user,
565 new_printername;
567 if (argc == 1 || argc > 3) {
568 printf("Usage: %s printername new_printername\n", argv[0]);
570 return WERR_OK;
573 /* Open a printer handle */
574 if (argc == 3) {
575 fstrcpy(new_printername, argv[2]);
578 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
579 strupper_m(servername);
580 fstrcpy(printername, argv[1]);
581 fstrcpy(user, cli->user_name);
583 /* get a printer handle */
584 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
585 PRINTER_ALL_ACCESS, servername,
586 user, &pol);
588 if (!W_ERROR_IS_OK(result))
589 goto done;
591 opened_hnd = True;
593 /* Get printer info */
594 result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed, &pol, info_level, &ctr);
596 if (W_ERROR_V(result) == ERRinsufficientbuffer)
597 result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, info_level, &ctr);
599 if (!W_ERROR_IS_OK(result))
600 goto done;
603 /* Modify the printername. */
604 init_unistr(&ctr.printers_2->printername, new_printername);
605 ctr.printers_2->devmode = NULL;
606 ctr.printers_2->secdesc = NULL;
608 result = cli_spoolss_setprinter(cli, mem_ctx, &pol, info_level, &ctr, 0);
609 if (W_ERROR_IS_OK(result))
610 printf("Success in setting printername.\n");
612 done:
613 if (opened_hnd)
614 cli_spoolss_close_printer(cli, mem_ctx, &pol);
616 return result;
619 /***********************************************************************
620 * Get printer information
622 static WERROR cmd_spoolss_getprinter(struct cli_state *cli,
623 TALLOC_CTX *mem_ctx,
624 int argc, const char **argv)
626 POLICY_HND pol;
627 WERROR result;
628 uint32 info_level = 1;
629 BOOL opened_hnd = False;
630 PRINTER_INFO_CTR ctr;
631 fstring printername,
632 servername,
633 user;
634 uint32 needed;
636 if (argc == 1 || argc > 3) {
637 printf("Usage: %s <printername> [level]\n", argv[0]);
638 return WERR_OK;
641 /* Open a printer handle */
642 if (argc == 3) {
643 info_level = atoi(argv[2]);
646 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
647 strupper_m(servername);
648 slprintf(printername, sizeof(printername)-1, "%s\\%s", servername, argv[1]);
649 fstrcpy(user, cli->user_name);
651 /* get a printer handle */
653 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
654 "", MAXIMUM_ALLOWED_ACCESS,
655 servername, user, &pol);
657 if (!W_ERROR_IS_OK(result))
658 goto done;
660 opened_hnd = True;
662 /* Get printer info */
664 result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed,
665 &pol, info_level, &ctr);
667 if (W_ERROR_V(result) == ERRinsufficientbuffer)
668 result = cli_spoolss_getprinter(
669 cli, mem_ctx, needed, NULL, &pol, info_level, &ctr);
671 if (!W_ERROR_IS_OK(result))
672 goto done;
674 /* Display printer info */
676 switch (info_level) {
677 case 0:
678 display_print_info_0(ctr.printers_0);
679 break;
680 case 1:
681 display_print_info_1(ctr.printers_1);
682 break;
683 case 2:
684 display_print_info_2(ctr.printers_2);
685 break;
686 case 3:
687 display_print_info_3(ctr.printers_3);
688 break;
689 default:
690 printf("unknown info level %d\n", info_level);
691 break;
694 done:
695 if (opened_hnd)
696 cli_spoolss_close_printer(cli, mem_ctx, &pol);
698 return result;
701 static void display_reg_value(REGISTRY_VALUE value)
703 pstring text;
705 switch(value.type) {
706 case REG_DWORD:
707 printf("%s: REG_DWORD: 0x%08x\n", value.valuename,
708 *((uint32 *) value.data_p));
709 break;
710 case REG_SZ:
711 rpcstr_pull(text, value.data_p, sizeof(text), value.size,
712 STR_TERMINATE);
713 printf("%s: REG_SZ: %s\n", value.valuename, text);
714 break;
715 case REG_BINARY:
716 printf("%s: REG_BINARY: unknown length value not displayed\n",
717 value.valuename);
718 break;
719 case REG_MULTI_SZ: {
720 uint16 *curstr = (uint16 *) value.data_p;
721 uint8 *start = value.data_p;
722 printf("%s: REG_MULTI_SZ:\n", value.valuename);
723 while ((*curstr != 0) &&
724 ((uint8 *) curstr < start + value.size)) {
725 rpcstr_pull(text, curstr, sizeof(text), -1,
726 STR_TERMINATE);
727 printf(" %s\n", text);
728 curstr += strlen(text) + 1;
731 break;
732 default:
733 printf("%s: unknown type %d\n", value.valuename, value.type);
738 /***********************************************************************
739 * Get printer data
741 static WERROR cmd_spoolss_getprinterdata(struct cli_state *cli,
742 TALLOC_CTX *mem_ctx,
743 int argc, const char **argv)
745 POLICY_HND pol;
746 WERROR result;
747 BOOL opened_hnd = False;
748 fstring printername,
749 servername,
750 user;
751 uint32 needed;
752 const char *valuename;
753 REGISTRY_VALUE value;
755 if (argc != 3) {
756 printf("Usage: %s <printername> <valuename>\n", argv[0]);
757 printf("<printername> of . queries print server\n");
758 return WERR_OK;
760 valuename = argv[2];
762 /* Open a printer handle */
764 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
765 strupper_m(servername);
766 if (strncmp(argv[1], ".", sizeof(".")) == 0)
767 fstrcpy(printername, servername);
768 else
769 slprintf(printername, sizeof(servername)-1, "%s\\%s",
770 servername, argv[1]);
771 fstrcpy(user, cli->user_name);
773 /* get a printer handle */
775 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
776 "", MAXIMUM_ALLOWED_ACCESS,
777 servername, user, &pol);
779 if (!W_ERROR_IS_OK(result))
780 goto done;
782 opened_hnd = True;
784 /* Get printer info */
786 result = cli_spoolss_getprinterdata(cli, mem_ctx, 0, &needed,
787 &pol, valuename, &value);
789 if (W_ERROR_V(result) == ERRmoredata)
790 result = cli_spoolss_getprinterdata(
791 cli, mem_ctx, needed, NULL, &pol, valuename, &value);
793 if (!W_ERROR_IS_OK(result))
794 goto done;
796 /* Display printer data */
798 fstrcpy(value.valuename, valuename);
799 display_reg_value(value);
802 done:
803 if (opened_hnd)
804 cli_spoolss_close_printer(cli, mem_ctx, &pol);
806 return result;
809 /***********************************************************************
810 * Get printer data
812 static WERROR cmd_spoolss_getprinterdataex(struct cli_state *cli,
813 TALLOC_CTX *mem_ctx,
814 int argc, const char **argv)
816 POLICY_HND pol;
817 WERROR result;
818 BOOL opened_hnd = False;
819 fstring printername,
820 servername,
821 user;
822 uint32 needed;
823 const char *valuename, *keyname;
824 REGISTRY_VALUE value;
826 if (argc != 4) {
827 printf("Usage: %s <printername> <keyname> <valuename>\n",
828 argv[0]);
829 printf("<printername> of . queries print server\n");
830 return WERR_OK;
832 valuename = argv[3];
833 keyname = argv[2];
835 /* Open a printer handle */
837 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
838 strupper_m(servername);
839 if (strncmp(argv[1], ".", sizeof(".")) == 0)
840 fstrcpy(printername, servername);
841 else
842 slprintf(printername, sizeof(printername)-1, "%s\\%s",
843 servername, argv[1]);
844 fstrcpy(user, cli->user_name);
846 /* get a printer handle */
848 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
849 "", MAXIMUM_ALLOWED_ACCESS,
850 servername, user, &pol);
852 if (!W_ERROR_IS_OK(result))
853 goto done;
855 opened_hnd = True;
857 /* Get printer info */
859 result = cli_spoolss_getprinterdataex(cli, mem_ctx, 0, &needed,
860 &pol, keyname, valuename,
861 &value);
863 if (W_ERROR_V(result) == ERRmoredata)
864 result = cli_spoolss_getprinterdataex(cli, mem_ctx, needed,
865 NULL, &pol, keyname,
866 valuename, &value);
868 if (!W_ERROR_IS_OK(result))
869 goto done;
871 /* Display printer data */
873 fstrcpy(value.valuename, valuename);
874 display_reg_value(value);
877 done:
878 if (opened_hnd)
879 cli_spoolss_close_printer(cli, mem_ctx, &pol);
881 return result;
884 /****************************************************************************
885 printer info level 0 display function
886 ****************************************************************************/
887 static void display_print_driver_1(DRIVER_INFO_1 *i1)
889 fstring name;
890 if (i1 == NULL)
891 return;
893 rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
895 printf ("Printer Driver Info 1:\n");
896 printf ("\tDriver Name: [%s]\n\n", name);
898 return;
901 /****************************************************************************
902 printer info level 1 display function
903 ****************************************************************************/
904 static void display_print_driver_2(DRIVER_INFO_2 *i1)
906 fstring name;
907 fstring architecture;
908 fstring driverpath;
909 fstring datafile;
910 fstring configfile;
911 if (i1 == NULL)
912 return;
914 rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
915 rpcstr_pull(architecture, i1->architecture.buffer, sizeof(architecture), -1, STR_TERMINATE);
916 rpcstr_pull(driverpath, i1->driverpath.buffer, sizeof(driverpath), -1, STR_TERMINATE);
917 rpcstr_pull(datafile, i1->datafile.buffer, sizeof(datafile), -1, STR_TERMINATE);
918 rpcstr_pull(configfile, i1->configfile.buffer, sizeof(configfile), -1, STR_TERMINATE);
920 printf ("Printer Driver Info 2:\n");
921 printf ("\tVersion: [%x]\n", i1->version);
922 printf ("\tDriver Name: [%s]\n", name);
923 printf ("\tArchitecture: [%s]\n", architecture);
924 printf ("\tDriver Path: [%s]\n", driverpath);
925 printf ("\tDatafile: [%s]\n", datafile);
926 printf ("\tConfigfile: [%s]\n\n", configfile);
928 return;
931 /****************************************************************************
932 printer info level 2 display function
933 ****************************************************************************/
934 static void display_print_driver_3(DRIVER_INFO_3 *i1)
936 fstring name = "";
937 fstring architecture = "";
938 fstring driverpath = "";
939 fstring datafile = "";
940 fstring configfile = "";
941 fstring helpfile = "";
942 fstring dependentfiles = "";
943 fstring monitorname = "";
944 fstring defaultdatatype = "";
946 int length=0;
947 BOOL valid = True;
949 if (i1 == NULL)
950 return;
952 rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
953 rpcstr_pull(architecture, i1->architecture.buffer, sizeof(architecture), -1, STR_TERMINATE);
954 rpcstr_pull(driverpath, i1->driverpath.buffer, sizeof(driverpath), -1, STR_TERMINATE);
955 rpcstr_pull(datafile, i1->datafile.buffer, sizeof(datafile), -1, STR_TERMINATE);
956 rpcstr_pull(configfile, i1->configfile.buffer, sizeof(configfile), -1, STR_TERMINATE);
957 rpcstr_pull(helpfile, i1->helpfile.buffer, sizeof(helpfile), -1, STR_TERMINATE);
958 rpcstr_pull(monitorname, i1->monitorname.buffer, sizeof(monitorname), -1, STR_TERMINATE);
959 rpcstr_pull(defaultdatatype, i1->defaultdatatype.buffer, sizeof(defaultdatatype), -1, STR_TERMINATE);
961 printf ("Printer Driver Info 3:\n");
962 printf ("\tVersion: [%x]\n", i1->version);
963 printf ("\tDriver Name: [%s]\n",name);
964 printf ("\tArchitecture: [%s]\n", architecture);
965 printf ("\tDriver Path: [%s]\n", driverpath);
966 printf ("\tDatafile: [%s]\n", datafile);
967 printf ("\tConfigfile: [%s]\n", configfile);
968 printf ("\tHelpfile: [%s]\n\n", helpfile);
970 while (valid)
972 rpcstr_pull(dependentfiles, i1->dependentfiles+length, sizeof(dependentfiles), -1, STR_TERMINATE);
974 length+=strlen(dependentfiles)+1;
976 if (strlen(dependentfiles) > 0)
978 printf ("\tDependentfiles: [%s]\n", dependentfiles);
980 else
982 valid = False;
986 printf ("\n");
988 printf ("\tMonitorname: [%s]\n", monitorname);
989 printf ("\tDefaultdatatype: [%s]\n\n", defaultdatatype);
991 return;
994 /***********************************************************************
995 * Get printer information
997 static WERROR cmd_spoolss_getdriver(struct cli_state *cli,
998 TALLOC_CTX *mem_ctx,
999 int argc, const char **argv)
1001 POLICY_HND pol;
1002 WERROR werror;
1003 uint32 info_level = 3;
1004 BOOL opened_hnd = False;
1005 PRINTER_DRIVER_CTR ctr;
1006 fstring printername,
1007 servername,
1008 user;
1009 uint32 i;
1010 BOOL success = False;
1012 if ((argc == 1) || (argc > 3))
1014 printf("Usage: %s <printername> [level]\n", argv[0]);
1015 return WERR_OK;
1018 /* get the arguments need to open the printer handle */
1019 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
1020 strupper_m(servername);
1021 fstrcpy(user, cli->user_name);
1022 fstrcpy(printername, argv[1]);
1023 if (argc == 3)
1024 info_level = atoi(argv[2]);
1026 /* Open a printer handle */
1028 werror = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
1029 PRINTER_ACCESS_USE,
1030 servername, user, &pol);
1032 if (!W_ERROR_IS_OK(werror)) {
1033 printf("Error opening printer handle for %s!\n", printername);
1034 return werror;
1037 opened_hnd = True;
1039 /* loop through and print driver info level for each architecture */
1041 for (i=0; archi_table[i].long_archi!=NULL; i++) {
1042 uint32 needed;
1044 werror = cli_spoolss_getprinterdriver(
1045 cli, mem_ctx, 0, &needed, &pol, info_level,
1046 archi_table[i].long_archi, archi_table[i].version,
1047 &ctr);
1049 if (W_ERROR_V(werror) == ERRinsufficientbuffer) {
1050 werror = cli_spoolss_getprinterdriver(
1051 cli, mem_ctx, needed, NULL, &pol, info_level,
1052 archi_table[i].long_archi, archi_table[i].version,
1053 &ctr);
1056 if (!W_ERROR_IS_OK(werror))
1057 continue;
1059 /* need at least one success */
1061 success = True;
1063 printf ("\n[%s]\n", archi_table[i].long_archi);
1065 switch (info_level) {
1066 case 1:
1067 display_print_driver_1 (ctr.info1);
1068 break;
1069 case 2:
1070 display_print_driver_2 (ctr.info2);
1071 break;
1072 case 3:
1073 display_print_driver_3 (ctr.info3);
1074 break;
1075 default:
1076 printf("unknown info level %d\n", info_level);
1077 break;
1081 /* Cleanup */
1083 if (opened_hnd)
1084 cli_spoolss_close_printer (cli, mem_ctx, &pol);
1086 if ( success )
1087 werror = WERR_OK;
1089 return werror;
1092 /***********************************************************************
1093 * Get printer information
1095 static WERROR cmd_spoolss_enum_drivers(struct cli_state *cli,
1096 TALLOC_CTX *mem_ctx,
1097 int argc, const char **argv)
1099 WERROR werror;
1100 uint32 info_level = 1;
1101 PRINTER_DRIVER_CTR ctr;
1102 uint32 i, j,
1103 returned;
1105 if (argc > 2)
1107 printf("Usage: enumdrivers [level]\n");
1108 return WERR_OK;
1111 if (argc == 2)
1112 info_level = atoi(argv[1]);
1115 /* loop through and print driver info level for each architecture */
1116 for (i=0; archi_table[i].long_archi!=NULL; i++)
1118 uint32 needed;
1120 werror = cli_spoolss_enumprinterdrivers(
1121 cli, mem_ctx, 0, &needed, info_level,
1122 archi_table[i].long_archi, &returned, &ctr);
1124 if (W_ERROR_V(werror) == ERRinsufficientbuffer)
1125 werror = cli_spoolss_enumprinterdrivers(
1126 cli, mem_ctx, needed, NULL, info_level,
1127 archi_table[i].long_archi, &returned, &ctr);
1129 if (returned == 0)
1130 continue;
1132 if (!W_ERROR_IS_OK(werror)) {
1133 printf ("Error getting driver for environment [%s] - %d\n",
1134 archi_table[i].long_archi, W_ERROR_V(werror));
1135 continue;
1138 printf ("\n[%s]\n", archi_table[i].long_archi);
1139 switch (info_level)
1142 case 1:
1143 for (j=0; j < returned; j++) {
1144 display_print_driver_1 (&(ctr.info1[j]));
1146 break;
1147 case 2:
1148 for (j=0; j < returned; j++) {
1149 display_print_driver_2 (&(ctr.info2[j]));
1151 break;
1152 case 3:
1153 for (j=0; j < returned; j++) {
1154 display_print_driver_3 (&(ctr.info3[j]));
1156 break;
1157 default:
1158 printf("unknown info level %d\n", info_level);
1159 break;
1163 return werror;
1166 /****************************************************************************
1167 printer info level 1 display function
1168 ****************************************************************************/
1169 static void display_printdriverdir_1(DRIVER_DIRECTORY_1 *i1)
1171 fstring name;
1172 if (i1 == NULL)
1173 return;
1175 rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
1177 printf ("\tDirectory Name:[%s]\n", name);
1180 /***********************************************************************
1181 * Get printer driver directory information
1183 static WERROR cmd_spoolss_getdriverdir(struct cli_state *cli,
1184 TALLOC_CTX *mem_ctx,
1185 int argc, const char **argv)
1187 WERROR result;
1188 fstring env;
1189 DRIVER_DIRECTORY_CTR ctr;
1190 uint32 needed;
1192 if (argc > 2) {
1193 printf("Usage: %s [environment]\n", argv[0]);
1194 return WERR_OK;
1197 /* Get the arguments need to open the printer handle */
1199 if (argc == 2)
1200 fstrcpy (env, argv[1]);
1201 else
1202 fstrcpy (env, "Windows NT x86");
1204 /* Get the directory. Only use Info level 1 */
1206 result = cli_spoolss_getprinterdriverdir(
1207 cli, mem_ctx, 0, &needed, 1, env, &ctr);
1209 if (W_ERROR_V(result) == ERRinsufficientbuffer)
1210 result = cli_spoolss_getprinterdriverdir(
1211 cli, mem_ctx, needed, NULL, 1, env, &ctr);
1213 if (W_ERROR_IS_OK(result))
1214 display_printdriverdir_1(ctr.info1);
1216 return result;
1219 /*******************************************************************************
1220 set the version and environment fields of a DRIVER_INFO_3 struct
1221 ******************************************************************************/
1222 void set_drv_info_3_env (DRIVER_INFO_3 *info, const char *arch)
1225 int i;
1227 for (i=0; archi_table[i].long_archi != NULL; i++)
1229 if (strcmp(arch, archi_table[i].short_archi) == 0)
1231 info->version = archi_table[i].version;
1232 init_unistr (&info->architecture, archi_table[i].long_archi);
1233 break;
1237 if (archi_table[i].long_archi == NULL)
1239 DEBUG(0, ("set_drv_info_3_env: Unknown arch [%s]\n", arch));
1242 return;
1246 /**************************************************************************
1247 wrapper for strtok to get the next parameter from a delimited list.
1248 Needed to handle the empty parameter string denoted by "NULL"
1249 *************************************************************************/
1250 static char* get_driver_3_param (const char* str, const char* delim, UNISTR* dest)
1252 char *ptr;
1254 /* get the next token */
1255 ptr = strtok(str, delim);
1257 /* a string of 'NULL' is used to represent an empty
1258 parameter because two consecutive delimiters
1259 will not return an empty string. See man strtok(3)
1260 for details */
1261 if (ptr && (StrCaseCmp(ptr, "NULL") == 0))
1262 ptr = NULL;
1264 if (dest != NULL)
1265 init_unistr(dest, ptr);
1267 return ptr;
1270 /********************************************************************************
1271 fill in the members of a DRIVER_INFO_3 struct using a character
1272 string in the form of
1273 <Long Printer Name>:<Driver File Name>:<Data File Name>:\
1274 <Config File Name>:<Help File Name>:<Language Monitor Name>:\
1275 <Default Data Type>:<Comma Separated list of Files>
1276 *******************************************************************************/
1277 static BOOL init_drv_info_3_members (
1278 TALLOC_CTX *mem_ctx,
1279 DRIVER_INFO_3 *info,
1280 const char *args
1283 char *str, *str2;
1284 uint32 len, i;
1286 /* fill in the UNISTR fields */
1287 str = get_driver_3_param (args, ":", &info->name);
1288 str = get_driver_3_param (NULL, ":", &info->driverpath);
1289 str = get_driver_3_param (NULL, ":", &info->datafile);
1290 str = get_driver_3_param (NULL, ":", &info->configfile);
1291 str = get_driver_3_param (NULL, ":", &info->helpfile);
1292 str = get_driver_3_param (NULL, ":", &info->monitorname);
1293 str = get_driver_3_param (NULL, ":", &info->defaultdatatype);
1295 /* <Comma Separated List of Dependent Files> */
1296 str2 = get_driver_3_param (NULL, ":", NULL); /* save the beginning of the string */
1297 str = str2;
1299 /* begin to strip out each filename */
1300 str = strtok(str, ",");
1301 len = 0;
1302 while (str != NULL)
1304 /* keep a cumlative count of the str lengths */
1305 len += strlen(str)+1;
1306 str = strtok(NULL, ",");
1309 /* allocate the space; add one extra slot for a terminating NULL.
1310 Each filename is NULL terminated and the end contains a double
1311 NULL */
1312 if ((info->dependentfiles=(uint16*)talloc(mem_ctx, (len+1)*sizeof(uint16))) == NULL)
1314 DEBUG(0,("init_drv_info_3_members: Unable to malloc memory for dependenfiles\n"));
1315 return False;
1317 for (i=0; i<len; i++)
1319 SSVAL(&info->dependentfiles[i], 0, str2[i]);
1321 info->dependentfiles[len] = '\0';
1323 return True;
1327 static WERROR cmd_spoolss_addprinterdriver(struct cli_state *cli,
1328 TALLOC_CTX *mem_ctx,
1329 int argc, const char **argv)
1331 WERROR result;
1332 uint32 level = 3;
1333 PRINTER_DRIVER_CTR ctr;
1334 DRIVER_INFO_3 info3;
1335 const char *arch;
1336 fstring driver_name;
1338 /* parse the command arguements */
1339 if (argc != 3 && argc != 4)
1341 printf ("Usage: %s <Environment> \\\n", argv[0]);
1342 printf ("\t<Long Printer Name>:<Driver File Name>:<Data File Name>:\\\n");
1343 printf ("\t<Config File Name>:<Help File Name>:<Language Monitor Name>:\\\n");
1344 printf ("\t<Default Data Type>:<Comma Separated list of Files> \\\n");
1345 printf ("\t[version]\n");
1347 return WERR_OK;
1350 /* Fill in the DRIVER_INFO_3 struct */
1351 ZERO_STRUCT(info3);
1352 if (!(arch = cmd_spoolss_get_short_archi(argv[1])))
1354 printf ("Error Unknown architechture [%s]\n", argv[1]);
1355 return WERR_INVALID_PARAM;
1357 else
1358 set_drv_info_3_env(&info3, arch);
1360 if (!init_drv_info_3_members(mem_ctx, &info3, argv[2]))
1362 printf ("Error Invalid parameter list - %s.\n", argv[2]);
1363 return WERR_INVALID_PARAM;
1366 /* if printer driver version specified, override the default version
1367 * used by the architecture. This allows installation of Windows
1368 * 2000 (version 3) printer drivers. */
1369 if (argc == 4)
1371 info3.version = atoi(argv[3]);
1375 ctr.info3 = &info3;
1376 result = cli_spoolss_addprinterdriver (cli, mem_ctx, level, &ctr);
1378 if (W_ERROR_IS_OK(result)) {
1379 rpcstr_pull(driver_name, info3.name.buffer,
1380 sizeof(driver_name), -1, STR_TERMINATE);
1381 printf ("Printer Driver %s successfully installed.\n",
1382 driver_name);
1385 return result;
1389 static WERROR cmd_spoolss_addprinterex(struct cli_state *cli,
1390 TALLOC_CTX *mem_ctx,
1391 int argc, const char **argv)
1393 WERROR result;
1394 uint32 level = 2;
1395 PRINTER_INFO_CTR ctr;
1396 PRINTER_INFO_2 info2;
1397 fstring servername;
1399 /* parse the command arguements */
1400 if (argc != 5)
1402 printf ("Usage: %s <name> <shared name> <driver> <port>\n", argv[0]);
1403 return WERR_OK;
1406 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
1407 strupper_m(servername);
1409 /* Fill in the DRIVER_INFO_2 struct */
1410 ZERO_STRUCT(info2);
1411 #if 0 /* JERRY */
1412 init_unistr( &info2.servername, servername);
1413 #endif
1414 init_unistr( &info2.printername, argv[1]);
1415 init_unistr( &info2.sharename, argv[2]);
1416 init_unistr( &info2.drivername, argv[3]);
1417 init_unistr( &info2.portname, argv[4]);
1418 init_unistr( &info2.comment, "Created by rpcclient");
1419 init_unistr( &info2.printprocessor, "winprint");
1420 init_unistr( &info2.datatype, "RAW");
1421 info2.devmode = NULL;
1422 info2.secdesc = NULL;
1423 info2.attributes = PRINTER_ATTRIBUTE_SHARED;
1424 info2.priority = 0;
1425 info2.defaultpriority = 0;
1426 info2.starttime = 0;
1427 info2.untiltime = 0;
1429 /* These three fields must not be used by AddPrinter()
1430 as defined in the MS Platform SDK documentation..
1431 --jerry
1432 info2.status = 0;
1433 info2.cjobs = 0;
1434 info2.averageppm = 0;
1437 ctr.printers_2 = &info2;
1438 result = cli_spoolss_addprinterex (cli, mem_ctx, level, &ctr);
1440 if (W_ERROR_IS_OK(result))
1441 printf ("Printer %s successfully installed.\n", argv[1]);
1443 return result;
1446 static WERROR cmd_spoolss_setdriver(struct cli_state *cli,
1447 TALLOC_CTX *mem_ctx,
1448 int argc, const char **argv)
1450 POLICY_HND pol;
1451 WERROR result;
1452 uint32 level = 2;
1453 BOOL opened_hnd = False;
1454 PRINTER_INFO_CTR ctr;
1455 PRINTER_INFO_2 info2;
1456 fstring servername,
1457 printername,
1458 user;
1459 uint32 needed;
1461 /* parse the command arguements */
1462 if (argc != 3)
1464 printf ("Usage: %s <printer> <driver>\n", argv[0]);
1465 return WERR_OK;
1468 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
1469 strupper_m(servername);
1470 slprintf(printername, sizeof(printername)-1, "%s\\%s", servername, argv[1]);
1471 fstrcpy(user, cli->user_name);
1473 /* Get a printer handle */
1475 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
1476 PRINTER_ALL_ACCESS,
1477 servername, user, &pol);
1479 if (!W_ERROR_IS_OK(result))
1480 goto done;
1482 opened_hnd = True;
1484 /* Get printer info */
1486 ZERO_STRUCT (info2);
1487 ctr.printers_2 = &info2;
1489 result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed,
1490 &pol, level, &ctr);
1492 if (W_ERROR_V(result) == ERRinsufficientbuffer)
1493 result = cli_spoolss_getprinter(
1494 cli, mem_ctx, needed, NULL, &pol, level, &ctr);
1496 if (!W_ERROR_IS_OK(result)) {
1497 printf ("Unable to retrieve printer information!\n");
1498 goto done;
1501 /* Set the printer driver */
1503 init_unistr(&ctr.printers_2->drivername, argv[2]);
1505 result = cli_spoolss_setprinter(cli, mem_ctx, &pol, level, &ctr, 0);
1507 if (!W_ERROR_IS_OK(result)) {
1508 printf("SetPrinter call failed!\n");
1509 goto done;;
1512 printf("Succesfully set %s to driver %s.\n", argv[1], argv[2]);
1514 done:
1515 /* Cleanup */
1517 if (opened_hnd)
1518 cli_spoolss_close_printer(cli, mem_ctx, &pol);
1520 return result;
1524 static WERROR cmd_spoolss_deletedriver(struct cli_state *cli,
1525 TALLOC_CTX *mem_ctx,
1526 int argc, const char **argv)
1528 WERROR result;
1529 fstring servername;
1530 int i;
1532 /* parse the command arguements */
1533 if (argc != 2)
1535 printf ("Usage: %s <driver>\n", argv[0]);
1536 return WERR_OK;
1539 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
1540 strupper_m(servername);
1542 /* delete the driver for all architectures */
1543 for (i=0; archi_table[i].long_archi; i++)
1545 /* make the call to remove the driver */
1546 result = cli_spoolss_deleteprinterdriver(
1547 cli, mem_ctx, archi_table[i].long_archi, argv[1]);
1549 if ( !W_ERROR_IS_OK(result) ) {
1550 if ( !W_ERROR_EQUAL(result, WERR_UNKNOWN_PRINTER_DRIVER) ) {
1551 printf ("Failed to remove driver %s for arch [%s] - error 0x%x!\n",
1552 argv[1], archi_table[i].long_archi,
1553 W_ERROR_V(result));
1556 else
1558 printf ("Driver %s removed for arch [%s].\n", argv[1],
1559 archi_table[i].long_archi);
1563 return result;
1566 static WERROR cmd_spoolss_getprintprocdir(struct cli_state *cli,
1567 TALLOC_CTX *mem_ctx,
1568 int argc, const char **argv)
1570 WERROR result;
1571 char *servername = NULL, *environment = NULL;
1572 fstring procdir;
1573 uint32 needed;
1575 /* parse the command arguements */
1576 if (argc > 2) {
1577 printf ("Usage: %s [environment]\n", argv[0]);
1578 return WERR_OK;
1581 if (asprintf(&servername, "\\\\%s", cli->desthost) < 0)
1582 return WERR_NOMEM;
1583 strupper_m(servername);
1585 if (asprintf(&environment, "%s", (argc == 2) ? argv[1] :
1586 PRINTER_DRIVER_ARCHITECTURE) < 0) {
1587 SAFE_FREE(servername);
1588 return WERR_NOMEM;
1591 result = cli_spoolss_getprintprocessordirectory(
1592 cli, mem_ctx, 0, &needed, servername, environment, procdir);
1594 if (W_ERROR_V(result) == ERRinsufficientbuffer)
1595 result = cli_spoolss_getprintprocessordirectory(
1596 cli, mem_ctx, needed, NULL, servername, environment,
1597 procdir);
1599 if (W_ERROR_IS_OK(result))
1600 printf("%s\n", procdir);
1602 SAFE_FREE(servername);
1603 SAFE_FREE(environment);
1605 return result;
1608 /* Add a form */
1610 static WERROR cmd_spoolss_addform(struct cli_state *cli, TALLOC_CTX *mem_ctx,
1611 int argc, const char **argv)
1613 POLICY_HND handle;
1614 WERROR werror;
1615 char *servername = NULL, *printername = NULL;
1616 FORM form;
1617 BOOL got_handle = False;
1619 /* Parse the command arguements */
1621 if (argc != 3) {
1622 printf ("Usage: %s <printer> <formname>\n", argv[0]);
1623 return WERR_OK;
1626 /* Get a printer handle */
1628 asprintf(&servername, "\\\\%s", cli->desthost);
1629 strupper_m(servername);
1630 asprintf(&printername, "%s\\%s", servername, argv[1]);
1632 werror = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
1633 MAXIMUM_ALLOWED_ACCESS,
1634 servername, cli->user_name, &handle);
1636 if (!W_ERROR_IS_OK(werror))
1637 goto done;
1639 got_handle = True;
1641 /* Dummy up some values for the form data */
1643 form.flags = FORM_USER;
1644 form.size_x = form.size_y = 100;
1645 form.left = 0;
1646 form.top = 10;
1647 form.right = 20;
1648 form.bottom = 30;
1650 init_unistr2(&form.name, argv[2], UNI_STR_TERMINATE);
1652 /* Add the form */
1655 werror = cli_spoolss_addform(cli, mem_ctx, &handle, 1, &form);
1657 done:
1658 if (got_handle)
1659 cli_spoolss_close_printer(cli, mem_ctx, &handle);
1661 SAFE_FREE(servername);
1662 SAFE_FREE(printername);
1664 return werror;
1667 /* Set a form */
1669 static WERROR cmd_spoolss_setform(struct cli_state *cli, TALLOC_CTX *mem_ctx,
1670 int argc, const char **argv)
1672 POLICY_HND handle;
1673 WERROR werror;
1674 char *servername = NULL, *printername = NULL;
1675 FORM form;
1676 BOOL got_handle = False;
1678 /* Parse the command arguements */
1680 if (argc != 3) {
1681 printf ("Usage: %s <printer> <formname>\n", argv[0]);
1682 return WERR_OK;
1685 /* Get a printer handle */
1687 asprintf(&servername, "\\\\%s", cli->desthost);
1688 strupper_m(servername);
1689 asprintf(&printername, "%s\\%s", servername, argv[1]);
1691 werror = cli_spoolss_open_printer_ex(
1692 cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
1693 servername, cli->user_name, &handle);
1695 if (!W_ERROR_IS_OK(werror))
1696 goto done;
1698 got_handle = True;
1700 /* Dummy up some values for the form data */
1702 form.flags = FORM_PRINTER;
1703 form.size_x = form.size_y = 100;
1704 form.left = 0;
1705 form.top = 1000;
1706 form.right = 2000;
1707 form.bottom = 3000;
1709 init_unistr2(&form.name, argv[2], UNI_STR_TERMINATE);
1711 /* Set the form */
1713 werror = cli_spoolss_setform(cli, mem_ctx, &handle, 1, argv[2], &form);
1715 done:
1716 if (got_handle)
1717 cli_spoolss_close_printer(cli, mem_ctx, &handle);
1719 SAFE_FREE(servername);
1720 SAFE_FREE(printername);
1722 return werror;
1725 /* Get a form */
1727 static WERROR cmd_spoolss_getform(struct cli_state *cli, TALLOC_CTX *mem_ctx,
1728 int argc, const char **argv)
1730 POLICY_HND handle;
1731 WERROR werror;
1732 char *servername = NULL, *printername = NULL;
1733 FORM_1 form;
1734 BOOL got_handle = False;
1735 uint32 needed;
1737 /* Parse the command arguements */
1739 if (argc != 3) {
1740 printf ("Usage: %s <printer> <formname>\n", argv[0]);
1741 return WERR_OK;
1744 /* Get a printer handle */
1746 asprintf(&servername, "\\\\%s", cli->desthost);
1747 strupper_m(servername);
1748 asprintf(&printername, "%s\\%s", servername, argv[1]);
1750 werror = cli_spoolss_open_printer_ex(
1751 cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
1752 servername, cli->user_name, &handle);
1754 if (!W_ERROR_IS_OK(werror))
1755 goto done;
1757 got_handle = True;
1759 /* Set the form */
1761 werror = cli_spoolss_getform(cli, mem_ctx, 0, &needed,
1762 &handle, argv[2], 1, &form);
1764 if (W_ERROR_V(werror) == ERRinsufficientbuffer)
1765 werror = cli_spoolss_getform(cli, mem_ctx, needed, NULL,
1766 &handle, argv[2], 1, &form);
1768 if (!W_ERROR_IS_OK(werror))
1769 goto done;
1771 printf("width: %d\n", form.width);
1772 printf("length: %d\n", form.length);
1773 printf("left: %d\n", form.left);
1774 printf("top: %d\n", form.top);
1775 printf("right: %d\n", form.right);
1776 printf("bottom: %d\n", form.bottom);
1778 done:
1779 if (got_handle)
1780 cli_spoolss_close_printer(cli, mem_ctx, &handle);
1782 SAFE_FREE(servername);
1783 SAFE_FREE(printername);
1785 return werror;
1788 /* Delete a form */
1790 static WERROR cmd_spoolss_deleteform(struct cli_state *cli,
1791 TALLOC_CTX *mem_ctx, int argc,
1792 const char **argv)
1794 POLICY_HND handle;
1795 WERROR werror;
1796 char *servername = NULL, *printername = NULL;
1797 BOOL got_handle = False;
1799 /* Parse the command arguements */
1801 if (argc != 3) {
1802 printf ("Usage: %s <printer> <formname>\n", argv[0]);
1803 return WERR_OK;
1806 /* Get a printer handle */
1808 asprintf(&servername, "\\\\%s", cli->desthost);
1809 strupper_m(servername);
1810 asprintf(&printername, "%s\\%s", servername, argv[1]);
1812 werror = cli_spoolss_open_printer_ex(
1813 cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
1814 servername, cli->user_name, &handle);
1816 if (!W_ERROR_IS_OK(werror))
1817 goto done;
1819 got_handle = True;
1821 /* Delete the form */
1823 werror = cli_spoolss_deleteform(cli, mem_ctx, &handle, argv[2]);
1825 done:
1826 if (got_handle)
1827 cli_spoolss_close_printer(cli, mem_ctx, &handle);
1829 SAFE_FREE(servername);
1830 SAFE_FREE(printername);
1832 return werror;
1835 /* Enumerate forms */
1837 static WERROR cmd_spoolss_enum_forms(struct cli_state *cli,
1838 TALLOC_CTX *mem_ctx, int argc,
1839 const char **argv)
1841 POLICY_HND handle;
1842 WERROR werror;
1843 char *servername = NULL, *printername = NULL;
1844 BOOL got_handle = False;
1845 uint32 needed, num_forms, level = 1, i;
1846 FORM_1 *forms;
1848 /* Parse the command arguements */
1850 if (argc != 2) {
1851 printf ("Usage: %s <printer>\n", argv[0]);
1852 return WERR_OK;
1855 /* Get a printer handle */
1857 asprintf(&servername, "\\\\%s", cli->desthost);
1858 strupper_m(servername);
1859 asprintf(&printername, "%s\\%s", servername, argv[1]);
1861 werror = cli_spoolss_open_printer_ex(
1862 cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
1863 servername, cli->user_name, &handle);
1865 if (!W_ERROR_IS_OK(werror))
1866 goto done;
1868 got_handle = True;
1870 /* Enumerate forms */
1872 werror = cli_spoolss_enumforms(
1873 cli, mem_ctx, 0, &needed, &handle, level, &num_forms, &forms);
1875 if (W_ERROR_V(werror) == ERRinsufficientbuffer)
1876 werror = cli_spoolss_enumforms(
1877 cli, mem_ctx, needed, NULL, &handle, level,
1878 &num_forms, &forms);
1880 if (!W_ERROR_IS_OK(werror))
1881 goto done;
1883 /* Display output */
1885 for (i = 0; i < num_forms; i++) {
1886 fstring form_name;
1888 if (forms[i].name.buffer)
1889 rpcstr_pull(form_name, forms[i].name.buffer,
1890 sizeof(form_name), -1, STR_TERMINATE);
1892 printf("%s\n", form_name);
1895 done:
1896 if (got_handle)
1897 cli_spoolss_close_printer(cli, mem_ctx, &handle);
1899 SAFE_FREE(servername);
1900 SAFE_FREE(printername);
1902 return werror;
1905 static WERROR cmd_spoolss_setprinterdata(struct cli_state *cli,
1906 TALLOC_CTX *mem_ctx,
1907 int argc, const char **argv)
1909 WERROR result;
1910 uint32 needed;
1911 fstring servername, printername, user;
1912 POLICY_HND pol;
1913 BOOL opened_hnd = False;
1914 PRINTER_INFO_CTR ctr;
1915 PRINTER_INFO_0 info;
1916 REGISTRY_VALUE value;
1917 UNISTR2 data;
1919 /* parse the command arguements */
1920 if (argc != 4) {
1921 printf ("Usage: %s <printer> <value> <data>\n", argv[0]);
1922 return WERR_OK;
1925 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
1926 strupper_m(servername);
1927 slprintf(printername, sizeof(servername)-1, "%s\\%s", servername, argv[1]);
1928 fstrcpy(user, cli->user_name);
1930 /* get a printer handle */
1931 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
1932 MAXIMUM_ALLOWED_ACCESS, servername,
1933 user, &pol);
1934 if (!W_ERROR_IS_OK(result))
1935 goto done;
1937 opened_hnd = True;
1939 ctr.printers_0 = &info;
1941 result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed,
1942 &pol, 0, &ctr);
1944 if (W_ERROR_V(result) == ERRinsufficientbuffer)
1945 result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, 0, &ctr);
1947 if (!W_ERROR_IS_OK(result))
1948 goto done;
1950 printf("%s\n", timestring(True));
1951 printf("\tchange_id (before set)\t:[0x%x]\n", info.change_id);
1953 /* Set the printer data */
1955 init_unistr2(&data, argv[3], UNI_STR_TERMINATE);
1956 fstrcpy(value.valuename, argv[2]);
1957 value.type = REG_SZ;
1958 value.size = data.uni_str_len * 2;
1959 value.data_p = talloc_memdup(mem_ctx, data.buffer, value.size);
1961 result = cli_spoolss_setprinterdata(cli, mem_ctx, &pol, &value);
1963 if (!W_ERROR_IS_OK(result)) {
1964 printf ("Unable to set [%s=%s]!\n", argv[2], argv[3]);
1965 goto done;
1967 printf("\tSetPrinterData succeeded [%s: %s]\n", argv[2], argv[3]);
1969 result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed, &pol, 0, &ctr);
1971 if (W_ERROR_V(result) == ERRinsufficientbuffer)
1972 result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, 0, &ctr);
1974 if (!W_ERROR_IS_OK(result))
1975 goto done;
1977 printf("%s\n", timestring(True));
1978 printf("\tchange_id (after set)\t:[0x%x]\n", info.change_id);
1980 done:
1981 /* cleanup */
1982 if (opened_hnd)
1983 cli_spoolss_close_printer(cli, mem_ctx, &pol);
1985 return result;
1988 static void display_job_info_1(JOB_INFO_1 *job)
1990 fstring username = "", document = "", text_status = "";
1992 rpcstr_pull(username, job->username.buffer,
1993 sizeof(username), -1, STR_TERMINATE);
1995 rpcstr_pull(document, job->document.buffer,
1996 sizeof(document), -1, STR_TERMINATE);
1998 rpcstr_pull(text_status, job->text_status.buffer,
1999 sizeof(text_status), -1, STR_TERMINATE);
2001 printf("%d: jobid[%d]: %s %s %s %d/%d pages\n", job->position, job->jobid,
2002 username, document, text_status, job->pagesprinted,
2003 job->totalpages);
2006 static void display_job_info_2(JOB_INFO_2 *job)
2008 fstring username = "", document = "", text_status = "";
2010 rpcstr_pull(username, job->username.buffer,
2011 sizeof(username), -1, STR_TERMINATE);
2013 rpcstr_pull(document, job->document.buffer,
2014 sizeof(document), -1, STR_TERMINATE);
2016 rpcstr_pull(text_status, job->text_status.buffer,
2017 sizeof(text_status), -1, STR_TERMINATE);
2019 printf("%d: jobid[%d]: %s %s %s %d/%d pages, %d bytes\n", job->position, job->jobid,
2020 username, document, text_status, job->pagesprinted,
2021 job->totalpages, job->size);
2024 /* Enumerate jobs */
2026 static WERROR cmd_spoolss_enum_jobs(struct cli_state *cli,
2027 TALLOC_CTX *mem_ctx, int argc,
2028 const char **argv)
2030 WERROR result;
2031 uint32 needed, level = 1, num_jobs, i;
2032 BOOL got_hnd = False;
2033 pstring printername;
2034 fstring servername, user;
2035 POLICY_HND hnd;
2036 JOB_INFO_CTR ctr;
2038 if (argc < 2 || argc > 3) {
2039 printf("Usage: %s printername [level]\n", argv[0]);
2040 return WERR_OK;
2043 if (argc == 3)
2044 level = atoi(argv[2]);
2046 /* Open printer handle */
2048 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
2049 strupper_m(servername);
2050 fstrcpy(user, cli->user_name);
2051 slprintf(printername, sizeof(servername)-1, "\\\\%s\\", cli->desthost);
2052 strupper_m(printername);
2053 pstrcat(printername, argv[1]);
2055 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
2056 "", MAXIMUM_ALLOWED_ACCESS,
2057 servername, user, &hnd);
2059 if (!W_ERROR_IS_OK(result))
2060 goto done;
2062 got_hnd = True;
2064 /* Enumerate ports */
2066 result = cli_spoolss_enumjobs(
2067 cli, mem_ctx, 0, &needed, &hnd, level, 0, 1000,
2068 &num_jobs, &ctr);
2070 if (W_ERROR_V(result) == ERRinsufficientbuffer)
2071 result = cli_spoolss_enumjobs(
2072 cli, mem_ctx, needed, NULL, &hnd, level, 0,
2073 1000, &num_jobs, &ctr);
2075 if (!W_ERROR_IS_OK(result))
2076 goto done;
2078 for (i = 0; i < num_jobs; i++) {
2079 switch(level) {
2080 case 1:
2081 display_job_info_1(&ctr.job.job_info_1[i]);
2082 break;
2083 case 2:
2084 display_job_info_2(&ctr.job.job_info_2[i]);
2085 break;
2086 default:
2087 d_printf("unknown info level %d\n", level);
2088 break;
2092 done:
2093 if (got_hnd)
2094 cli_spoolss_close_printer(cli, mem_ctx, &hnd);
2096 return result;
2099 /* enumerate data */
2101 static WERROR cmd_spoolss_enum_data( struct cli_state *cli,
2102 TALLOC_CTX *mem_ctx, int argc,
2103 const char **argv)
2105 WERROR result;
2106 uint32 i=0, val_needed, data_needed;
2107 BOOL got_hnd = False;
2108 pstring printername;
2109 fstring servername, user;
2110 POLICY_HND hnd;
2112 if (argc != 2) {
2113 printf("Usage: %s printername\n", argv[0]);
2114 return WERR_OK;
2117 /* Open printer handle */
2119 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
2120 strupper_m(servername);
2121 fstrcpy(user, cli->user_name);
2122 slprintf(printername, sizeof(printername)-1, "\\\\%s\\", cli->desthost);
2123 strupper_m(printername);
2124 pstrcat(printername, argv[1]);
2126 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
2127 "", MAXIMUM_ALLOWED_ACCESS,
2128 servername, user, &hnd);
2130 if (!W_ERROR_IS_OK(result))
2131 goto done;
2133 got_hnd = True;
2135 /* Enumerate data */
2137 result = cli_spoolss_enumprinterdata(cli, mem_ctx, &hnd, i, 0, 0,
2138 &val_needed, &data_needed,
2139 NULL);
2140 while (W_ERROR_IS_OK(result)) {
2141 REGISTRY_VALUE value;
2142 result = cli_spoolss_enumprinterdata(
2143 cli, mem_ctx, &hnd, i++, val_needed,
2144 data_needed, 0, 0, &value);
2145 if (W_ERROR_IS_OK(result))
2146 display_reg_value(value);
2148 if (W_ERROR_V(result) == ERRnomoreitems)
2149 result = W_ERROR(ERRsuccess);
2151 done:
2152 if (got_hnd)
2153 cli_spoolss_close_printer(cli, mem_ctx, &hnd);
2155 return result;
2158 /* enumerate data for a given key */
2160 static WERROR cmd_spoolss_enum_data_ex( struct cli_state *cli,
2161 TALLOC_CTX *mem_ctx, int argc,
2162 const char **argv)
2164 WERROR result;
2165 uint32 needed, i;
2166 BOOL got_hnd = False;
2167 pstring printername;
2168 fstring servername, user;
2169 const char *keyname = NULL;
2170 POLICY_HND hnd;
2171 REGVAL_CTR ctr;
2173 if (argc != 3) {
2174 printf("Usage: %s printername <keyname>\n", argv[0]);
2175 return WERR_OK;
2178 keyname = argv[2];
2180 /* Open printer handle */
2182 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
2183 strupper_m(servername);
2184 fstrcpy(user, cli->user_name);
2185 slprintf(printername, sizeof(printername)-1, "\\\\%s\\", cli->desthost);
2186 strupper_m(printername);
2187 pstrcat(printername, argv[1]);
2189 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
2190 "", MAXIMUM_ALLOWED_ACCESS,
2191 servername, user, &hnd);
2193 if (!W_ERROR_IS_OK(result))
2194 goto done;
2196 got_hnd = True;
2198 /* Enumerate subkeys */
2200 result = cli_spoolss_enumprinterdataex(
2201 cli, mem_ctx, 0, &needed, &hnd, keyname, NULL);
2203 if (W_ERROR_V(result) == ERRmoredata)
2204 result = cli_spoolss_enumprinterdataex(
2205 cli, mem_ctx, needed, NULL, &hnd, keyname, &ctr);
2207 if (!W_ERROR_IS_OK(result))
2208 goto done;
2210 for (i=0; i < ctr.num_values; i++) {
2211 display_reg_value(*(ctr.values[i]));
2214 regval_ctr_destroy(&ctr);
2216 done:
2217 if (got_hnd)
2218 cli_spoolss_close_printer(cli, mem_ctx, &hnd);
2220 return result;
2223 /* enumerate subkeys */
2225 static WERROR cmd_spoolss_enum_printerkey( struct cli_state *cli,
2226 TALLOC_CTX *mem_ctx, int argc,
2227 const char **argv)
2229 WERROR result;
2230 uint32 needed, returned;
2231 BOOL got_hnd = False;
2232 pstring printername;
2233 fstring servername, user;
2234 const char *keyname = NULL;
2235 POLICY_HND hnd;
2236 uint16 *keylist = NULL, *curkey;
2238 if (argc < 2 || argc > 3) {
2239 printf("Usage: %s printername [keyname]\n", argv[0]);
2240 return WERR_OK;
2243 if (argc == 3)
2244 keyname = argv[2];
2245 else
2246 keyname = "";
2248 /* Open printer handle */
2250 slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
2251 strupper_m(servername);
2252 fstrcpy(user, cli->user_name);
2253 slprintf(printername, sizeof(printername)-1, "\\\\%s\\", cli->desthost);
2254 strupper_m(printername);
2255 pstrcat(printername, argv[1]);
2257 result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
2258 "", MAXIMUM_ALLOWED_ACCESS,
2259 servername, user, &hnd);
2261 if (!W_ERROR_IS_OK(result))
2262 goto done;
2264 got_hnd = True;
2266 /* Enumerate subkeys */
2268 result = cli_spoolss_enumprinterkey(
2269 cli, mem_ctx, 0, &needed, &hnd, keyname, NULL, NULL);
2271 if (W_ERROR_V(result) == ERRmoredata)
2272 result = cli_spoolss_enumprinterkey(
2273 cli, mem_ctx, needed, NULL, &hnd, keyname, &keylist,
2274 &returned);
2276 if (!W_ERROR_IS_OK(result))
2277 goto done;
2279 curkey = keylist;
2280 while (*curkey != 0) {
2281 pstring subkey;
2282 rpcstr_pull(subkey, curkey, sizeof(subkey), -1,
2283 STR_TERMINATE);
2284 printf("%s\n", subkey);
2285 curkey += strlen(subkey) + 1;
2288 safe_free(keylist);
2290 done:
2291 if (got_hnd)
2292 cli_spoolss_close_printer(cli, mem_ctx, &hnd);
2294 return result;
2297 static WERROR cmd_spoolss_rffpcnex(struct cli_state *cli,
2298 TALLOC_CTX *mem_ctx, int argc,
2299 const char **argv)
2301 fstring servername, printername;
2302 POLICY_HND hnd;
2303 BOOL got_hnd = False;
2304 WERROR result;
2305 SPOOL_NOTIFY_OPTION option;
2307 if (argc != 2) {
2308 printf("Usage: %s printername\n", argv[0]);
2309 result = WERR_OK;
2310 goto done;
2313 /* Open printer */
2315 slprintf(servername, sizeof(servername) - 1, "\\\\%s", cli->desthost);
2316 strupper_m(servername);
2318 slprintf(printername, sizeof(printername) - 1, "\\\\%s\\%s", cli->desthost,
2319 argv[1]);
2320 strupper_m(printername);
2322 result = cli_spoolss_open_printer_ex(
2323 cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
2324 servername, cli->user_name, &hnd);
2326 if (!W_ERROR_IS_OK(result)) {
2327 printf("Error opening %s\n", argv[1]);
2328 goto done;
2331 got_hnd = True;
2333 /* Create spool options */
2335 ZERO_STRUCT(option);
2337 option.version = 2;
2338 option.option_type_ptr = 1;
2339 option.count = option.ctr.count = 2;
2341 option.ctr.type = (SPOOL_NOTIFY_OPTION_TYPE *)talloc(
2342 mem_ctx, sizeof(SPOOL_NOTIFY_OPTION_TYPE) * 2);
2344 ZERO_STRUCT(option.ctr.type[0]);
2345 option.ctr.type[0].type = PRINTER_NOTIFY_TYPE;
2346 option.ctr.type[0].count = option.ctr.type[0].count2 = 1;
2347 option.ctr.type[0].fields_ptr = 1;
2348 option.ctr.type[0].fields[0] = PRINTER_NOTIFY_SERVER_NAME;
2350 ZERO_STRUCT(option.ctr.type[1]);
2351 option.ctr.type[1].type = JOB_NOTIFY_TYPE;
2352 option.ctr.type[1].count = option.ctr.type[1].count2 = 1;
2353 option.ctr.type[1].fields_ptr = 1;
2354 option.ctr.type[1].fields[0] = JOB_NOTIFY_PRINTER_NAME;
2356 /* Send rffpcnex */
2358 slprintf(servername, sizeof(servername) - 1, "\\\\%s", myhostname());
2359 strupper_m(servername);
2361 result = cli_spoolss_rffpcnex(
2362 cli, mem_ctx, &hnd, 0, 0, servername, 123, &option);
2364 if (!W_ERROR_IS_OK(result)) {
2365 printf("Error rffpcnex %s\n", argv[1]);
2366 goto done;
2369 done:
2370 if (got_hnd)
2371 cli_spoolss_close_printer(cli, mem_ctx, &hnd);
2373 return result;
2376 /* List of commands exported by this module */
2377 struct cmd_set spoolss_commands[] = {
2379 { "SPOOLSS" },
2381 { "adddriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addprinterdriver, PI_SPOOLSS, "Add a print driver", "" },
2382 { "addprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addprinterex, PI_SPOOLSS, "Add a printer", "" },
2383 { "deldriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deletedriver, PI_SPOOLSS, "Delete a printer driver", "" },
2384 { "enumdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_data, PI_SPOOLSS, "Enumerate printer data", "" },
2385 { "enumdataex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_data_ex, PI_SPOOLSS, "Enumerate printer data for a key", "" },
2386 { "enumkey", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_printerkey, PI_SPOOLSS, "Enumerate printer keys", "" },
2387 { "enumjobs", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_jobs, PI_SPOOLSS, "Enumerate print jobs", "" },
2388 { "enumports", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_ports, PI_SPOOLSS, "Enumerate printer ports", "" },
2389 { "enumdrivers", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_drivers, PI_SPOOLSS, "Enumerate installed printer drivers", "" },
2390 { "enumprinters", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_printers, PI_SPOOLSS, "Enumerate printers", "" },
2391 { "getdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinterdata, PI_SPOOLSS, "Get print driver data", "" },
2392 { "getdataex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinterdataex, PI_SPOOLSS, "Get printer driver data with keyname", ""},
2393 { "getdriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getdriver, PI_SPOOLSS, "Get print driver information", "" },
2394 { "getdriverdir", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getdriverdir, PI_SPOOLSS, "Get print driver upload directory", "" },
2395 { "getprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinter, PI_SPOOLSS, "Get printer info", "" },
2396 { "openprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_open_printer_ex, PI_SPOOLSS, "Open printer handle", "" },
2397 { "setdriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setdriver, PI_SPOOLSS, "Set printer driver", "" },
2398 { "getprintprocdir", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprintprocdir, PI_SPOOLSS, "Get print processor directory", "" },
2399 { "addform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addform, PI_SPOOLSS, "Add form", "" },
2400 { "setform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setform, PI_SPOOLSS, "Set form", "" },
2401 { "getform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getform, PI_SPOOLSS, "Get form", "" },
2402 { "deleteform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deleteform, PI_SPOOLSS, "Delete form", "" },
2403 { "enumforms", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_forms, PI_SPOOLSS, "Enumerate forms", "" },
2404 { "setprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprinter, PI_SPOOLSS, "Set printer comment", "" },
2405 { "setprintername", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprintername, PI_SPOOLSS, "Set printername", "" },
2406 { "setprinterdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprinterdata, PI_SPOOLSS, "Set REG_SZ printer data", "" },
2407 { "rffpcnex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_rffpcnex, PI_SPOOLSS, "Rffpcnex test", "" },
2409 { NULL }