1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-25 06:04:04 +03:00
samba-mirror/source/rpcclient/cmd_spoolss.c

1386 lines
37 KiB
C
Raw Normal View History

/*
Unix SMB/Netbios implementation.
Version 2.2
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"
extern int DEBUGLEVEL;
extern pstring server;
extern pstring global_myname;
extern pstring username, password;
extern pstring workgroup;
struct table_node {
char *long_archi;
char *short_archi;
int version;
};
struct table_node archi_table[]= {
{"Windows 4.0", "WIN40", 0 },
{"Windows NT x86", "W32X86", 2 },
{"Windows NT R4000", "W32MIPS", 2 },
{"Windows NT Alpha_AXP", "W32ALPHA", 2 },
{"Windows NT PowerPC", "W32PPC", 2 },
{NULL, "", -1 }
};
/****************************************************************************
function to do the mapping between the long architecture name and
the short one.
****************************************************************************/
BOOL get_short_archi(char *short_archi, 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 FALSE;
}
StrnCpy (short_archi, archi_table[i].short_archi, strlen(archi_table[i].short_archi));
DEBUGADD(108,("index: [%d]\n", i));
DEBUGADD(108,("long architecture: [%s]\n", long_archi));
DEBUGADD(108,("short architecture: [%s]\n", short_archi));
return True;
}
/**********************************************************************
* dummy function -- placeholder
*/
static uint32 cmd_spoolss_not_implemented (struct cli_state *cli,
int argc, char **argv)
{
printf ("(*) This command is not currently implemented.\n");
return NT_STATUS_NO_PROBLEMO;
}
/****************************************************************************
display sec_ace structure
****************************************************************************/
static void display_sec_ace(SEC_ACE *ace)
{
fstring sid_str;
sid_to_string(sid_str, &ace->sid);
printf("\t\tSID: %s\n", sid_str);
printf("\t\ttype:[%d], flags:[0x%02x], mask:[0x%08x]\n",
ace->type, ace->flags, ace->info.mask);
}
/****************************************************************************
display sec_acl structure
****************************************************************************/
static void display_sec_acl(SEC_ACL *acl)
{
if (acl->size != 0 && acl->num_aces != 0) {
int i;
printf("\t\tRevision:[%d]\n", acl->revision);
for (i = 0; i < acl->num_aces; i++) {
display_sec_ace(&acl->ace[i]);
}
}
}
/****************************************************************************
display sec_desc structure
****************************************************************************/
static void display_sec_desc(SEC_DESC *sec)
{
fstring sid_str;
printf("\tRevision:[%d]\n", sec->revision);
if (sec->off_owner_sid) {
sid_to_string(sid_str, sec->owner_sid);
printf("\tOwner SID: %s\n", sid_str);
}
if (sec->off_grp_sid) {
sid_to_string(sid_str, sec->grp_sid);
printf("\tGroup SID: %s\n", sid_str);
}
if (sec->off_sacl) display_sec_acl(sec->sacl);
if (sec->off_dacl) display_sec_acl(sec->dacl);
}
/***********************************************************************
* Get printer information
*/
static uint32 cmd_spoolss_open_printer_ex(struct cli_state *cli, int argc, char **argv)
{
uint32 result = NT_STATUS_UNSUCCESSFUL;
pstring printername;
fstring servername, user;
POLICY_HND hnd;
TALLOC_CTX *mem_ctx;
if (argc != 2) {
printf("Usage: %s <printername>\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!cli)
return NT_STATUS_UNSUCCESSFUL;
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_open_printer_ex: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
fstrcpy (user, cli->user_name);
fstrcpy (printername, argv[1]);
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS)) {
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* Open the printer handle */
result = cli_spoolss_open_printer_ex (cli, mem_ctx, printername, "",
MAXIMUM_ALLOWED_ACCESS, servername, user, &hnd);
if (result == NT_STATUS_NOPROBLEMO) {
printf ("Printer %s opened successfully\n", printername);
result = cli_spoolss_close_printer (cli, mem_ctx, &hnd);
if (result != NT_STATUS_NOPROBLEMO) {
printf ("Error closing printer handle! (%s)\n", get_nt_error_msg(result));
}
}
cli_nt_session_close(cli);
talloc_destroy(mem_ctx);
return result;
}
/****************************************************************************
printer info level 0 display function
****************************************************************************/
static void display_print_info_0(PRINTER_INFO_0 *i1)
{
fstring name;
fstring servername;
unistr_to_ascii(name, i1->printername.buffer, sizeof(name) - 1);
unistr_to_ascii(servername, i1->servername.buffer, sizeof(servername) - 1);
printf("\tprintername:[%s]\n", name);
printf("\tservername:[%s]\n", servername);
printf("\tcjobs:[0x%x]\n", i1->cjobs);
printf("\ttotal_jobs:[0x%x]\n", i1->total_jobs);
printf("\t:date: [%d]-[%d]-[%d] (%d)\n", i1->year, i1->month,
i1->day, i1->dayofweek);
printf("\t:time: [%d]-[%d]-[%d]-[%d]\n", i1->hour, i1->minute,
i1->second, i1->milliseconds);
printf("\tglobal_counter:[0x%x]\n", i1->global_counter);
printf("\ttotal_pages:[0x%x]\n", i1->total_pages);
printf("\tmajorversion:[0x%x]\n", i1->major_version);
printf("\tbuildversion:[0x%x]\n", i1->build_version);
printf("\tunknown7:[0x%x]\n", i1->unknown7);
printf("\tunknown8:[0x%x]\n", i1->unknown8);
printf("\tunknown9:[0x%x]\n", i1->unknown9);
printf("\tsession_counter:[0x%x]\n", i1->session_counter);
printf("\tunknown11:[0x%x]\n", i1->unknown11);
printf("\tprinter_errors:[0x%x]\n", i1->printer_errors);
printf("\tunknown13:[0x%x]\n", i1->unknown13);
printf("\tunknown14:[0x%x]\n", i1->unknown14);
printf("\tunknown15:[0x%x]\n", i1->unknown15);
printf("\tunknown16:[0x%x]\n", i1->unknown16);
printf("\tchange_id:[0x%x]\n", i1->change_id);
printf("\tunknown18:[0x%x]\n", i1->unknown18);
printf("\tstatus:[0x%x]\n", i1->status);
printf("\tunknown20:[0x%x]\n", i1->unknown20);
printf("\tc_setprinter:[0x%x]\n", i1->c_setprinter);
printf("\tunknown22:[0x%x]\n", i1->unknown22);
printf("\tunknown23:[0x%x]\n", i1->unknown23);
printf("\tunknown24:[0x%x]\n", i1->unknown24);
printf("\tunknown25:[0x%x]\n", i1->unknown25);
printf("\tunknown26:[0x%x]\n", i1->unknown26);
printf("\tunknown27:[0x%x]\n", i1->unknown27);
printf("\tunknown28:[0x%x]\n", i1->unknown28);
printf("\tunknown29:[0x%x]\n", i1->unknown29);
}
/****************************************************************************
printer info level 1 display function
****************************************************************************/
static void display_print_info_1(PRINTER_INFO_1 *i1)
{
fstring desc;
fstring name;
fstring comm;
unistr_to_ascii(desc, i1->description.buffer, sizeof(desc) - 1);
unistr_to_ascii(name, i1->name .buffer, sizeof(name) - 1);
unistr_to_ascii(comm, i1->comment .buffer, sizeof(comm) - 1);
printf("\tflags:[0x%x]\n", i1->flags);
printf("\tname:[%s]\n", name);
printf("\tdescription:[%s]\n", desc);
printf("\tcomment:[%s]\n\n", comm);
}
/****************************************************************************
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;
unistr_to_ascii(servername, i2->servername.buffer,
sizeof(servername) - 1);
unistr_to_ascii(printername, i2->printername.buffer,
sizeof(printername) - 1);
unistr_to_ascii(sharename, i2->sharename.buffer,
sizeof(sharename) - 1);
unistr_to_ascii(portname, i2->portname.buffer, sizeof(portname) - 1);
unistr_to_ascii(drivername, i2->drivername.buffer,
sizeof(drivername) - 1);
unistr_to_ascii(comment, i2->comment.buffer, sizeof(comment) - 1);
unistr_to_ascii(location, i2->location.buffer, sizeof(location) - 1);
unistr_to_ascii(sepfile, i2->sepfile.buffer, sizeof(sepfile) - 1);
unistr_to_ascii(printprocessor, i2->printprocessor.buffer,
sizeof(printprocessor) - 1);
unistr_to_ascii(datatype, i2->datatype.buffer, sizeof(datatype) - 1);
unistr_to_ascii(parameters, i2->parameters.buffer,
sizeof(parameters) - 1);
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);
}
/****************************************************************************
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);
}
/* Enumerate printers */
static uint32 cmd_spoolss_enum_printers(struct cli_state *cli, int argc, char **argv)
{
uint32 result = NT_STATUS_UNSUCCESSFUL,
info_level = 1;
PRINTER_INFO_CTR ctr;
int returned;
uint32 i = 0;
TALLOC_CTX *mem_ctx;
if (argc > 2)
{
printf("Usage: %s [level]\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_enum_printers: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
if (argc == 2) {
info_level = atoi(argv[1]);
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS)) {
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* 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, PRINTER_ENUM_LOCAL,
info_level, &returned, &ctr);
if (result == NT_STATUS_NOPROBLEMO)
{
if (!returned)
printf ("No Printers printers returned.\n");
switch(info_level) {
case 0:
for (i=0; i<returned; i++) {
display_print_info_0(&(ctr.printers_0[i]));
}
break;
case 1:
for (i=0; i<returned; i++) {
display_print_info_1(&(ctr.printers_1[i]));
}
break;
case 2:
for (i=0; i<returned; i++) {
display_print_info_2(&(ctr.printers_2[i]));
}
break;
case 3:
for (i=0; i<returned; i++) {
display_print_info_3(&(ctr.printers_3[i]));
}
break;
default:
printf("unknown info level %d\n", info_level);
break;
}
}
cli_nt_session_close(cli);
talloc_destroy(mem_ctx);
return result;
}
/****************************************************************************
port info level 1 display function
****************************************************************************/
static void display_port_info_1(PORT_INFO_1 *i1)
{
fstring buffer;
unistr_to_ascii(buffer, i1->port_name.buffer, sizeof(buffer)-1);
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;
unistr_to_ascii(buffer, i2->port_name.buffer, sizeof(buffer) - 1);
printf("\tPort Name:\t[%s]\n", buffer);
unistr_to_ascii(buffer, i2->monitor_name.buffer, sizeof(buffer) - 1);
printf("\tMonitor Name:\t[%s]\n", buffer);
unistr_to_ascii(buffer, i2->description.buffer, sizeof(buffer) - 1);
printf("\tDescription:\t[%s]\n", buffer);
printf("\tPort Type:\t[%d]\n", i2->port_type);
printf("\tReserved:\t[%d]\n", i2->reserved);
printf("\n");
}
/* Enumerate ports */
static uint32 cmd_spoolss_enum_ports(struct cli_state *cli, int argc, char **argv)
{
uint32 result = NT_STATUS_UNSUCCESSFUL,
info_level = 1;
PORT_INFO_CTR ctr;
int returned;
TALLOC_CTX *mem_ctx;
if (argc > 2) {
printf("Usage: %s [level]\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_enum_ports: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
if (argc == 2) {
info_level = atoi(argv[1]);
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS)) {
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* Enumerate ports */
ZERO_STRUCT(ctr);
result = cli_spoolss_enum_ports(cli, mem_ctx, info_level, &returned, &ctr);
if (result == NT_STATUS_NOPROBLEMO) {
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;
}
}
}
cli_nt_session_close(cli);
talloc_destroy(mem_ctx);
return result;
}
/***********************************************************************
* Get printer information
*/
static uint32 cmd_spoolss_getprinter(struct cli_state *cli, int argc, char **argv)
{
POLICY_HND pol;
uint32 result,
info_level = 1;
BOOL opened_hnd = False;
PRINTER_INFO_CTR ctr;
fstring printername,
servername,
user;
TALLOC_CTX *mem_ctx;
if (argc == 1 || argc > 3) {
printf("Usage: %s <printername> [level]\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_getprinter: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS)) {
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* Open a printer handle */
if (argc == 3) {
info_level = atoi(argv[2]);
}
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
slprintf (printername, sizeof(fstring)-1, "%s\\%s", servername, argv[1]);
fstrcpy (user, cli->user_name);
/* get a printer handle */
if ((result = cli_spoolss_open_printer_ex(
cli, mem_ctx, printername, "", MAXIMUM_ALLOWED_ACCESS, servername,
user, &pol)) != NT_STATUS_NOPROBLEMO) {
goto done;
}
opened_hnd = True;
/* Get printer info */
if ((result = cli_spoolss_getprinter(cli, mem_ctx, &pol, info_level, &ctr))
!= NT_STATUS_NOPROBLEMO) {
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);
cli_nt_session_close(cli);
talloc_destroy(mem_ctx);
return result;
}
/****************************************************************************
printer info level 0 display function
****************************************************************************/
static void display_print_driver_1(DRIVER_INFO_1 *i1)
{
fstring name;
if (i1 == NULL)
return;
unistr_to_ascii(name, i1->name.buffer, sizeof(name)-1);
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;
unistr_to_ascii(name, i1->name.buffer, sizeof(name)-1);
unistr_to_ascii(architecture, i1->architecture.buffer, sizeof(architecture)-1);
unistr_to_ascii(driverpath, i1->driverpath.buffer, sizeof(driverpath)-1);
unistr_to_ascii(datafile, i1->datafile.buffer, sizeof(datafile)-1);
unistr_to_ascii(configfile, i1->configfile.buffer, sizeof(configfile)-1);
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;
unistr_to_ascii(name, i1->name.buffer, sizeof(name)-1);
unistr_to_ascii(architecture, i1->architecture.buffer, sizeof(architecture)-1);
unistr_to_ascii(driverpath, i1->driverpath.buffer, sizeof(driverpath)-1);
unistr_to_ascii(datafile, i1->datafile.buffer, sizeof(datafile)-1);
unistr_to_ascii(configfile, i1->configfile.buffer, sizeof(configfile)-1);
unistr_to_ascii(helpfile, i1->helpfile.buffer, sizeof(helpfile)-1);
unistr_to_ascii(monitorname, i1->monitorname.buffer, sizeof(monitorname)-1);
unistr_to_ascii(defaultdatatype, i1->defaultdatatype.buffer, sizeof(defaultdatatype)-1);
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)
{
unistr_to_ascii(dependentfiles, i1->dependentfiles+length, sizeof(dependentfiles)-1);
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 uint32 cmd_spoolss_getdriver(struct cli_state *cli, int argc, char **argv)
{
POLICY_HND pol;
uint32 result,
info_level = 3;
BOOL opened_hnd = False;
PRINTER_DRIVER_CTR ctr;
fstring printername,
servername,
user;
uint32 i;
TALLOC_CTX *mem_ctx;
if ((argc == 1) || (argc > 3))
{
printf("Usage: %s <printername> [level]\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_getdriver: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* get the arguments need to open the printer handle */
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
fstrcpy (user, cli->user_name);
fstrcpy (printername, argv[1]);
if (argc == 3)
info_level = atoi(argv[2]);
/* Open a printer handle */
if ((result=cli_spoolss_open_printer_ex (cli, mem_ctx, printername, "",
MAXIMUM_ALLOWED_ACCESS, servername, user, &pol)) != NT_STATUS_NO_PROBLEMO)
{
printf ("Error opening printer handle for %s!\n", printername);
return result;
}
opened_hnd = True;
/* loop through and print driver info level for each architecture */
for (i=0; archi_table[i].long_archi!=NULL; i++)
{
result = cli_spoolss_getprinterdriver (cli, mem_ctx, &pol, info_level,
archi_table[i].long_archi, &ctr);
switch (result)
{
case NT_STATUS_NO_PROBLEMO:
break;
case ERROR_UNKNOWN_PRINTER_DRIVER:
continue;
default:
printf ("Error getting driver for %s [%s] - %s\n", printername,
archi_table[i].long_archi, get_nt_error_msg(result));
continue;
}
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);
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
if (result==ERROR_UNKNOWN_PRINTER_DRIVER)
return NT_STATUS_NO_PROBLEMO;
else
return result;
}
/***********************************************************************
* Get printer information
*/
static uint32 cmd_spoolss_enum_drivers(struct cli_state *cli, int argc, char **argv)
{
uint32 result=0,
info_level = 1;
PRINTER_DRIVER_CTR ctr;
fstring servername;
uint32 i, j,
returned;
TALLOC_CTX *mem_ctx;
if (argc > 2)
{
printf("Usage: enumdrivers [level]\n");
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_enum_drivers: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* get the arguments need to open the printer handle */
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
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++)
{
returned = 0;
result = cli_spoolss_enumprinterdrivers (cli, mem_ctx, info_level,
archi_table[i].long_archi, &returned, &ctr);
if (returned == 0)
continue;
if (result != NT_STATUS_NO_PROBLEMO)
{
printf ("Error getting driver for environment [%s] - %s\n",
archi_table[i].long_archi, get_nt_error_msg(result));
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;
}
}
/* cleanup */
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
if (result==ERROR_UNKNOWN_PRINTER_DRIVER)
return NT_STATUS_NO_PROBLEMO;
else
return result;
}
/****************************************************************************
printer info level 1 display function
****************************************************************************/
static void display_printdriverdir_1(DRIVER_DIRECTORY_1 *i1)
{
fstring name;
if (i1 == NULL)
return;
unistr_to_ascii(name, i1->name.buffer, sizeof(name)-1);
printf ("\tDirectory Name:[%s]\n", name);
}
/***********************************************************************
* Get printer driver directory information
*/
static uint32 cmd_spoolss_getdriverdir(struct cli_state *cli, int argc, char **argv)
{
uint32 result;
fstring env;
DRIVER_DIRECTORY_CTR ctr;
TALLOC_CTX *mem_ctx;
if (argc > 2)
{
printf("Usage: %s [environment]\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_getdriverdir: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
/* 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 */
if ((result = cli_spoolss_getprinterdriverdir (cli, mem_ctx, 1, env, &ctr))
!= NT_STATUS_NO_PROBLEMO)
{
return result;
}
display_printdriverdir_1 (ctr.info1);
/* cleanup */
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
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 (char* str, 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 (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,
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++)
{
info->dependentfiles[i] = (uint16)str2[i];
}
info->dependentfiles[len] = '\0';
return True;
}
static uint32 cmd_spoolss_addprinterdriver (struct cli_state *cli, int argc, char **argv)
{
uint32 result,
level = 3;
PRINTER_DRIVER_CTR ctr;
DRIVER_INFO_3 info3;
fstring arch;
fstring driver_name;
TALLOC_CTX *mem_ctx = NULL;
/* parse the command arguements */
if (argc != 3)
{
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");
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_addprinterdriver: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* Fill in the DRIVER_INFO_3 struct */
ZERO_STRUCT(info3);
if (!get_short_archi(arch, argv[1]))
{
printf ("Error Unknown architechture [%s]\n", argv[1]);
return NT_STATUS_INVALID_PARAMETER;
}
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 NT_STATUS_INVALID_PARAMETER;
}
ctr.info3 = &info3;
if ((result = cli_spoolss_addprinterdriver (cli, mem_ctx, level, &ctr))
!= NT_STATUS_NO_PROBLEMO)
{
return result;
}
unistr_to_ascii (driver_name, info3.name.buffer, sizeof(driver_name)-1);
printf ("Printer Driver %s successfully installed.\n", driver_name);
/* cleanup */
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
return result;
}
static uint32 cmd_spoolss_addprinterex (struct cli_state *cli, int argc, char **argv)
{
uint32 result,
level = 2;
PRINTER_INFO_CTR ctr;
PRINTER_INFO_2 info2;
fstring servername;
TALLOC_CTX *mem_ctx = NULL;
/* parse the command arguements */
if (argc != 5)
{
printf ("Usage: %s <name> <shared name> <driver> <port>\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_addprinterex: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* Fill in the DRIVER_INFO_3 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;
if ((result = cli_spoolss_addprinterex (cli, mem_ctx, level, &ctr))
!= NT_STATUS_NO_PROBLEMO)
{
cli_nt_session_close (cli);
return result;
}
printf ("Printer %s successfully installed.\n", argv[1]);
/* cleanup */
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
return result;
}
static uint32 cmd_spoolss_setdriver (struct cli_state *cli, int argc, char **argv)
{
POLICY_HND pol;
uint32 result,
level = 2;
BOOL opened_hnd = False;
PRINTER_INFO_CTR ctr;
PRINTER_INFO_2 info2;
fstring servername,
printername,
user;
TALLOC_CTX *mem_ctx = NULL;
/* parse the command arguements */
if (argc != 3)
{
printf ("Usage: %s <printer> <driver>\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_setdriver: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
slprintf (printername, sizeof(fstring)-1, "%s\\%s", servername, argv[1]);
fstrcpy (user, cli->user_name);
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* get a printer handle */
if ((result = cli_spoolss_open_printer_ex(cli, mem_ctx, printername, "",
MAXIMUM_ALLOWED_ACCESS, servername, user, &pol))
!= NT_STATUS_NOPROBLEMO)
{
goto done;
}
opened_hnd = True;
/* Get printer info */
ZERO_STRUCT (info2);
ctr.printers_2 = &info2;
if ((result = cli_spoolss_getprinter(cli, mem_ctx, &pol, level, &ctr)) != NT_STATUS_NOPROBLEMO)
{
printf ("Unable to retrieve printer information!\n");
goto done;
}
/* set the printer driver */
init_unistr(&ctr.printers_2->drivername, argv[2]);
if ((result = cli_spoolss_setprinter(cli, mem_ctx, &pol, level, &ctr, 0)) != NT_STATUS_NO_PROBLEMO)
{
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);
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
return result;
}
static uint32 cmd_spoolss_deletedriver (struct cli_state *cli, int argc, char **argv)
{
uint32 result = NT_STATUS_UNSUCCESSFUL;
fstring servername;
TALLOC_CTX *mem_ctx = NULL;
int i;
/* parse the command arguements */
if (argc != 2)
{
printf ("Usage: %s <driver>\n", argv[0]);
return NT_STATUS_NOPROBLEMO;
}
if (!(mem_ctx=talloc_init()))
{
DEBUG(0,("cmd_spoolss_deletedriver: talloc_init returned NULL!\n"));
return NT_STATUS_UNSUCCESSFUL;
}
slprintf (servername, sizeof(fstring)-1, "\\\\%s", cli->desthost);
strupper (servername);
/* Initialise RPC connection */
if (!cli_nt_session_open (cli, PIPE_SPOOLSS))
{
fprintf (stderr, "Could not initialize spoolss pipe!\n");
return NT_STATUS_UNSUCCESSFUL;
}
/* delete the driver for all architectures */
for (i=0; archi_table[i].long_archi; i++)
{
/* make the call to remove the driver */
if ((result = cli_spoolss_deleteprinterdriver(cli, mem_ctx,
archi_table[i].long_archi, argv[1])) != NT_STATUS_NO_PROBLEMO)
{
printf ("Failed to remove driver %s for arch [%s] - error %s!\n",
argv[1], archi_table[i].long_archi, get_nt_error_msg(result));
}
else
printf ("Driver %s removed for arch [%s].\n", argv[1], archi_table[i].long_archi);
}
/* cleanup */
cli_nt_session_close (cli);
talloc_destroy(mem_ctx);
return NT_STATUS_NO_PROBLEMO;
}
/* List of commands exported by this module */
struct cmd_set spoolss_commands[] = {
{ "SPOOLSS", NULL, "" },
{ "adddriver", cmd_spoolss_addprinterdriver, "Add a print driver" },
{ "addprinter", cmd_spoolss_addprinterex, "Add a printer" },
{ "deldriver", cmd_spoolss_deletedriver, "Delete a printer driver" },
{ "enumdata", cmd_spoolss_not_implemented, "Enumerate printer data (*)" },
{ "enumjobs", cmd_spoolss_not_implemented, "Enumerate print jobs (*)" },
{ "enumports", cmd_spoolss_enum_ports, "Enumerate printer ports" },
{ "enumdrivers", cmd_spoolss_enum_drivers, "Enumerate installed printer drivers" },
{ "enumprinters", cmd_spoolss_enum_printers, "Enumerate printers" },
{ "getdata", cmd_spoolss_not_implemented, "Get print driver data (*)" },
{ "getdriver", cmd_spoolss_getdriver, "Get print driver information" },
{ "getdriverdir", cmd_spoolss_getdriverdir, "Get print driver upload directory" },
{ "getprinter", cmd_spoolss_getprinter, "Get printer info" },
{ "openprinter", cmd_spoolss_open_printer_ex, "Open printer handle" },
{ "setdriver", cmd_spoolss_setdriver, "Set printer driver" },
{ NULL, NULL, NULL }
};