mirror of
https://github.com/samba-team/samba.git
synced 2024-12-25 23:21:54 +03:00
2339 lines
62 KiB
C
2339 lines
62 KiB
C
/*
|
|
Unix SMB/CIFS implementation.
|
|
RPC pipe client
|
|
|
|
Copyright (C) Gerald Carter 2001
|
|
Copyright (C) Tim Potter 2000
|
|
Copyright (C) Andrew Tridgell 1992-1999
|
|
Copyright (C) Luke Kenneth Casson Leighton 1996-1999
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "rpcclient.h"
|
|
|
|
struct table_node {
|
|
const char *long_archi;
|
|
const char *short_archi;
|
|
int version;
|
|
};
|
|
|
|
static const struct table_node archi_table[]= {
|
|
|
|
{"Windows 4.0", "WIN40", 0 },
|
|
{"Windows NT x86", "W32X86", 2 },
|
|
{"Windows NT x86", "W32X86", 3 },
|
|
{"Windows NT R4000", "W32MIPS", 2 },
|
|
{"Windows NT Alpha_AXP", "W32ALPHA", 2 },
|
|
{"Windows NT PowerPC", "W32PPC", 2 },
|
|
{NULL, "", -1 }
|
|
};
|
|
|
|
/**
|
|
* @file
|
|
*
|
|
* rpcclient module for SPOOLSS rpc pipe.
|
|
*
|
|
* This generally just parses and checks command lines, and then calls
|
|
* a cli_spoolss function.
|
|
**/
|
|
|
|
/****************************************************************************
|
|
function to do the mapping between the long architecture name and
|
|
the short one.
|
|
****************************************************************************/
|
|
static const char *cmd_spoolss_get_short_archi(const char *long_archi)
|
|
{
|
|
int i=-1;
|
|
|
|
DEBUG(107,("Getting architecture dependant directory\n"));
|
|
do {
|
|
i++;
|
|
} while ( (archi_table[i].long_archi!=NULL ) &&
|
|
StrCaseCmp(long_archi, archi_table[i].long_archi) );
|
|
|
|
if (archi_table[i].long_archi==NULL) {
|
|
DEBUGADD(10,("Unknown architecture [%s] !\n", long_archi));
|
|
return NULL;
|
|
}
|
|
|
|
/* this might be client code - but shouldn't this be an fstrcpy etc? */
|
|
|
|
|
|
DEBUGADD(108,("index: [%d]\n", i));
|
|
DEBUGADD(108,("long architecture: [%s]\n", archi_table[i].long_archi));
|
|
DEBUGADD(108,("short architecture: [%s]\n", archi_table[i].short_archi));
|
|
|
|
return archi_table[i].short_archi;
|
|
}
|
|
|
|
#if 0
|
|
/**********************************************************************
|
|
* dummy function -- placeholder
|
|
*/
|
|
static WERROR cmd_spoolss_not_implemented(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
printf ("(*) This command is not currently implemented.\n");
|
|
return WERR_OK;
|
|
}
|
|
#endif
|
|
|
|
/***********************************************************************
|
|
* Get printer information
|
|
*/
|
|
static WERROR cmd_spoolss_open_printer_ex(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR werror;
|
|
fstring printername;
|
|
fstring servername, user;
|
|
POLICY_HND hnd;
|
|
|
|
if (argc != 2) {
|
|
printf("Usage: %s <printername>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (!cli)
|
|
return WERR_GENERAL_FAILURE;
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(user, cli->user_name);
|
|
fstrcpy(printername, argv[1]);
|
|
|
|
/* Open the printer handle */
|
|
|
|
werror = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", PRINTER_ALL_ACCESS,
|
|
servername, user, &hnd);
|
|
|
|
if (W_ERROR_IS_OK(werror)) {
|
|
printf("Printer %s opened successfully\n", printername);
|
|
werror = cli_spoolss_close_printer(cli, mem_ctx, &hnd);
|
|
|
|
if (!W_ERROR_IS_OK(werror)) {
|
|
printf("Error closing printer handle! (%s)\n",
|
|
get_dos_error_msg(werror));
|
|
}
|
|
}
|
|
|
|
return werror;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
printer info level 0 display function
|
|
****************************************************************************/
|
|
static void display_print_info_0(PRINTER_INFO_0 *i0)
|
|
{
|
|
fstring name = "";
|
|
fstring servername = "";
|
|
|
|
if (!i0)
|
|
return;
|
|
|
|
rpcstr_pull(name, i0->printername.buffer, sizeof(name), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(servername, i0->servername.buffer, sizeof(servername), -1,STR_TERMINATE);
|
|
|
|
printf("\tprintername:[%s]\n", name);
|
|
printf("\tservername:[%s]\n", servername);
|
|
printf("\tcjobs:[0x%x]\n", i0->cjobs);
|
|
printf("\ttotal_jobs:[0x%x]\n", i0->total_jobs);
|
|
|
|
printf("\t:date: [%d]-[%d]-[%d] (%d)\n", i0->year, i0->month,
|
|
i0->day, i0->dayofweek);
|
|
printf("\t:time: [%d]-[%d]-[%d]-[%d]\n", i0->hour, i0->minute,
|
|
i0->second, i0->milliseconds);
|
|
|
|
printf("\tglobal_counter:[0x%x]\n", i0->global_counter);
|
|
printf("\ttotal_pages:[0x%x]\n", i0->total_pages);
|
|
|
|
printf("\tmajorversion:[0x%x]\n", i0->major_version);
|
|
printf("\tbuildversion:[0x%x]\n", i0->build_version);
|
|
|
|
printf("\tunknown7:[0x%x]\n", i0->unknown7);
|
|
printf("\tunknown8:[0x%x]\n", i0->unknown8);
|
|
printf("\tunknown9:[0x%x]\n", i0->unknown9);
|
|
printf("\tsession_counter:[0x%x]\n", i0->session_counter);
|
|
printf("\tunknown11:[0x%x]\n", i0->unknown11);
|
|
printf("\tprinter_errors:[0x%x]\n", i0->printer_errors);
|
|
printf("\tunknown13:[0x%x]\n", i0->unknown13);
|
|
printf("\tunknown14:[0x%x]\n", i0->unknown14);
|
|
printf("\tunknown15:[0x%x]\n", i0->unknown15);
|
|
printf("\tunknown16:[0x%x]\n", i0->unknown16);
|
|
printf("\tchange_id:[0x%x]\n", i0->change_id);
|
|
printf("\tunknown18:[0x%x]\n", i0->unknown18);
|
|
printf("\tstatus:[0x%x]\n", i0->status);
|
|
printf("\tunknown20:[0x%x]\n", i0->unknown20);
|
|
printf("\tc_setprinter:[0x%x]\n", i0->c_setprinter);
|
|
printf("\tunknown22:[0x%x]\n", i0->unknown22);
|
|
printf("\tunknown23:[0x%x]\n", i0->unknown23);
|
|
printf("\tunknown24:[0x%x]\n", i0->unknown24);
|
|
printf("\tunknown25:[0x%x]\n", i0->unknown25);
|
|
printf("\tunknown26:[0x%x]\n", i0->unknown26);
|
|
printf("\tunknown27:[0x%x]\n", i0->unknown27);
|
|
printf("\tunknown28:[0x%x]\n", i0->unknown28);
|
|
printf("\tunknown29:[0x%x]\n", i0->unknown29);
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 1 display function
|
|
****************************************************************************/
|
|
static void display_print_info_1(PRINTER_INFO_1 *i1)
|
|
{
|
|
fstring desc = "";
|
|
fstring name = "";
|
|
fstring comm = "";
|
|
|
|
rpcstr_pull(desc, i1->description.buffer, sizeof(desc), -1,
|
|
STR_TERMINATE);
|
|
|
|
rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
|
|
rpcstr_pull(comm, i1->comment.buffer, sizeof(comm), -1, STR_TERMINATE);
|
|
|
|
printf("\tflags:[0x%x]\n", i1->flags);
|
|
printf("\tname:[%s]\n", name);
|
|
printf("\tdescription:[%s]\n", desc);
|
|
printf("\tcomment:[%s]\n", comm);
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 2 display function
|
|
****************************************************************************/
|
|
static void display_print_info_2(PRINTER_INFO_2 *i2)
|
|
{
|
|
fstring servername = "";
|
|
fstring printername = "";
|
|
fstring sharename = "";
|
|
fstring portname = "";
|
|
fstring drivername = "";
|
|
fstring comment = "";
|
|
fstring location = "";
|
|
fstring sepfile = "";
|
|
fstring printprocessor = "";
|
|
fstring datatype = "";
|
|
fstring parameters = "";
|
|
|
|
rpcstr_pull(servername, i2->servername.buffer,sizeof(servername), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(printername, i2->printername.buffer,sizeof(printername), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(sharename, i2->sharename.buffer,sizeof(sharename), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(portname, i2->portname.buffer,sizeof(portname), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(drivername, i2->drivername.buffer,sizeof(drivername), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(comment, i2->comment.buffer,sizeof(comment), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(location, i2->location.buffer,sizeof(location), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(sepfile, i2->sepfile.buffer,sizeof(sepfile), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(printprocessor, i2->printprocessor.buffer,sizeof(printprocessor), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(datatype, i2->datatype.buffer,sizeof(datatype), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(parameters, i2->parameters.buffer,sizeof(parameters), -1, STR_TERMINATE);
|
|
|
|
printf("\tservername:[%s]\n", servername);
|
|
printf("\tprintername:[%s]\n", printername);
|
|
printf("\tsharename:[%s]\n", sharename);
|
|
printf("\tportname:[%s]\n", portname);
|
|
printf("\tdrivername:[%s]\n", drivername);
|
|
printf("\tcomment:[%s]\n", comment);
|
|
printf("\tlocation:[%s]\n", location);
|
|
printf("\tsepfile:[%s]\n", sepfile);
|
|
printf("\tprintprocessor:[%s]\n", printprocessor);
|
|
printf("\tdatatype:[%s]\n", datatype);
|
|
printf("\tparameters:[%s]\n", parameters);
|
|
printf("\tattributes:[0x%x]\n", i2->attributes);
|
|
printf("\tpriority:[0x%x]\n", i2->priority);
|
|
printf("\tdefaultpriority:[0x%x]\n", i2->defaultpriority);
|
|
printf("\tstarttime:[0x%x]\n", i2->starttime);
|
|
printf("\tuntiltime:[0x%x]\n", i2->untiltime);
|
|
printf("\tstatus:[0x%x]\n", i2->status);
|
|
printf("\tcjobs:[0x%x]\n", i2->cjobs);
|
|
printf("\taverageppm:[0x%x]\n", i2->averageppm);
|
|
|
|
if (i2->secdesc)
|
|
display_sec_desc(i2->secdesc);
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 3 display function
|
|
****************************************************************************/
|
|
static void display_print_info_3(PRINTER_INFO_3 *i3)
|
|
{
|
|
printf("\tflags:[0x%x]\n", i3->flags);
|
|
|
|
display_sec_desc(i3->secdesc);
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
/* Enumerate printers */
|
|
|
|
static WERROR cmd_spoolss_enum_printers(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 info_level = 1;
|
|
PRINTER_INFO_CTR ctr;
|
|
uint32 i = 0, num_printers, needed;
|
|
fstring name;
|
|
|
|
if (argc > 3)
|
|
{
|
|
printf("Usage: %s [level] [name]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (argc == 2)
|
|
info_level = atoi(argv[1]);
|
|
|
|
if (argc == 3)
|
|
fstrcpy(name, argv[2]);
|
|
else {
|
|
slprintf(name, sizeof(name)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(name);
|
|
}
|
|
|
|
/* Enumerate printers -- Should we enumerate types other
|
|
than PRINTER_ENUM_LOCAL? Maybe accept as a parameter? --jerry */
|
|
|
|
ZERO_STRUCT(ctr);
|
|
|
|
result = cli_spoolss_enum_printers(
|
|
cli, mem_ctx, 0, &needed, name, PRINTER_ENUM_LOCAL,
|
|
info_level, &num_printers, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_enum_printers(
|
|
cli, mem_ctx, needed, NULL, name, PRINTER_ENUM_LOCAL,
|
|
info_level, &num_printers, &ctr);
|
|
|
|
if (W_ERROR_IS_OK(result)) {
|
|
|
|
if (!num_printers) {
|
|
printf ("No printers returned.\n");
|
|
goto done;
|
|
}
|
|
|
|
for (i = 0; i < num_printers; i++) {
|
|
switch(info_level) {
|
|
case 0:
|
|
display_print_info_0(&ctr.printers_0[i]);
|
|
break;
|
|
case 1:
|
|
display_print_info_1(&ctr.printers_1[i]);
|
|
break;
|
|
case 2:
|
|
display_print_info_2(&ctr.printers_2[i]);
|
|
break;
|
|
case 3:
|
|
display_print_info_3(&ctr.printers_3[i]);
|
|
break;
|
|
default:
|
|
printf("unknown info level %d\n", info_level);
|
|
goto done;
|
|
}
|
|
}
|
|
}
|
|
done:
|
|
|
|
return result;
|
|
}
|
|
|
|
/****************************************************************************
|
|
port info level 1 display function
|
|
****************************************************************************/
|
|
static void display_port_info_1(PORT_INFO_1 *i1)
|
|
{
|
|
fstring buffer;
|
|
|
|
rpcstr_pull(buffer, i1->port_name.buffer, sizeof(buffer), -1, STR_TERMINATE);
|
|
printf("\tPort Name:\t[%s]\n", buffer);
|
|
}
|
|
|
|
/****************************************************************************
|
|
port info level 2 display function
|
|
****************************************************************************/
|
|
static void display_port_info_2(PORT_INFO_2 *i2)
|
|
{
|
|
fstring buffer;
|
|
|
|
rpcstr_pull(buffer, i2->port_name.buffer, sizeof(buffer), -1, STR_TERMINATE);
|
|
printf("\tPort Name:\t[%s]\n", buffer);
|
|
rpcstr_pull(buffer, i2->monitor_name.buffer, sizeof(buffer), -1, STR_TERMINATE);
|
|
|
|
printf("\tMonitor Name:\t[%s]\n", buffer);
|
|
rpcstr_pull(buffer, i2->description.buffer, sizeof(buffer), -1, STR_TERMINATE);
|
|
|
|
printf("\tDescription:\t[%s]\n", buffer);
|
|
printf("\tPort Type:\t" );
|
|
if ( i2->port_type ) {
|
|
int comma = 0; /* hack */
|
|
printf( "[" );
|
|
if ( i2->port_type & PORT_TYPE_READ ) {
|
|
printf( "Read" );
|
|
comma = 1;
|
|
}
|
|
if ( i2->port_type & PORT_TYPE_WRITE ) {
|
|
printf( "%sWrite", comma ? ", " : "" );
|
|
comma = 1;
|
|
}
|
|
/* These two have slightly different interpretations
|
|
on 95/98/ME but I'm disregarding that for now */
|
|
if ( i2->port_type & PORT_TYPE_REDIRECTED ) {
|
|
printf( "%sRedirected", comma ? ", " : "" );
|
|
comma = 1;
|
|
}
|
|
if ( i2->port_type & PORT_TYPE_NET_ATTACHED ) {
|
|
printf( "%sNet-Attached", comma ? ", " : "" );
|
|
}
|
|
printf( "]\n" );
|
|
} else {
|
|
printf( "[Unset]\n" );
|
|
}
|
|
printf("\tReserved:\t[%d]\n", i2->reserved);
|
|
printf("\n");
|
|
}
|
|
|
|
/* Enumerate ports */
|
|
|
|
static WERROR cmd_spoolss_enum_ports(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 needed, info_level = 1;
|
|
PORT_INFO_CTR ctr;
|
|
int returned;
|
|
|
|
if (argc > 2) {
|
|
printf("Usage: %s [level]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (argc == 2)
|
|
info_level = atoi(argv[1]);
|
|
|
|
/* Enumerate ports */
|
|
|
|
ZERO_STRUCT(ctr);
|
|
|
|
result = cli_spoolss_enum_ports(cli, mem_ctx, 0, &needed, info_level,
|
|
&returned, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_enum_ports(cli, mem_ctx, needed, NULL,
|
|
info_level, &returned, &ctr);
|
|
|
|
if (W_ERROR_IS_OK(result)) {
|
|
int i;
|
|
|
|
for (i = 0; i < returned; i++) {
|
|
switch (info_level) {
|
|
case 1:
|
|
display_port_info_1(&ctr.port.info_1[i]);
|
|
break;
|
|
case 2:
|
|
display_port_info_2(&ctr.port.info_2[i]);
|
|
break;
|
|
default:
|
|
printf("unknown info level %d\n", info_level);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Set printer comment - use a level2 set.
|
|
*/
|
|
static WERROR cmd_spoolss_setprinter(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND pol;
|
|
WERROR result;
|
|
uint32 needed;
|
|
uint32 info_level = 2;
|
|
BOOL opened_hnd = False;
|
|
PRINTER_INFO_CTR ctr;
|
|
fstring printername,
|
|
servername,
|
|
user,
|
|
comment;
|
|
|
|
if (argc == 1 || argc > 3) {
|
|
printf("Usage: %s printername comment\n", argv[0]);
|
|
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Open a printer handle */
|
|
if (argc == 3) {
|
|
fstrcpy(comment, argv[2]);
|
|
}
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(printername, argv[1]);
|
|
fstrcpy(user, cli->user_name);
|
|
|
|
/* get a printer handle */
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
|
|
PRINTER_ALL_ACCESS, servername,
|
|
user, &pol);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
opened_hnd = True;
|
|
|
|
/* Get printer info */
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed, &pol, info_level, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, info_level, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
|
|
/* Modify the comment. */
|
|
init_unistr(&ctr.printers_2->comment, comment);
|
|
ctr.printers_2->devmode = NULL;
|
|
ctr.printers_2->secdesc = NULL;
|
|
|
|
result = cli_spoolss_setprinter(cli, mem_ctx, &pol, info_level, &ctr, 0);
|
|
if (W_ERROR_IS_OK(result))
|
|
printf("Success in setting comment.\n");
|
|
|
|
done:
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &pol);
|
|
|
|
return result;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Get printer information
|
|
*/
|
|
static WERROR cmd_spoolss_getprinter(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND pol;
|
|
WERROR result;
|
|
uint32 info_level = 1;
|
|
BOOL opened_hnd = False;
|
|
PRINTER_INFO_CTR ctr;
|
|
fstring printername,
|
|
servername,
|
|
user;
|
|
uint32 needed;
|
|
|
|
if (argc == 1 || argc > 3) {
|
|
printf("Usage: %s <printername> [level]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Open a printer handle */
|
|
if (argc == 3) {
|
|
info_level = atoi(argv[2]);
|
|
}
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
slprintf(printername, sizeof(printername)-1, "%s\\%s", servername, argv[1]);
|
|
fstrcpy(user, cli->user_name);
|
|
|
|
/* get a printer handle */
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &pol);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
opened_hnd = True;
|
|
|
|
/* Get printer info */
|
|
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed,
|
|
&pol, info_level, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprinter(
|
|
cli, mem_ctx, needed, NULL, &pol, info_level, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
/* Display printer info */
|
|
|
|
switch (info_level) {
|
|
case 0:
|
|
display_print_info_0(ctr.printers_0);
|
|
break;
|
|
case 1:
|
|
display_print_info_1(ctr.printers_1);
|
|
break;
|
|
case 2:
|
|
display_print_info_2(ctr.printers_2);
|
|
break;
|
|
case 3:
|
|
display_print_info_3(ctr.printers_3);
|
|
break;
|
|
default:
|
|
printf("unknown info level %d\n", info_level);
|
|
break;
|
|
}
|
|
|
|
done:
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &pol);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void display_reg_value(REGISTRY_VALUE value)
|
|
{
|
|
pstring text;
|
|
|
|
switch(value.type) {
|
|
case REG_DWORD:
|
|
printf("%s: REG_DWORD: 0x%08x\n", value.valuename,
|
|
*((uint32 *) value.data_p));
|
|
break;
|
|
case REG_SZ:
|
|
rpcstr_pull(text, value.data_p, sizeof(text), value.size,
|
|
STR_TERMINATE);
|
|
printf("%s: REG_SZ: %s\n", value.valuename, text);
|
|
break;
|
|
case REG_BINARY:
|
|
printf("%s: REG_BINARY: unknown length value not displayed\n",
|
|
value.valuename);
|
|
break;
|
|
case REG_MULTI_SZ: {
|
|
uint16 *curstr = (uint16 *) value.data_p;
|
|
uint8 *start = value.data_p;
|
|
printf("%s: REG_MULTI_SZ:\n", value.valuename);
|
|
while ((*curstr != 0) &&
|
|
((uint8 *) curstr < start + value.size)) {
|
|
rpcstr_pull(text, curstr, sizeof(text), -1,
|
|
STR_TERMINATE);
|
|
printf(" %s\n", text);
|
|
curstr += strlen(text) + 1;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
printf("%s: unknown type %d\n", value.valuename, value.type);
|
|
}
|
|
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Get printer data
|
|
*/
|
|
static WERROR cmd_spoolss_getprinterdata(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND pol;
|
|
WERROR result;
|
|
BOOL opened_hnd = False;
|
|
fstring printername,
|
|
servername,
|
|
user;
|
|
uint32 needed;
|
|
const char *valuename;
|
|
REGISTRY_VALUE value;
|
|
|
|
if (argc != 3) {
|
|
printf("Usage: %s <printername> <valuename>\n", argv[0]);
|
|
printf("<printername> of . queries print server\n");
|
|
return WERR_OK;
|
|
}
|
|
valuename = argv[2];
|
|
|
|
/* Open a printer handle */
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
if (strncmp(argv[1], ".", sizeof(".")) == 0)
|
|
fstrcpy(printername, servername);
|
|
else
|
|
slprintf(printername, sizeof(servername)-1, "%s\\%s",
|
|
servername, argv[1]);
|
|
fstrcpy(user, cli->user_name);
|
|
|
|
/* get a printer handle */
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &pol);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
opened_hnd = True;
|
|
|
|
/* Get printer info */
|
|
|
|
result = cli_spoolss_getprinterdata(cli, mem_ctx, 0, &needed,
|
|
&pol, valuename, &value);
|
|
|
|
if (W_ERROR_V(result) == ERRmoredata)
|
|
result = cli_spoolss_getprinterdata(
|
|
cli, mem_ctx, needed, NULL, &pol, valuename, &value);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
/* Display printer data */
|
|
|
|
fstrcpy(value.valuename, valuename);
|
|
display_reg_value(value);
|
|
|
|
|
|
done:
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &pol);
|
|
|
|
return result;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Get printer data
|
|
*/
|
|
static WERROR cmd_spoolss_getprinterdataex(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND pol;
|
|
WERROR result;
|
|
BOOL opened_hnd = False;
|
|
fstring printername,
|
|
servername,
|
|
user;
|
|
uint32 needed;
|
|
const char *valuename, *keyname;
|
|
REGISTRY_VALUE value;
|
|
|
|
if (argc != 4) {
|
|
printf("Usage: %s <printername> <keyname> <valuename>\n",
|
|
argv[0]);
|
|
printf("<printername> of . queries print server\n");
|
|
return WERR_OK;
|
|
}
|
|
valuename = argv[3];
|
|
keyname = argv[2];
|
|
|
|
/* Open a printer handle */
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
if (strncmp(argv[1], ".", sizeof(".")) == 0)
|
|
fstrcpy(printername, servername);
|
|
else
|
|
slprintf(printername, sizeof(printername)-1, "%s\\%s",
|
|
servername, argv[1]);
|
|
fstrcpy(user, cli->user_name);
|
|
|
|
/* get a printer handle */
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &pol);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
opened_hnd = True;
|
|
|
|
/* Get printer info */
|
|
|
|
result = cli_spoolss_getprinterdataex(cli, mem_ctx, 0, &needed,
|
|
&pol, keyname, valuename,
|
|
&value);
|
|
|
|
if (W_ERROR_V(result) == ERRmoredata)
|
|
result = cli_spoolss_getprinterdataex(cli, mem_ctx, needed,
|
|
NULL, &pol, keyname,
|
|
valuename, &value);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
/* Display printer data */
|
|
|
|
fstrcpy(value.valuename, valuename);
|
|
display_reg_value(value);
|
|
|
|
|
|
done:
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &pol);
|
|
|
|
return result;
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 0 display function
|
|
****************************************************************************/
|
|
static void display_print_driver_1(DRIVER_INFO_1 *i1)
|
|
{
|
|
fstring name;
|
|
if (i1 == NULL)
|
|
return;
|
|
|
|
rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
|
|
|
|
printf ("Printer Driver Info 1:\n");
|
|
printf ("\tDriver Name: [%s]\n\n", name);
|
|
|
|
return;
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 1 display function
|
|
****************************************************************************/
|
|
static void display_print_driver_2(DRIVER_INFO_2 *i1)
|
|
{
|
|
fstring name;
|
|
fstring architecture;
|
|
fstring driverpath;
|
|
fstring datafile;
|
|
fstring configfile;
|
|
if (i1 == NULL)
|
|
return;
|
|
|
|
rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
|
|
rpcstr_pull(architecture, i1->architecture.buffer, sizeof(architecture), -1, STR_TERMINATE);
|
|
rpcstr_pull(driverpath, i1->driverpath.buffer, sizeof(driverpath), -1, STR_TERMINATE);
|
|
rpcstr_pull(datafile, i1->datafile.buffer, sizeof(datafile), -1, STR_TERMINATE);
|
|
rpcstr_pull(configfile, i1->configfile.buffer, sizeof(configfile), -1, STR_TERMINATE);
|
|
|
|
printf ("Printer Driver Info 2:\n");
|
|
printf ("\tVersion: [%x]\n", i1->version);
|
|
printf ("\tDriver Name: [%s]\n", name);
|
|
printf ("\tArchitecture: [%s]\n", architecture);
|
|
printf ("\tDriver Path: [%s]\n", driverpath);
|
|
printf ("\tDatafile: [%s]\n", datafile);
|
|
printf ("\tConfigfile: [%s]\n\n", configfile);
|
|
|
|
return;
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 2 display function
|
|
****************************************************************************/
|
|
static void display_print_driver_3(DRIVER_INFO_3 *i1)
|
|
{
|
|
fstring name = "";
|
|
fstring architecture = "";
|
|
fstring driverpath = "";
|
|
fstring datafile = "";
|
|
fstring configfile = "";
|
|
fstring helpfile = "";
|
|
fstring dependentfiles = "";
|
|
fstring monitorname = "";
|
|
fstring defaultdatatype = "";
|
|
|
|
int length=0;
|
|
BOOL valid = True;
|
|
|
|
if (i1 == NULL)
|
|
return;
|
|
|
|
rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
|
|
rpcstr_pull(architecture, i1->architecture.buffer, sizeof(architecture), -1, STR_TERMINATE);
|
|
rpcstr_pull(driverpath, i1->driverpath.buffer, sizeof(driverpath), -1, STR_TERMINATE);
|
|
rpcstr_pull(datafile, i1->datafile.buffer, sizeof(datafile), -1, STR_TERMINATE);
|
|
rpcstr_pull(configfile, i1->configfile.buffer, sizeof(configfile), -1, STR_TERMINATE);
|
|
rpcstr_pull(helpfile, i1->helpfile.buffer, sizeof(helpfile), -1, STR_TERMINATE);
|
|
rpcstr_pull(monitorname, i1->monitorname.buffer, sizeof(monitorname), -1, STR_TERMINATE);
|
|
rpcstr_pull(defaultdatatype, i1->defaultdatatype.buffer, sizeof(defaultdatatype), -1, STR_TERMINATE);
|
|
|
|
printf ("Printer Driver Info 3:\n");
|
|
printf ("\tVersion: [%x]\n", i1->version);
|
|
printf ("\tDriver Name: [%s]\n",name);
|
|
printf ("\tArchitecture: [%s]\n", architecture);
|
|
printf ("\tDriver Path: [%s]\n", driverpath);
|
|
printf ("\tDatafile: [%s]\n", datafile);
|
|
printf ("\tConfigfile: [%s]\n", configfile);
|
|
printf ("\tHelpfile: [%s]\n\n", helpfile);
|
|
|
|
while (valid)
|
|
{
|
|
rpcstr_pull(dependentfiles, i1->dependentfiles+length, sizeof(dependentfiles), -1, STR_TERMINATE);
|
|
|
|
length+=strlen(dependentfiles)+1;
|
|
|
|
if (strlen(dependentfiles) > 0)
|
|
{
|
|
printf ("\tDependentfiles: [%s]\n", dependentfiles);
|
|
}
|
|
else
|
|
{
|
|
valid = False;
|
|
}
|
|
}
|
|
|
|
printf ("\n");
|
|
|
|
printf ("\tMonitorname: [%s]\n", monitorname);
|
|
printf ("\tDefaultdatatype: [%s]\n\n", defaultdatatype);
|
|
|
|
return;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Get printer information
|
|
*/
|
|
static WERROR cmd_spoolss_getdriver(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND pol;
|
|
WERROR werror;
|
|
uint32 info_level = 3;
|
|
BOOL opened_hnd = False;
|
|
PRINTER_DRIVER_CTR ctr;
|
|
fstring printername,
|
|
servername,
|
|
user;
|
|
uint32 i;
|
|
BOOL success = False;
|
|
|
|
if ((argc == 1) || (argc > 3))
|
|
{
|
|
printf("Usage: %s <printername> [level]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* get the arguments need to open the printer handle */
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(user, cli->user_name);
|
|
fstrcpy(printername, argv[1]);
|
|
if (argc == 3)
|
|
info_level = atoi(argv[2]);
|
|
|
|
/* Open a printer handle */
|
|
|
|
werror = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
|
|
PRINTER_ACCESS_USE,
|
|
servername, user, &pol);
|
|
|
|
if (!W_ERROR_IS_OK(werror)) {
|
|
printf("Error opening printer handle for %s!\n", printername);
|
|
return werror;
|
|
}
|
|
|
|
opened_hnd = True;
|
|
|
|
/* loop through and print driver info level for each architecture */
|
|
|
|
for (i=0; archi_table[i].long_archi!=NULL; i++) {
|
|
uint32 needed;
|
|
|
|
werror = cli_spoolss_getprinterdriver(
|
|
cli, mem_ctx, 0, &needed, &pol, info_level,
|
|
archi_table[i].long_archi, archi_table[i].version,
|
|
&ctr);
|
|
|
|
if (W_ERROR_V(werror) == ERRinsufficientbuffer) {
|
|
werror = cli_spoolss_getprinterdriver(
|
|
cli, mem_ctx, needed, NULL, &pol, info_level,
|
|
archi_table[i].long_archi, archi_table[i].version,
|
|
&ctr);
|
|
}
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
continue;
|
|
|
|
/* need at least one success */
|
|
|
|
success = True;
|
|
|
|
printf ("\n[%s]\n", archi_table[i].long_archi);
|
|
|
|
switch (info_level) {
|
|
case 1:
|
|
display_print_driver_1 (ctr.info1);
|
|
break;
|
|
case 2:
|
|
display_print_driver_2 (ctr.info2);
|
|
break;
|
|
case 3:
|
|
display_print_driver_3 (ctr.info3);
|
|
break;
|
|
default:
|
|
printf("unknown info level %d\n", info_level);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Cleanup */
|
|
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer (cli, mem_ctx, &pol);
|
|
|
|
if ( success )
|
|
werror = WERR_OK;
|
|
|
|
return werror;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Get printer information
|
|
*/
|
|
static WERROR cmd_spoolss_enum_drivers(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR werror;
|
|
uint32 info_level = 1;
|
|
PRINTER_DRIVER_CTR ctr;
|
|
uint32 i, j,
|
|
returned;
|
|
|
|
if (argc > 2)
|
|
{
|
|
printf("Usage: enumdrivers [level]\n");
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (argc == 2)
|
|
info_level = atoi(argv[1]);
|
|
|
|
|
|
/* loop through and print driver info level for each architecture */
|
|
for (i=0; archi_table[i].long_archi!=NULL; i++)
|
|
{
|
|
uint32 needed;
|
|
|
|
werror = cli_spoolss_enumprinterdrivers(
|
|
cli, mem_ctx, 0, &needed, info_level,
|
|
archi_table[i].long_archi, &returned, &ctr);
|
|
|
|
if (W_ERROR_V(werror) == ERRinsufficientbuffer)
|
|
werror = cli_spoolss_enumprinterdrivers(
|
|
cli, mem_ctx, needed, NULL, info_level,
|
|
archi_table[i].long_archi, &returned, &ctr);
|
|
|
|
if (returned == 0)
|
|
continue;
|
|
|
|
if (!W_ERROR_IS_OK(werror)) {
|
|
printf ("Error getting driver for environment [%s] - %d\n",
|
|
archi_table[i].long_archi, W_ERROR_V(werror));
|
|
continue;
|
|
}
|
|
|
|
printf ("\n[%s]\n", archi_table[i].long_archi);
|
|
switch (info_level)
|
|
{
|
|
|
|
case 1:
|
|
for (j=0; j < returned; j++) {
|
|
display_print_driver_1 (&(ctr.info1[j]));
|
|
}
|
|
break;
|
|
case 2:
|
|
for (j=0; j < returned; j++) {
|
|
display_print_driver_2 (&(ctr.info2[j]));
|
|
}
|
|
break;
|
|
case 3:
|
|
for (j=0; j < returned; j++) {
|
|
display_print_driver_3 (&(ctr.info3[j]));
|
|
}
|
|
break;
|
|
default:
|
|
printf("unknown info level %d\n", info_level);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return werror;
|
|
}
|
|
|
|
/****************************************************************************
|
|
printer info level 1 display function
|
|
****************************************************************************/
|
|
static void display_printdriverdir_1(DRIVER_DIRECTORY_1 *i1)
|
|
{
|
|
fstring name;
|
|
if (i1 == NULL)
|
|
return;
|
|
|
|
rpcstr_pull(name, i1->name.buffer, sizeof(name), -1, STR_TERMINATE);
|
|
|
|
printf ("\tDirectory Name:[%s]\n", name);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Get printer driver directory information
|
|
*/
|
|
static WERROR cmd_spoolss_getdriverdir(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
fstring env;
|
|
DRIVER_DIRECTORY_CTR ctr;
|
|
uint32 needed;
|
|
|
|
if (argc > 2) {
|
|
printf("Usage: %s [environment]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Get the arguments need to open the printer handle */
|
|
|
|
if (argc == 2)
|
|
fstrcpy (env, argv[1]);
|
|
else
|
|
fstrcpy (env, "Windows NT x86");
|
|
|
|
/* Get the directory. Only use Info level 1 */
|
|
|
|
result = cli_spoolss_getprinterdriverdir(
|
|
cli, mem_ctx, 0, &needed, 1, env, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprinterdriverdir(
|
|
cli, mem_ctx, needed, NULL, 1, env, &ctr);
|
|
|
|
if (W_ERROR_IS_OK(result))
|
|
display_printdriverdir_1(ctr.info1);
|
|
|
|
return result;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
set the version and environment fields of a DRIVER_INFO_3 struct
|
|
******************************************************************************/
|
|
void set_drv_info_3_env (DRIVER_INFO_3 *info, const char *arch)
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i=0; archi_table[i].long_archi != NULL; i++)
|
|
{
|
|
if (strcmp(arch, archi_table[i].short_archi) == 0)
|
|
{
|
|
info->version = archi_table[i].version;
|
|
init_unistr (&info->architecture, archi_table[i].long_archi);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (archi_table[i].long_archi == NULL)
|
|
{
|
|
DEBUG(0, ("set_drv_info_3_env: Unknown arch [%s]\n", arch));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
wrapper for strtok to get the next parameter from a delimited list.
|
|
Needed to handle the empty parameter string denoted by "NULL"
|
|
*************************************************************************/
|
|
static char* get_driver_3_param (const char* str, const char* delim, UNISTR* dest)
|
|
{
|
|
char *ptr;
|
|
|
|
/* get the next token */
|
|
ptr = strtok(str, delim);
|
|
|
|
/* a string of 'NULL' is used to represent an empty
|
|
parameter because two consecutive delimiters
|
|
will not return an empty string. See man strtok(3)
|
|
for details */
|
|
if (ptr && (StrCaseCmp(ptr, "NULL") == 0))
|
|
ptr = NULL;
|
|
|
|
if (dest != NULL)
|
|
init_unistr(dest, ptr);
|
|
|
|
return ptr;
|
|
}
|
|
|
|
/********************************************************************************
|
|
fill in the members of a DRIVER_INFO_3 struct using a character
|
|
string in the form of
|
|
<Long Printer Name>:<Driver File Name>:<Data File Name>:\
|
|
<Config File Name>:<Help File Name>:<Language Monitor Name>:\
|
|
<Default Data Type>:<Comma Separated list of Files>
|
|
*******************************************************************************/
|
|
static BOOL init_drv_info_3_members (
|
|
TALLOC_CTX *mem_ctx,
|
|
DRIVER_INFO_3 *info,
|
|
const char *args
|
|
)
|
|
{
|
|
char *str, *str2;
|
|
uint32 len, i;
|
|
|
|
/* fill in the UNISTR fields */
|
|
str = get_driver_3_param (args, ":", &info->name);
|
|
str = get_driver_3_param (NULL, ":", &info->driverpath);
|
|
str = get_driver_3_param (NULL, ":", &info->datafile);
|
|
str = get_driver_3_param (NULL, ":", &info->configfile);
|
|
str = get_driver_3_param (NULL, ":", &info->helpfile);
|
|
str = get_driver_3_param (NULL, ":", &info->monitorname);
|
|
str = get_driver_3_param (NULL, ":", &info->defaultdatatype);
|
|
|
|
/* <Comma Separated List of Dependent Files> */
|
|
str2 = get_driver_3_param (NULL, ":", NULL); /* save the beginning of the string */
|
|
str = str2;
|
|
|
|
/* begin to strip out each filename */
|
|
str = strtok(str, ",");
|
|
len = 0;
|
|
while (str != NULL)
|
|
{
|
|
/* keep a cumlative count of the str lengths */
|
|
len += strlen(str)+1;
|
|
str = strtok(NULL, ",");
|
|
}
|
|
|
|
/* allocate the space; add one extra slot for a terminating NULL.
|
|
Each filename is NULL terminated and the end contains a double
|
|
NULL */
|
|
if ((info->dependentfiles=(uint16*)talloc(mem_ctx, (len+1)*sizeof(uint16))) == NULL)
|
|
{
|
|
DEBUG(0,("init_drv_info_3_members: Unable to malloc memory for dependenfiles\n"));
|
|
return False;
|
|
}
|
|
for (i=0; i<len; i++)
|
|
{
|
|
SSVAL(&info->dependentfiles[i], 0, str2[i]);
|
|
}
|
|
info->dependentfiles[len] = '\0';
|
|
|
|
return True;
|
|
}
|
|
|
|
|
|
static WERROR cmd_spoolss_addprinterdriver(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 level = 3;
|
|
PRINTER_DRIVER_CTR ctr;
|
|
DRIVER_INFO_3 info3;
|
|
const char *arch;
|
|
fstring driver_name;
|
|
|
|
/* parse the command arguements */
|
|
if (argc != 3 && argc != 4)
|
|
{
|
|
printf ("Usage: %s <Environment> \\\n", argv[0]);
|
|
printf ("\t<Long Printer Name>:<Driver File Name>:<Data File Name>:\\\n");
|
|
printf ("\t<Config File Name>:<Help File Name>:<Language Monitor Name>:\\\n");
|
|
printf ("\t<Default Data Type>:<Comma Separated list of Files> \\\n");
|
|
printf ("\t[version]\n");
|
|
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Fill in the DRIVER_INFO_3 struct */
|
|
ZERO_STRUCT(info3);
|
|
if (!(arch = cmd_spoolss_get_short_archi(argv[1])))
|
|
{
|
|
printf ("Error Unknown architechture [%s]\n", argv[1]);
|
|
return WERR_INVALID_PARAM;
|
|
}
|
|
else
|
|
set_drv_info_3_env(&info3, arch);
|
|
|
|
if (!init_drv_info_3_members(mem_ctx, &info3, argv[2]))
|
|
{
|
|
printf ("Error Invalid parameter list - %s.\n", argv[2]);
|
|
return WERR_INVALID_PARAM;
|
|
}
|
|
|
|
/* if printer driver version specified, override the default version
|
|
* used by the architecture. This allows installation of Windows
|
|
* 2000 (version 3) printer drivers. */
|
|
if (argc == 4)
|
|
{
|
|
info3.version = atoi(argv[3]);
|
|
}
|
|
|
|
|
|
ctr.info3 = &info3;
|
|
result = cli_spoolss_addprinterdriver (cli, mem_ctx, level, &ctr);
|
|
|
|
if (W_ERROR_IS_OK(result)) {
|
|
rpcstr_pull(driver_name, info3.name.buffer,
|
|
sizeof(driver_name), -1, STR_TERMINATE);
|
|
printf ("Printer Driver %s successfully installed.\n",
|
|
driver_name);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
static WERROR cmd_spoolss_addprinterex(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 level = 2;
|
|
PRINTER_INFO_CTR ctr;
|
|
PRINTER_INFO_2 info2;
|
|
fstring servername;
|
|
|
|
/* parse the command arguements */
|
|
if (argc != 5)
|
|
{
|
|
printf ("Usage: %s <name> <shared name> <driver> <port>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
|
|
/* Fill in the DRIVER_INFO_2 struct */
|
|
ZERO_STRUCT(info2);
|
|
#if 0 /* JERRY */
|
|
init_unistr( &info2.servername, servername);
|
|
#endif
|
|
init_unistr( &info2.printername, argv[1]);
|
|
init_unistr( &info2.sharename, argv[2]);
|
|
init_unistr( &info2.drivername, argv[3]);
|
|
init_unistr( &info2.portname, argv[4]);
|
|
init_unistr( &info2.comment, "Created by rpcclient");
|
|
init_unistr( &info2.printprocessor, "winprint");
|
|
init_unistr( &info2.datatype, "RAW");
|
|
info2.devmode = NULL;
|
|
info2.secdesc = NULL;
|
|
info2.attributes = PRINTER_ATTRIBUTE_SHARED;
|
|
info2.priority = 0;
|
|
info2.defaultpriority = 0;
|
|
info2.starttime = 0;
|
|
info2.untiltime = 0;
|
|
|
|
/* These three fields must not be used by AddPrinter()
|
|
as defined in the MS Platform SDK documentation..
|
|
--jerry
|
|
info2.status = 0;
|
|
info2.cjobs = 0;
|
|
info2.averageppm = 0;
|
|
*/
|
|
|
|
ctr.printers_2 = &info2;
|
|
result = cli_spoolss_addprinterex (cli, mem_ctx, level, &ctr);
|
|
|
|
if (W_ERROR_IS_OK(result))
|
|
printf ("Printer %s successfully installed.\n", argv[1]);
|
|
|
|
return result;
|
|
}
|
|
|
|
static WERROR cmd_spoolss_setdriver(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND pol;
|
|
WERROR result;
|
|
uint32 level = 2;
|
|
BOOL opened_hnd = False;
|
|
PRINTER_INFO_CTR ctr;
|
|
PRINTER_INFO_2 info2;
|
|
fstring servername,
|
|
printername,
|
|
user;
|
|
uint32 needed;
|
|
|
|
/* parse the command arguements */
|
|
if (argc != 3)
|
|
{
|
|
printf ("Usage: %s <printer> <driver>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
slprintf(printername, sizeof(printername)-1, "%s\\%s", servername, argv[1]);
|
|
fstrcpy(user, cli->user_name);
|
|
|
|
/* Get a printer handle */
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
|
|
PRINTER_ALL_ACCESS,
|
|
servername, user, &pol);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
opened_hnd = True;
|
|
|
|
/* Get printer info */
|
|
|
|
ZERO_STRUCT (info2);
|
|
ctr.printers_2 = &info2;
|
|
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed,
|
|
&pol, level, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprinter(
|
|
cli, mem_ctx, needed, NULL, &pol, level, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result)) {
|
|
printf ("Unable to retrieve printer information!\n");
|
|
goto done;
|
|
}
|
|
|
|
/* Set the printer driver */
|
|
|
|
init_unistr(&ctr.printers_2->drivername, argv[2]);
|
|
|
|
result = cli_spoolss_setprinter(cli, mem_ctx, &pol, level, &ctr, 0);
|
|
|
|
if (!W_ERROR_IS_OK(result)) {
|
|
printf("SetPrinter call failed!\n");
|
|
goto done;;
|
|
}
|
|
|
|
printf("Succesfully set %s to driver %s.\n", argv[1], argv[2]);
|
|
|
|
done:
|
|
/* Cleanup */
|
|
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &pol);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
static WERROR cmd_spoolss_deletedriver(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
fstring servername;
|
|
int i;
|
|
|
|
/* parse the command arguements */
|
|
if (argc != 2)
|
|
{
|
|
printf ("Usage: %s <driver>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
|
|
/* delete the driver for all architectures */
|
|
for (i=0; archi_table[i].long_archi; i++)
|
|
{
|
|
/* make the call to remove the driver */
|
|
result = cli_spoolss_deleteprinterdriver(
|
|
cli, mem_ctx, archi_table[i].long_archi, argv[1]);
|
|
|
|
if ( !W_ERROR_IS_OK(result) ) {
|
|
if ( !W_ERROR_EQUAL(result, WERR_UNKNOWN_PRINTER_DRIVER) ) {
|
|
printf ("Failed to remove driver %s for arch [%s] - error 0x%x!\n",
|
|
argv[1], archi_table[i].long_archi,
|
|
W_ERROR_V(result));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
printf ("Driver %s removed for arch [%s].\n", argv[1],
|
|
archi_table[i].long_archi);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static WERROR cmd_spoolss_getprintprocdir(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
char *servername = NULL, *environment = NULL;
|
|
fstring procdir;
|
|
uint32 needed;
|
|
|
|
/* parse the command arguements */
|
|
if (argc > 2) {
|
|
printf ("Usage: %s [environment]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (asprintf(&servername, "\\\\%s", cli->desthost) < 0)
|
|
return WERR_NOMEM;
|
|
strupper_m(servername);
|
|
|
|
if (asprintf(&environment, "%s", (argc == 2) ? argv[1] :
|
|
PRINTER_DRIVER_ARCHITECTURE) < 0) {
|
|
SAFE_FREE(servername);
|
|
return WERR_NOMEM;
|
|
}
|
|
|
|
result = cli_spoolss_getprintprocessordirectory(
|
|
cli, mem_ctx, 0, &needed, servername, environment, procdir);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprintprocessordirectory(
|
|
cli, mem_ctx, needed, NULL, servername, environment,
|
|
procdir);
|
|
|
|
if (W_ERROR_IS_OK(result))
|
|
printf("%s\n", procdir);
|
|
|
|
SAFE_FREE(servername);
|
|
SAFE_FREE(environment);
|
|
|
|
return result;
|
|
}
|
|
|
|
/* Add a form */
|
|
|
|
static WERROR cmd_spoolss_addform(struct cli_state *cli, TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND handle;
|
|
WERROR werror;
|
|
char *servername = NULL, *printername = NULL;
|
|
FORM form;
|
|
BOOL got_handle = False;
|
|
|
|
/* Parse the command arguements */
|
|
|
|
if (argc != 3) {
|
|
printf ("Usage: %s <printer> <formname>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Get a printer handle */
|
|
|
|
asprintf(&servername, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
asprintf(&printername, "%s\\%s", servername, argv[1]);
|
|
|
|
werror = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
|
|
MAXIMUM_ALLOWED_ACCESS,
|
|
servername, cli->user_name, &handle);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
got_handle = True;
|
|
|
|
/* Dummy up some values for the form data */
|
|
|
|
form.flags = FORM_USER;
|
|
form.size_x = form.size_y = 100;
|
|
form.left = 0;
|
|
form.top = 10;
|
|
form.right = 20;
|
|
form.bottom = 30;
|
|
|
|
init_unistr2(&form.name, argv[2], UNI_STR_TERMINATE);
|
|
|
|
/* Add the form */
|
|
|
|
|
|
werror = cli_spoolss_addform(cli, mem_ctx, &handle, 1, &form);
|
|
|
|
done:
|
|
if (got_handle)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &handle);
|
|
|
|
SAFE_FREE(servername);
|
|
SAFE_FREE(printername);
|
|
|
|
return werror;
|
|
}
|
|
|
|
/* Set a form */
|
|
|
|
static WERROR cmd_spoolss_setform(struct cli_state *cli, TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND handle;
|
|
WERROR werror;
|
|
char *servername = NULL, *printername = NULL;
|
|
FORM form;
|
|
BOOL got_handle = False;
|
|
|
|
/* Parse the command arguements */
|
|
|
|
if (argc != 3) {
|
|
printf ("Usage: %s <printer> <formname>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Get a printer handle */
|
|
|
|
asprintf(&servername, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
asprintf(&printername, "%s\\%s", servername, argv[1]);
|
|
|
|
werror = cli_spoolss_open_printer_ex(
|
|
cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, cli->user_name, &handle);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
got_handle = True;
|
|
|
|
/* Dummy up some values for the form data */
|
|
|
|
form.flags = FORM_PRINTER;
|
|
form.size_x = form.size_y = 100;
|
|
form.left = 0;
|
|
form.top = 1000;
|
|
form.right = 2000;
|
|
form.bottom = 3000;
|
|
|
|
init_unistr2(&form.name, argv[2], UNI_STR_TERMINATE);
|
|
|
|
/* Set the form */
|
|
|
|
werror = cli_spoolss_setform(cli, mem_ctx, &handle, 1, argv[2], &form);
|
|
|
|
done:
|
|
if (got_handle)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &handle);
|
|
|
|
SAFE_FREE(servername);
|
|
SAFE_FREE(printername);
|
|
|
|
return werror;
|
|
}
|
|
|
|
/* Get a form */
|
|
|
|
static WERROR cmd_spoolss_getform(struct cli_state *cli, TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
POLICY_HND handle;
|
|
WERROR werror;
|
|
char *servername = NULL, *printername = NULL;
|
|
FORM_1 form;
|
|
BOOL got_handle = False;
|
|
uint32 needed;
|
|
|
|
/* Parse the command arguements */
|
|
|
|
if (argc != 3) {
|
|
printf ("Usage: %s <printer> <formname>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Get a printer handle */
|
|
|
|
asprintf(&servername, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
asprintf(&printername, "%s\\%s", servername, argv[1]);
|
|
|
|
werror = cli_spoolss_open_printer_ex(
|
|
cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, cli->user_name, &handle);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
got_handle = True;
|
|
|
|
/* Set the form */
|
|
|
|
werror = cli_spoolss_getform(cli, mem_ctx, 0, &needed,
|
|
&handle, argv[2], 1, &form);
|
|
|
|
if (W_ERROR_V(werror) == ERRinsufficientbuffer)
|
|
werror = cli_spoolss_getform(cli, mem_ctx, needed, NULL,
|
|
&handle, argv[2], 1, &form);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
printf("width: %d\n", form.width);
|
|
printf("length: %d\n", form.length);
|
|
printf("left: %d\n", form.left);
|
|
printf("top: %d\n", form.top);
|
|
printf("right: %d\n", form.right);
|
|
printf("bottom: %d\n", form.bottom);
|
|
|
|
done:
|
|
if (got_handle)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &handle);
|
|
|
|
SAFE_FREE(servername);
|
|
SAFE_FREE(printername);
|
|
|
|
return werror;
|
|
}
|
|
|
|
/* Delete a form */
|
|
|
|
static WERROR cmd_spoolss_deleteform(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
POLICY_HND handle;
|
|
WERROR werror;
|
|
char *servername = NULL, *printername = NULL;
|
|
BOOL got_handle = False;
|
|
|
|
/* Parse the command arguements */
|
|
|
|
if (argc != 3) {
|
|
printf ("Usage: %s <printer> <formname>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Get a printer handle */
|
|
|
|
asprintf(&servername, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
asprintf(&printername, "%s\\%s", servername, argv[1]);
|
|
|
|
werror = cli_spoolss_open_printer_ex(
|
|
cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, cli->user_name, &handle);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
got_handle = True;
|
|
|
|
/* Delete the form */
|
|
|
|
werror = cli_spoolss_deleteform(cli, mem_ctx, &handle, argv[2]);
|
|
|
|
done:
|
|
if (got_handle)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &handle);
|
|
|
|
SAFE_FREE(servername);
|
|
SAFE_FREE(printername);
|
|
|
|
return werror;
|
|
}
|
|
|
|
/* Enumerate forms */
|
|
|
|
static WERROR cmd_spoolss_enum_forms(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
POLICY_HND handle;
|
|
WERROR werror;
|
|
char *servername = NULL, *printername = NULL;
|
|
BOOL got_handle = False;
|
|
uint32 needed, num_forms, level = 1, i;
|
|
FORM_1 *forms;
|
|
|
|
/* Parse the command arguements */
|
|
|
|
if (argc != 2) {
|
|
printf ("Usage: %s <printer>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Get a printer handle */
|
|
|
|
asprintf(&servername, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
asprintf(&printername, "%s\\%s", servername, argv[1]);
|
|
|
|
werror = cli_spoolss_open_printer_ex(
|
|
cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, cli->user_name, &handle);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
got_handle = True;
|
|
|
|
/* Enumerate forms */
|
|
|
|
werror = cli_spoolss_enumforms(
|
|
cli, mem_ctx, 0, &needed, &handle, level, &num_forms, &forms);
|
|
|
|
if (W_ERROR_V(werror) == ERRinsufficientbuffer)
|
|
werror = cli_spoolss_enumforms(
|
|
cli, mem_ctx, needed, NULL, &handle, level,
|
|
&num_forms, &forms);
|
|
|
|
if (!W_ERROR_IS_OK(werror))
|
|
goto done;
|
|
|
|
/* Display output */
|
|
|
|
for (i = 0; i < num_forms; i++) {
|
|
fstring form_name;
|
|
|
|
if (forms[i].name.buffer)
|
|
rpcstr_pull(form_name, forms[i].name.buffer,
|
|
sizeof(form_name), -1, STR_TERMINATE);
|
|
|
|
printf("%s\n", form_name);
|
|
}
|
|
|
|
done:
|
|
if (got_handle)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &handle);
|
|
|
|
SAFE_FREE(servername);
|
|
SAFE_FREE(printername);
|
|
|
|
return werror;
|
|
}
|
|
|
|
static WERROR cmd_spoolss_setprinterdata(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx,
|
|
int argc, const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 needed;
|
|
fstring servername, printername, user;
|
|
POLICY_HND pol;
|
|
BOOL opened_hnd = False;
|
|
PRINTER_INFO_CTR ctr;
|
|
PRINTER_INFO_0 info;
|
|
REGISTRY_VALUE value;
|
|
|
|
/* parse the command arguements */
|
|
if (argc != 4) {
|
|
printf ("Usage: %s <printer> <value> <data>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
slprintf(printername, sizeof(servername)-1, "%s\\%s", servername, argv[1]);
|
|
fstrcpy(user, cli->user_name);
|
|
|
|
/* get a printer handle */
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
|
|
MAXIMUM_ALLOWED_ACCESS, servername,
|
|
user, &pol);
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
opened_hnd = True;
|
|
|
|
ctr.printers_0 = &info;
|
|
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed,
|
|
&pol, 0, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, 0, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
printf("%s\n", timestring(True));
|
|
printf("\tchange_id (before set)\t:[0x%x]\n", info.change_id);
|
|
|
|
/* Set the printer data */
|
|
|
|
fstrcpy(value.valuename, argv[2]);
|
|
value.type = REG_SZ;
|
|
value.size = strlen(argv[3]) + 1;
|
|
value.data_p = talloc_memdup(mem_ctx, argv[3], value.size);
|
|
|
|
result = cli_spoolss_setprinterdata(cli, mem_ctx, &pol, &value);
|
|
|
|
if (!W_ERROR_IS_OK(result)) {
|
|
printf ("Unable to set [%s=%s]!\n", argv[2], argv[3]);
|
|
goto done;
|
|
}
|
|
printf("\tSetPrinterData succeeded [%s: %s]\n", argv[2], argv[3]);
|
|
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, 0, &needed, &pol, 0, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_getprinter(cli, mem_ctx, needed, NULL, &pol, 0, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
printf("%s\n", timestring(True));
|
|
printf("\tchange_id (after set)\t:[0x%x]\n", info.change_id);
|
|
|
|
done:
|
|
/* cleanup */
|
|
if (opened_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &pol);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void display_job_info_1(JOB_INFO_1 *job)
|
|
{
|
|
fstring username = "", document = "", text_status = "";
|
|
|
|
rpcstr_pull(username, job->username.buffer,
|
|
sizeof(username), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(document, job->document.buffer,
|
|
sizeof(document), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(text_status, job->text_status.buffer,
|
|
sizeof(text_status), -1, STR_TERMINATE);
|
|
|
|
printf("%d: jobid[%d]: %s %s %s %d/%d pages\n", job->position, job->jobid,
|
|
username, document, text_status, job->pagesprinted,
|
|
job->totalpages);
|
|
}
|
|
|
|
static void display_job_info_2(JOB_INFO_2 *job)
|
|
{
|
|
fstring username = "", document = "", text_status = "";
|
|
|
|
rpcstr_pull(username, job->username.buffer,
|
|
sizeof(username), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(document, job->document.buffer,
|
|
sizeof(document), -1, STR_TERMINATE);
|
|
|
|
rpcstr_pull(text_status, job->text_status.buffer,
|
|
sizeof(text_status), -1, STR_TERMINATE);
|
|
|
|
printf("%d: jobid[%d]: %s %s %s %d/%d pages, %d bytes\n", job->position, job->jobid,
|
|
username, document, text_status, job->pagesprinted,
|
|
job->totalpages, job->size);
|
|
}
|
|
|
|
/* Enumerate jobs */
|
|
|
|
static WERROR cmd_spoolss_enum_jobs(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 needed, level = 1, num_jobs, i;
|
|
BOOL got_hnd = False;
|
|
pstring printername;
|
|
fstring servername, user;
|
|
POLICY_HND hnd;
|
|
JOB_INFO_CTR ctr;
|
|
|
|
if (argc < 2 || argc > 3) {
|
|
printf("Usage: %s printername [level]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (argc == 3)
|
|
level = atoi(argv[2]);
|
|
|
|
/* Open printer handle */
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(user, cli->user_name);
|
|
slprintf(printername, sizeof(servername)-1, "\\\\%s\\", cli->desthost);
|
|
strupper_m(printername);
|
|
pstrcat(printername, argv[1]);
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &hnd);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
got_hnd = True;
|
|
|
|
/* Enumerate ports */
|
|
|
|
result = cli_spoolss_enumjobs(
|
|
cli, mem_ctx, 0, &needed, &hnd, level, 0, 1000,
|
|
&num_jobs, &ctr);
|
|
|
|
if (W_ERROR_V(result) == ERRinsufficientbuffer)
|
|
result = cli_spoolss_enumjobs(
|
|
cli, mem_ctx, needed, NULL, &hnd, level, 0,
|
|
1000, &num_jobs, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
for (i = 0; i < num_jobs; i++) {
|
|
switch(level) {
|
|
case 1:
|
|
display_job_info_1(&ctr.job.job_info_1[i]);
|
|
break;
|
|
case 2:
|
|
display_job_info_2(&ctr.job.job_info_2[i]);
|
|
break;
|
|
default:
|
|
d_printf("unknown info level %d\n", level);
|
|
break;
|
|
}
|
|
}
|
|
|
|
done:
|
|
if (got_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &hnd);
|
|
|
|
return result;
|
|
}
|
|
|
|
/* enumerate data */
|
|
|
|
static WERROR cmd_spoolss_enum_data( struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 i=0, val_needed, data_needed;
|
|
BOOL got_hnd = False;
|
|
pstring printername;
|
|
fstring servername, user;
|
|
POLICY_HND hnd;
|
|
|
|
if (argc != 2) {
|
|
printf("Usage: %s printername\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
/* Open printer handle */
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(user, cli->user_name);
|
|
slprintf(printername, sizeof(printername)-1, "\\\\%s\\", cli->desthost);
|
|
strupper_m(printername);
|
|
pstrcat(printername, argv[1]);
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &hnd);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
got_hnd = True;
|
|
|
|
/* Enumerate data */
|
|
|
|
result = cli_spoolss_enumprinterdata(cli, mem_ctx, &hnd, i, 0, 0,
|
|
&val_needed, &data_needed,
|
|
NULL);
|
|
while (W_ERROR_IS_OK(result)) {
|
|
REGISTRY_VALUE value;
|
|
result = cli_spoolss_enumprinterdata(
|
|
cli, mem_ctx, &hnd, i++, val_needed,
|
|
data_needed, 0, 0, &value);
|
|
if (W_ERROR_IS_OK(result))
|
|
display_reg_value(value);
|
|
}
|
|
if (W_ERROR_V(result) == ERRnomoreitems)
|
|
result = W_ERROR(ERRsuccess);
|
|
|
|
done:
|
|
if (got_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &hnd);
|
|
|
|
return result;
|
|
}
|
|
|
|
/* enumerate data for a given key */
|
|
|
|
static WERROR cmd_spoolss_enum_data_ex( struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 needed, i;
|
|
BOOL got_hnd = False;
|
|
pstring printername;
|
|
fstring servername, user;
|
|
const char *keyname = NULL;
|
|
POLICY_HND hnd;
|
|
REGVAL_CTR ctr;
|
|
|
|
if (argc != 3) {
|
|
printf("Usage: %s printername <keyname>\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
keyname = argv[2];
|
|
|
|
/* Open printer handle */
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(user, cli->user_name);
|
|
slprintf(printername, sizeof(printername)-1, "\\\\%s\\", cli->desthost);
|
|
strupper_m(printername);
|
|
pstrcat(printername, argv[1]);
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &hnd);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
got_hnd = True;
|
|
|
|
/* Enumerate subkeys */
|
|
|
|
result = cli_spoolss_enumprinterdataex(
|
|
cli, mem_ctx, 0, &needed, &hnd, keyname, NULL);
|
|
|
|
if (W_ERROR_V(result) == ERRmoredata)
|
|
result = cli_spoolss_enumprinterdataex(
|
|
cli, mem_ctx, needed, NULL, &hnd, keyname, &ctr);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
for (i=0; i < ctr.num_values; i++) {
|
|
display_reg_value(*(ctr.values[i]));
|
|
}
|
|
|
|
regval_ctr_destroy(&ctr);
|
|
|
|
done:
|
|
if (got_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &hnd);
|
|
|
|
return result;
|
|
}
|
|
|
|
/* enumerate subkeys */
|
|
|
|
static WERROR cmd_spoolss_enum_printerkey( struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
WERROR result;
|
|
uint32 needed, returned;
|
|
BOOL got_hnd = False;
|
|
pstring printername;
|
|
fstring servername, user;
|
|
const char *keyname = NULL;
|
|
POLICY_HND hnd;
|
|
uint16 *keylist = NULL, *curkey;
|
|
|
|
if (argc < 2 || argc > 3) {
|
|
printf("Usage: %s printername [keyname]\n", argv[0]);
|
|
return WERR_OK;
|
|
}
|
|
|
|
if (argc == 3)
|
|
keyname = argv[2];
|
|
else
|
|
keyname = "";
|
|
|
|
/* Open printer handle */
|
|
|
|
slprintf(servername, sizeof(servername)-1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
fstrcpy(user, cli->user_name);
|
|
slprintf(printername, sizeof(printername)-1, "\\\\%s\\", cli->desthost);
|
|
strupper_m(printername);
|
|
pstrcat(printername, argv[1]);
|
|
|
|
result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername,
|
|
"", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, user, &hnd);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
got_hnd = True;
|
|
|
|
/* Enumerate subkeys */
|
|
|
|
result = cli_spoolss_enumprinterkey(
|
|
cli, mem_ctx, 0, &needed, &hnd, keyname, NULL, NULL);
|
|
|
|
if (W_ERROR_V(result) == ERRmoredata)
|
|
result = cli_spoolss_enumprinterkey(
|
|
cli, mem_ctx, needed, NULL, &hnd, keyname, &keylist,
|
|
&returned);
|
|
|
|
if (!W_ERROR_IS_OK(result))
|
|
goto done;
|
|
|
|
curkey = keylist;
|
|
while (*curkey != 0) {
|
|
pstring subkey;
|
|
rpcstr_pull(subkey, curkey, sizeof(subkey), -1,
|
|
STR_TERMINATE);
|
|
printf("%s\n", subkey);
|
|
curkey += strlen(subkey) + 1;
|
|
}
|
|
|
|
safe_free(keylist);
|
|
|
|
done:
|
|
if (got_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &hnd);
|
|
|
|
return result;
|
|
}
|
|
|
|
static WERROR cmd_spoolss_rffpcnex(struct cli_state *cli,
|
|
TALLOC_CTX *mem_ctx, int argc,
|
|
const char **argv)
|
|
{
|
|
fstring servername, printername;
|
|
POLICY_HND hnd;
|
|
BOOL got_hnd = False;
|
|
WERROR result;
|
|
SPOOL_NOTIFY_OPTION option;
|
|
|
|
if (argc != 2) {
|
|
printf("Usage: %s printername\n", argv[0]);
|
|
result = WERR_OK;
|
|
goto done;
|
|
}
|
|
|
|
/* Open printer */
|
|
|
|
slprintf(servername, sizeof(servername) - 1, "\\\\%s", cli->desthost);
|
|
strupper_m(servername);
|
|
|
|
slprintf(printername, sizeof(printername) - 1, "\\\\%s\\%s", cli->desthost,
|
|
argv[1]);
|
|
strupper_m(printername);
|
|
|
|
result = cli_spoolss_open_printer_ex(
|
|
cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS,
|
|
servername, cli->user_name, &hnd);
|
|
|
|
if (!W_ERROR_IS_OK(result)) {
|
|
printf("Error opening %s\n", argv[1]);
|
|
goto done;
|
|
}
|
|
|
|
got_hnd = True;
|
|
|
|
/* Create spool options */
|
|
|
|
ZERO_STRUCT(option);
|
|
|
|
option.version = 2;
|
|
option.option_type_ptr = 1;
|
|
option.count = option.ctr.count = 2;
|
|
|
|
option.ctr.type = (SPOOL_NOTIFY_OPTION_TYPE *)talloc(
|
|
mem_ctx, sizeof(SPOOL_NOTIFY_OPTION_TYPE) * 2);
|
|
|
|
ZERO_STRUCT(option.ctr.type[0]);
|
|
option.ctr.type[0].type = PRINTER_NOTIFY_TYPE;
|
|
option.ctr.type[0].count = option.ctr.type[0].count2 = 1;
|
|
option.ctr.type[0].fields_ptr = 1;
|
|
option.ctr.type[0].fields[0] = PRINTER_NOTIFY_SERVER_NAME;
|
|
|
|
ZERO_STRUCT(option.ctr.type[1]);
|
|
option.ctr.type[1].type = JOB_NOTIFY_TYPE;
|
|
option.ctr.type[1].count = option.ctr.type[1].count2 = 1;
|
|
option.ctr.type[1].fields_ptr = 1;
|
|
option.ctr.type[1].fields[0] = JOB_NOTIFY_PRINTER_NAME;
|
|
|
|
/* Send rffpcnex */
|
|
|
|
slprintf(servername, sizeof(servername) - 1, "\\\\%s", myhostname());
|
|
strupper_m(servername);
|
|
|
|
result = cli_spoolss_rffpcnex(
|
|
cli, mem_ctx, &hnd, 0, 0, servername, 123, &option);
|
|
|
|
if (!W_ERROR_IS_OK(result)) {
|
|
printf("Error rffpcnex %s\n", argv[1]);
|
|
goto done;
|
|
}
|
|
|
|
done:
|
|
if (got_hnd)
|
|
cli_spoolss_close_printer(cli, mem_ctx, &hnd);
|
|
|
|
return result;
|
|
}
|
|
|
|
/* List of commands exported by this module */
|
|
struct cmd_set spoolss_commands[] = {
|
|
|
|
{ "SPOOLSS" },
|
|
|
|
{ "adddriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addprinterdriver, PI_SPOOLSS, "Add a print driver", "" },
|
|
{ "addprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addprinterex, PI_SPOOLSS, "Add a printer", "" },
|
|
{ "deldriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deletedriver, PI_SPOOLSS, "Delete a printer driver", "" },
|
|
{ "enumdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_data, PI_SPOOLSS, "Enumerate printer data", "" },
|
|
{ "enumdataex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_data_ex, PI_SPOOLSS, "Enumerate printer data for a key", "" },
|
|
{ "enumkey", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_printerkey, PI_SPOOLSS, "Enumerate printer keys", "" },
|
|
{ "enumjobs", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_jobs, PI_SPOOLSS, "Enumerate print jobs", "" },
|
|
{ "enumports", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_ports, PI_SPOOLSS, "Enumerate printer ports", "" },
|
|
{ "enumdrivers", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_drivers, PI_SPOOLSS, "Enumerate installed printer drivers", "" },
|
|
{ "enumprinters", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_printers, PI_SPOOLSS, "Enumerate printers", "" },
|
|
{ "getdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinterdata, PI_SPOOLSS, "Get print driver data", "" },
|
|
{ "getdataex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinterdataex, PI_SPOOLSS, "Get printer driver data with keyname", ""},
|
|
{ "getdriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getdriver, PI_SPOOLSS, "Get print driver information", "" },
|
|
{ "getdriverdir", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getdriverdir, PI_SPOOLSS, "Get print driver upload directory", "" },
|
|
{ "getprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprinter, PI_SPOOLSS, "Get printer info", "" },
|
|
{ "getprintprocdir",RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprintprocdir, PI_SPOOLSS, "Get print processor directory", "" },
|
|
{ "openprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_open_printer_ex, PI_SPOOLSS, "Open printer handle", "" },
|
|
{ "setdriver", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setdriver, PI_SPOOLSS, "Set printer driver", "" },
|
|
{ "getprintprocdir", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getprintprocdir, PI_SPOOLSS, "Get print processor directory", "" },
|
|
{ "addform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_addform, PI_SPOOLSS, "Add form", "" },
|
|
{ "setform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setform, PI_SPOOLSS, "Set form", "" },
|
|
{ "getform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_getform, PI_SPOOLSS, "Get form", "" },
|
|
{ "deleteform", RPC_RTYPE_WERROR, NULL, cmd_spoolss_deleteform, PI_SPOOLSS, "Delete form", "" },
|
|
{ "enumforms", RPC_RTYPE_WERROR, NULL, cmd_spoolss_enum_forms, PI_SPOOLSS, "Enumerate forms", "" },
|
|
{ "setprinter", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprinter, PI_SPOOLSS, "Set printer comment", "" },
|
|
{ "setprinterdata", RPC_RTYPE_WERROR, NULL, cmd_spoolss_setprinterdata, PI_SPOOLSS, "Set REG_SZ printer data", "" },
|
|
{ "rffpcnex", RPC_RTYPE_WERROR, NULL, cmd_spoolss_rffpcnex, PI_SPOOLSS, "Rffpcnex test", "" },
|
|
|
|
{ NULL }
|
|
};
|