2001-03-16 00:35:57 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2001-03-16 00:35:57 +00:00
printing command routines
Copyright ( C ) Andrew Tridgell 1992 - 2000
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 0213 9 , USA .
*/
2003-11-12 01:51:10 +00:00
# include "includes.h"
2001-03-16 00:35:57 +00:00
# include "printing.h"
/****************************************************************************
run a given print command
a null terminated list of value / substitute pairs is provided
for local substitution strings
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-09-30 17:13:37 +00:00
static int print_run_command ( int snum , const char * printername , BOOL do_sub ,
const char * command , int * outfd , . . . )
2001-03-16 00:35:57 +00:00
{
2001-04-10 22:01:53 +00:00
2001-03-16 00:35:57 +00:00
pstring syscmd ;
2002-07-15 10:35:28 +00:00
char * arg ;
2001-03-16 00:35:57 +00:00
int ret ;
va_list ap ;
2001-04-13 19:12:06 +00:00
va_start ( ap , outfd ) ;
2001-04-10 22:01:53 +00:00
2004-10-19 17:05:01 +00:00
/* check for a valid system printername and valid command to run */
2001-03-16 00:35:57 +00:00
2004-10-19 17:05:01 +00:00
if ( ! printername | | ! * printername )
return - 1 ;
if ( ! command | | ! * command )
2001-03-16 00:35:57 +00:00
return - 1 ;
pstrcpy ( syscmd , command ) ;
while ( ( arg = va_arg ( ap , char * ) ) ) {
char * value = va_arg ( ap , char * ) ;
pstring_sub ( syscmd , arg , value ) ;
}
va_end ( ap ) ;
2004-10-19 17:05:01 +00:00
pstring_sub ( syscmd , " %p " , printername ) ;
2001-03-16 00:35:57 +00:00
2004-10-19 17:05:01 +00:00
if ( do_sub & & snum ! = - 1 )
standard_sub_snum ( snum , syscmd , sizeof ( syscmd ) ) ;
2001-04-13 19:12:06 +00:00
ret = smbrun ( syscmd , outfd ) ;
2001-03-16 00:35:57 +00:00
DEBUG ( 3 , ( " Running the command `%s' gave %d \n " , syscmd , ret ) ) ;
return ret ;
}
/****************************************************************************
delete a print job
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-09-30 17:13:37 +00:00
static int generic_job_delete ( const char * sharename , const char * lprm_command , struct printjob * pjob )
2001-03-16 00:35:57 +00:00
{
fstring jobstr ;
/* need to delete the spooled entry */
slprintf ( jobstr , sizeof ( jobstr ) - 1 , " %d " , pjob - > sysjob ) ;
2005-09-30 17:13:37 +00:00
return print_run_command ( - 1 , sharename , False , lprm_command , NULL ,
2001-03-16 00:35:57 +00:00
" %j " , jobstr ,
" %T " , http_timestring ( pjob - > starttime ) ,
NULL ) ;
}
/****************************************************************************
pause a job
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int generic_job_pause ( int snum , struct printjob * pjob )
{
fstring jobstr ;
/* need to pause the spooled entry */
slprintf ( jobstr , sizeof ( jobstr ) - 1 , " %d " , pjob - > sysjob ) ;
2004-10-19 17:05:01 +00:00
return print_run_command ( snum , PRINTERNAME ( snum ) , True ,
2001-04-14 00:19:12 +00:00
lp_lppausecommand ( snum ) , NULL ,
2001-03-16 00:35:57 +00:00
" %j " , jobstr ,
NULL ) ;
}
/****************************************************************************
resume a job
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int generic_job_resume ( int snum , struct printjob * pjob )
{
fstring jobstr ;
/* need to pause the spooled entry */
slprintf ( jobstr , sizeof ( jobstr ) - 1 , " %d " , pjob - > sysjob ) ;
2004-10-19 17:05:01 +00:00
return print_run_command ( snum , PRINTERNAME ( snum ) , True ,
2001-04-14 00:19:12 +00:00
lp_lpresumecommand ( snum ) , NULL ,
2001-03-16 00:35:57 +00:00
" %j " , jobstr ,
NULL ) ;
}
/****************************************************************************
Submit a file for printing - called from print_job_end ( )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int generic_job_submit ( int snum , struct printjob * pjob )
{
int ret ;
pstring current_directory ;
pstring print_directory ;
char * wd , * p ;
pstring jobname ;
2002-03-19 23:33:32 +00:00
fstring job_page_count , job_size ;
2001-03-16 00:35:57 +00:00
/* we print from the directory path to give the best chance of
parsing the lpq output */
wd = sys_getwd ( current_directory ) ;
if ( ! wd )
return 0 ;
pstrcpy ( print_directory , pjob - > filename ) ;
2001-07-04 07:36:09 +00:00
p = strrchr_m ( print_directory , ' / ' ) ;
2001-03-16 00:35:57 +00:00
if ( ! p )
return 0 ;
* p + + = 0 ;
if ( chdir ( print_directory ) ! = 0 )
return 0 ;
pstrcpy ( jobname , pjob - > jobname ) ;
pstring_sub ( jobname , " ' " , " _ " ) ;
2002-03-19 23:33:32 +00:00
slprintf ( job_page_count , sizeof ( job_page_count ) - 1 , " %d " , pjob - > page_count ) ;
2003-11-03 14:34:25 +00:00
slprintf ( job_size , sizeof ( job_size ) - 1 , " %lu " , ( unsigned long ) pjob - > size ) ;
2001-03-16 00:35:57 +00:00
/* send it to the system spooler */
2004-10-19 17:05:01 +00:00
ret = print_run_command ( snum , PRINTERNAME ( snum ) , True ,
2002-03-19 23:33:32 +00:00
lp_printcommand ( snum ) , NULL ,
" %s " , p ,
" %J " , jobname ,
" %f " , p ,
" %z " , job_size ,
" %c " , job_page_count ,
NULL ) ;
2001-03-16 00:35:57 +00:00
chdir ( wd ) ;
return ret ;
}
/****************************************************************************
get the current list of queued jobs
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-10-19 17:05:01 +00:00
static int generic_queue_get ( const char * printer_name ,
enum printing_types printing_type ,
char * lpq_command ,
print_queue_struct * * q ,
print_status_struct * status )
2001-03-16 00:35:57 +00:00
{
char * * qlines ;
2001-04-13 00:37:00 +00:00
int fd ;
2001-03-16 00:35:57 +00:00
int numlines , i , qcount ;
2001-04-13 19:12:06 +00:00
print_queue_struct * queue = NULL ;
2001-03-16 00:35:57 +00:00
2004-10-19 17:05:01 +00:00
/* never do substitution when running the 'lpq command' since we can't
get it rigt when using the background update daemon . Make the caller
do it before passing off the command string to us here . */
print_run_command ( - 1 , printer_name , False , lpq_command , & fd , NULL ) ;
2001-03-16 00:35:57 +00:00
2001-04-13 00:37:00 +00:00
if ( fd = = - 1 ) {
DEBUG ( 5 , ( " generic_queue_get: Can't read print queue status for printer %s \n " ,
printer_name ) ) ;
return 0 ;
}
2001-03-16 00:35:57 +00:00
numlines = 0 ;
2006-02-03 22:19:41 +00:00
qlines = fd_lines_load ( fd , & numlines , 0 ) ;
2001-04-13 00:37:00 +00:00
close ( fd ) ;
2001-03-16 00:35:57 +00:00
/* turn the lpq output into a series of job structures */
qcount = 0 ;
2001-03-28 18:12:49 +00:00
ZERO_STRUCTP ( status ) ;
2001-03-16 00:35:57 +00:00
if ( numlines )
2004-12-07 18:25:53 +00:00
queue = SMB_MALLOC_ARRAY ( print_queue_struct , numlines + 1 ) ;
2001-03-16 00:35:57 +00:00
if ( queue ) {
2003-04-09 00:00:08 +00:00
memset ( queue , ' \0 ' , sizeof ( print_queue_struct ) * ( numlines + 1 ) ) ;
2001-03-16 00:35:57 +00:00
for ( i = 0 ; i < numlines ; i + + ) {
/* parse the line */
2004-10-19 17:05:01 +00:00
if ( parse_lpq_entry ( printing_type , qlines [ i ] ,
2001-03-16 00:35:57 +00:00
& queue [ qcount ] , status , qcount = = 0 ) ) {
qcount + + ;
}
}
}
file_lines_free ( qlines ) ;
* q = queue ;
return qcount ;
}
/****************************************************************************
pause a queue
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int generic_queue_pause ( int snum )
{
2004-10-19 17:05:01 +00:00
return print_run_command ( snum , PRINTERNAME ( snum ) , True , lp_queuepausecommand ( snum ) , NULL , NULL ) ;
2001-03-16 00:35:57 +00:00
}
/****************************************************************************
resume a queue
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int generic_queue_resume ( int snum )
{
2004-10-19 17:05:01 +00:00
return print_run_command ( snum , PRINTERNAME ( snum ) , True , lp_queueresumecommand ( snum ) , NULL , NULL ) ;
2001-03-16 00:35:57 +00:00
}
2004-10-19 17:05:01 +00:00
/****************************************************************************
* Generic printing interface definitions . . .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct printif generic_printif =
{
DEFAULT_PRINTING ,
generic_queue_get ,
generic_queue_pause ,
generic_queue_resume ,
generic_job_delete ,
generic_job_pause ,
generic_job_resume ,
generic_job_submit ,
} ;