1999-12-17 04:46:15 +03:00
/*
* Support code for the Common UNIX Printing System ( " CUPS " )
*
2003-01-28 04:58:38 +03:00
* Copyright 1999 - 2003 by Michael R Sweet .
1999-12-17 04:46:15 +03:00
*
* 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 04:51:10 +03:00
# include "includes.h"
2001-03-16 08:55:30 +03:00
# include "printing.h"
1999-12-17 04:46:15 +03:00
2001-08-23 23:06:20 +04:00
# ifdef HAVE_CUPS
1999-12-17 04:46:15 +03:00
# include <cups/cups.h>
# include <cups/language.h>
2001-03-16 08:55:30 +03:00
/*
* ' cups_passwd_cb ( ) ' - The CUPS password callback . . .
*/
2003-01-15 21:57:41 +03:00
static const char * /* O - Password or NULL */
2001-03-16 08:55:30 +03:00
cups_passwd_cb ( const char * prompt ) /* I - Prompt */
{
2005-01-21 03:29:38 +03:00
/*
* Always return NULL to indicate that no password is available . . .
*/
2001-03-16 08:55:30 +03:00
2005-01-21 03:29:38 +03:00
return ( NULL ) ;
2001-03-16 08:55:30 +03:00
}
2004-06-02 18:58:18 +04:00
static const char * cups_server ( void )
{
if ( ( lp_cups_server ( ) ! = NULL ) & & ( strlen ( lp_cups_server ( ) ) > 0 ) ) {
DEBUG ( 10 , ( " cups server explicitly set to %s \n " ,
lp_cups_server ( ) ) ) ;
return lp_cups_server ( ) ;
}
DEBUG ( 10 , ( " cups server left to default %s \n " , cupsServer ( ) ) ) ;
return cupsServer ( ) ;
}
2001-03-16 08:55:30 +03:00
2005-01-05 19:20:35 +03:00
BOOL cups_cache_reload ( void )
1999-12-17 04:46:15 +03:00
{
2005-01-21 03:29:38 +03:00
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
1999-12-17 04:46:15 +03:00
ipp_attribute_t * attr ; /* Current attribute */
2005-01-21 03:29:38 +03:00
cups_lang_t * language = NULL ; /* Default language */
1999-12-17 04:46:15 +03:00
char * name , /* printer-name attribute */
2001-03-16 08:55:30 +03:00
* info ; /* printer-info attribute */
static const char * requested [ ] = /* Requested attributes */
{
" printer-name " ,
" printer-info "
} ;
2005-01-21 03:29:38 +03:00
BOOL ret = False ;
2001-03-16 08:55:30 +03:00
2005-01-05 19:20:35 +03:00
DEBUG ( 5 , ( " reloading cups printcap cache \n " ) ) ;
2001-03-16 08:55:30 +03:00
/*
* Make sure we don ' t ask for passwords . . .
*/
1999-12-17 04:46:15 +03:00
2001-03-16 08:55:30 +03:00
cupsSetPasswordCB ( cups_passwd_cb ) ;
1999-12-17 04:46:15 +03:00
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
1999-12-17 04:46:15 +03:00
/*
* Build a CUPS_GET_PRINTERS request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
2001-03-16 08:55:30 +03:00
* requested - attributes
1999-12-17 04:46:15 +03:00
*/
request = ippNew ( ) ;
request - > request . op . operation_id = CUPS_GET_PRINTERS ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
2001-03-16 08:55:30 +03:00
ippAddStrings ( request , IPP_TAG_OPERATION , IPP_TAG_NAME ,
" requested-attributes " ,
( sizeof ( requested ) / sizeof ( requested [ 0 ] ) ) ,
NULL , requested ) ;
1999-12-17 04:46:15 +03:00
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " / " ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to get printer list - %s \n " ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
1999-12-17 04:46:15 +03:00
}
2005-01-21 03:29:38 +03:00
for ( attr = response - > attrs ; attr ! = NULL ; ) {
1999-12-17 04:46:15 +03:00
/*
* Skip leading attributes until we hit a printer . . .
*/
while ( attr ! = NULL & & attr - > group_tag ! = IPP_TAG_PRINTER )
attr = attr - > next ;
if ( attr = = NULL )
break ;
/*
* Pull the needed attributes from this printer . . .
*/
name = NULL ;
2001-03-16 08:55:30 +03:00
info = NULL ;
1999-12-17 04:46:15 +03:00
2005-01-21 03:29:38 +03:00
while ( attr ! = NULL & & attr - > group_tag = = IPP_TAG_PRINTER ) {
1999-12-17 04:46:15 +03:00
if ( strcmp ( attr - > name , " printer-name " ) = = 0 & &
attr - > value_tag = = IPP_TAG_NAME )
name = attr - > values [ 0 ] . string . text ;
2001-03-16 08:55:30 +03:00
if ( strcmp ( attr - > name , " printer-info " ) = = 0 & &
attr - > value_tag = = IPP_TAG_TEXT )
info = attr - > values [ 0 ] . string . text ;
1999-12-17 04:46:15 +03:00
attr = attr - > next ;
}
/*
* See if we have everything needed . . .
*/
if ( name = = NULL )
break ;
2005-01-05 19:20:35 +03:00
if ( ! pcap_cache_add ( name , info ) ) {
2005-01-21 03:29:38 +03:00
goto out ;
2005-01-05 19:20:35 +03:00
}
1999-12-17 04:46:15 +03:00
}
ippDelete ( response ) ;
2005-01-21 03:29:38 +03:00
response = NULL ;
2003-01-28 04:58:38 +03:00
/*
* Build a CUPS_GET_CLASSES request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* requested - attributes
*/
2005-01-21 21:14:31 +03:00
request = ippNew ( ) ;
2003-01-28 04:58:38 +03:00
request - > request . op . operation_id = CUPS_GET_CLASSES ;
request - > request . op . request_id = 1 ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
ippAddStrings ( request , IPP_TAG_OPERATION , IPP_TAG_NAME ,
" requested-attributes " ,
( sizeof ( requested ) / sizeof ( requested [ 0 ] ) ) ,
NULL , requested ) ;
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " / " ) ) = = NULL ) {
2003-01-28 04:58:38 +03:00
DEBUG ( 0 , ( " Unable to get printer list - %s \n " ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2003-01-28 04:58:38 +03:00
}
2005-01-21 03:29:38 +03:00
for ( attr = response - > attrs ; attr ! = NULL ; ) {
2003-01-28 04:58:38 +03:00
/*
* Skip leading attributes until we hit a printer . . .
*/
while ( attr ! = NULL & & attr - > group_tag ! = IPP_TAG_PRINTER )
attr = attr - > next ;
if ( attr = = NULL )
break ;
/*
* Pull the needed attributes from this printer . . .
*/
name = NULL ;
info = NULL ;
2005-01-21 03:29:38 +03:00
while ( attr ! = NULL & & attr - > group_tag = = IPP_TAG_PRINTER ) {
2003-01-28 04:58:38 +03:00
if ( strcmp ( attr - > name , " printer-name " ) = = 0 & &
attr - > value_tag = = IPP_TAG_NAME )
name = attr - > values [ 0 ] . string . text ;
if ( strcmp ( attr - > name , " printer-info " ) = = 0 & &
attr - > value_tag = = IPP_TAG_TEXT )
info = attr - > values [ 0 ] . string . text ;
attr = attr - > next ;
}
/*
* See if we have everything needed . . .
*/
if ( name = = NULL )
break ;
2005-01-05 19:20:35 +03:00
if ( ! pcap_cache_add ( name , info ) ) {
2005-01-21 03:29:38 +03:00
goto out ;
2005-01-05 19:20:35 +03:00
}
2003-01-28 04:58:38 +03:00
}
2005-01-21 03:29:38 +03:00
ret = True ;
2003-01-28 04:58:38 +03:00
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
2003-01-28 04:58:38 +03:00
2005-01-21 03:29:38 +03:00
return ret ;
1999-12-17 04:46:15 +03:00
}
2001-03-16 08:55:30 +03:00
/*
* ' cups_job_delete ( ) ' - Delete a job .
*/
2005-01-21 03:29:38 +03:00
static int cups_job_delete ( int snum , struct printjob * pjob )
2001-03-16 08:55:30 +03:00
{
2005-01-21 03:29:38 +03:00
int ret = 1 ; /* Return value */
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
DEBUG ( 5 , ( " cups_job_delete(%d, %p (%d)) \n " , snum , pjob , pjob - > sysjob ) ) ;
/*
* Make sure we don ' t ask for passwords . . .
*/
cupsSetPasswordCB ( cups_passwd_cb ) ;
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Build an IPP_CANCEL_JOB request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* job - uri
* requesting - user - name
*/
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_CANCEL_JOB ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/jobs/%d " , pjob - > sysjob ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI , " job-uri " , NULL , uri ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " requesting-user-name " ,
2001-07-17 04:39:07 +04:00
NULL , pjob - > user ) ;
2001-03-16 08:55:30 +03:00
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " /jobs " ) ) ! = NULL ) {
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
DEBUG ( 0 , ( " Unable to cancel job %d - %s \n " , pjob - > sysjob ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
} else {
ret = 0 ;
}
} else {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to cancel job %d - %s \n " , pjob - > sysjob ,
2005-01-21 03:29:38 +03:00
ippErrorString ( cupsLastError ( ) ) ) ) ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return ret ;
2001-03-16 08:55:30 +03:00
}
/*
* ' cups_job_pause ( ) ' - Pause a job .
*/
2005-01-21 03:29:38 +03:00
static int cups_job_pause ( int snum , struct printjob * pjob )
2001-03-16 08:55:30 +03:00
{
2005-01-21 03:29:38 +03:00
int ret = 1 ; /* Return value */
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
DEBUG ( 5 , ( " cups_job_pause(%d, %p (%d)) \n " , snum , pjob , pjob - > sysjob ) ) ;
/*
* Make sure we don ' t ask for passwords . . .
*/
cupsSetPasswordCB ( cups_passwd_cb ) ;
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Build an IPP_HOLD_JOB request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* job - uri
* requesting - user - name
*/
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_HOLD_JOB ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/jobs/%d " , pjob - > sysjob ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI , " job-uri " , NULL , uri ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " requesting-user-name " ,
2001-07-17 04:39:07 +04:00
NULL , pjob - > user ) ;
2001-03-16 08:55:30 +03:00
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " /jobs " ) ) ! = NULL ) {
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
DEBUG ( 0 , ( " Unable to hold job %d - %s \n " , pjob - > sysjob ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
} else {
ret = 0 ;
}
} else {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to hold job %d - %s \n " , pjob - > sysjob ,
2005-01-21 03:29:38 +03:00
ippErrorString ( cupsLastError ( ) ) ) ) ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return ret ;
2001-03-16 08:55:30 +03:00
}
/*
* ' cups_job_resume ( ) ' - Resume a paused job .
*/
2005-01-21 03:29:38 +03:00
static int cups_job_resume ( int snum , struct printjob * pjob )
2001-03-16 08:55:30 +03:00
{
2005-01-21 03:29:38 +03:00
int ret = 1 ; /* Return value */
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
DEBUG ( 5 , ( " cups_job_resume(%d, %p (%d)) \n " , snum , pjob , pjob - > sysjob ) ) ;
/*
* Make sure we don ' t ask for passwords . . .
*/
cupsSetPasswordCB ( cups_passwd_cb ) ;
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Build an IPP_RELEASE_JOB request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* job - uri
* requesting - user - name
*/
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_RELEASE_JOB ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/jobs/%d " , pjob - > sysjob ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI , " job-uri " , NULL , uri ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " requesting-user-name " ,
2001-07-17 04:39:07 +04:00
NULL , pjob - > user ) ;
2001-03-16 08:55:30 +03:00
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " /jobs " ) ) ! = NULL ) {
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
DEBUG ( 0 , ( " Unable to release job %d - %s \n " , pjob - > sysjob ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
} else {
ret = 0 ;
}
} else {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to release job %d - %s \n " , pjob - > sysjob ,
2005-01-21 03:29:38 +03:00
ippErrorString ( cupsLastError ( ) ) ) ) ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return ret ;
2001-03-16 08:55:30 +03:00
}
/*
* ' cups_job_submit ( ) ' - Submit a job for printing .
*/
2005-01-21 03:29:38 +03:00
static int cups_job_submit ( int snum , struct printjob * pjob )
2001-03-16 08:55:30 +03:00
{
2005-01-21 03:29:38 +03:00
int ret = 1 ; /* Return value */
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
2005-01-21 03:29:38 +03:00
char * clientname = NULL ; /* hostname of client for job-originating-host attribute */
2003-11-25 22:16:35 +03:00
pstring new_jobname ;
2004-04-16 00:40:26 +04:00
int num_options = 0 ;
2005-01-21 03:29:38 +03:00
cups_option_t * options = NULL ;
2001-03-16 08:55:30 +03:00
DEBUG ( 5 , ( " cups_job_submit(%d, %p (%d)) \n " , snum , pjob , pjob - > sysjob ) ) ;
/*
* Make sure we don ' t ask for passwords . . .
*/
cupsSetPasswordCB ( cups_passwd_cb ) ;
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Build an IPP_PRINT_JOB request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* printer - uri
* requesting - user - name
* [ document - data ]
*/
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_PRINT_JOB ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/printers/%s " ,
PRINTERNAME ( snum ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI ,
" printer-uri " , NULL , uri ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " requesting-user-name " ,
2001-07-17 04:39:07 +04:00
NULL , pjob - > user ) ;
2001-03-16 08:55:30 +03:00
2003-11-24 21:37:19 +03:00
clientname = client_name ( ) ;
if ( strcmp ( clientname , " UNKNOWN " ) = = 0 ) {
clientname = client_addr ( ) ;
}
2003-02-05 09:37:27 +03:00
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME ,
" job-originating-host-name " , NULL ,
2003-11-24 21:37:19 +03:00
clientname ) ;
2003-02-05 09:37:27 +03:00
2003-11-25 22:16:35 +03:00
pstr_sprintf ( new_jobname , " %s%.8u %s " , PRINT_SPOOL_PREFIX ,
( unsigned int ) pjob - > smbjob , pjob - > jobname ) ;
2001-03-16 08:55:30 +03:00
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " job-name " , NULL ,
2003-11-25 22:16:35 +03:00
new_jobname ) ;
2001-03-16 08:55:30 +03:00
2004-04-16 00:40:26 +04:00
/*
* add any options defined in smb . conf
*/
num_options = 0 ;
options = NULL ;
num_options = cupsParseOptions ( lp_cups_options ( snum ) , num_options , & options ) ;
if ( num_options )
cupsEncodeOptions ( request , num_options , options ) ;
2001-03-16 08:55:30 +03:00
/*
* Do the request and get back a response . . .
*/
slprintf ( uri , sizeof ( uri ) - 1 , " /printers/%s " , PRINTERNAME ( snum ) ) ;
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoFileRequest ( http , request , uri , pjob - > filename ) ) ! = NULL ) {
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to print file to %s - %s \n " , PRINTERNAME ( snum ) ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
2005-01-21 03:29:38 +03:00
} else {
2001-03-16 08:55:30 +03:00
ret = 0 ;
2005-01-21 03:29:38 +03:00
}
} else {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to print file to `%s' - %s \n " , PRINTERNAME ( snum ) ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
2005-01-21 03:29:38 +03:00
}
2001-03-16 08:55:30 +03:00
2002-09-25 19:19:00 +04:00
if ( ret = = 0 )
unlink ( pjob - > filename ) ;
/* else print_job_end will do it for us */
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return ret ;
2001-03-16 08:55:30 +03:00
}
/*
* ' cups_queue_get ( ) ' - Get all the jobs in the print queue .
*/
2005-02-12 17:41:00 +03:00
static int cups_queue_get ( const char * sharename ,
2004-10-19 21:05:01 +04:00
enum printing_types printing_type ,
char * lpq_command ,
print_queue_struct * * q ,
print_status_struct * status )
2001-03-16 08:55:30 +03:00
{
2005-02-12 17:41:00 +03:00
fstring printername ;
2005-01-21 03:29:38 +03:00
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
ipp_attribute_t * attr = NULL ; /* Current attribute */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
2005-01-21 03:29:38 +03:00
int qcount = 0 , /* Number of active queue entries */
qalloc = 0 ; /* Number of queue entries allocated */
print_queue_struct * queue = NULL , /* Queue entries */
2001-03-16 08:55:30 +03:00
* temp ; /* Temporary pointer for queue */
const char * user_name , /* job-originating-user-name attribute */
* job_name ; /* job-name attribute */
int job_id ; /* job-id attribute */
int job_k_octets ; /* job-k-octets attribute */
time_t job_time ; /* time-at-creation attribute */
ipp_jstate_t job_status ; /* job-status attribute */
int job_priority ; /* job-priority attribute */
static const char * jattrs [ ] = /* Requested job attributes */
{
" job-id " ,
" job-k-octets " ,
" job-name " ,
" job-originating-user-name " ,
" job-priority " ,
" job-state " ,
" time-at-creation " ,
} ;
static const char * pattrs [ ] = /* Requested printer attributes */
{
" printer-state " ,
" printer-state-message "
} ;
2005-01-21 03:29:38 +03:00
* q = NULL ;
2001-03-16 08:55:30 +03:00
2005-02-12 17:41:00 +03:00
/* HACK ALERT!!! The porblem with support the 'printer name'
option is that we key the tdb off the sharename . So we will
overload the lpq_command string to pass in the printername
( which is basically what we do for non - cups printers . . . using
the lpq_command to get the queue listing ) . */
fstrcpy ( printername , lpq_command ) ;
DEBUG ( 5 , ( " cups_queue_get(%s, %p, %p) \n " , printername , q , status ) ) ;
2001-03-16 08:55:30 +03:00
/*
* Make sure we don ' t ask for passwords . . .
*/
cupsSetPasswordCB ( cups_passwd_cb ) ;
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Generate the printer URI . . .
*/
2005-02-12 17:41:00 +03:00
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/printers/%s " , printername ) ;
2001-03-16 08:55:30 +03:00
/*
* Build an IPP_GET_JOBS request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* requested - attributes
* printer - uri
*/
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_GET_JOBS ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
ippAddStrings ( request , IPP_TAG_OPERATION , IPP_TAG_NAME ,
" requested-attributes " ,
( sizeof ( jattrs ) / sizeof ( jattrs [ 0 ] ) ) ,
NULL , jattrs ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI ,
" printer-uri " , NULL , uri ) ;
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " / " ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to get jobs for %s - %s \n " , uri ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to get jobs for %s - %s \n " , uri ,
ippErrorString ( response - > request . status . status_code ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Process the jobs . . .
*/
qcount = 0 ;
qalloc = 0 ;
queue = NULL ;
2005-01-21 03:29:38 +03:00
for ( attr = response - > attrs ; attr ! = NULL ; attr = attr - > next ) {
2001-03-16 08:55:30 +03:00
/*
* Skip leading attributes until we hit a job . . .
*/
while ( attr ! = NULL & & attr - > group_tag ! = IPP_TAG_JOB )
attr = attr - > next ;
if ( attr = = NULL )
break ;
/*
* Allocate memory as needed . . .
*/
2005-01-21 03:29:38 +03:00
if ( qcount > = qalloc ) {
2001-03-16 08:55:30 +03:00
qalloc + = 16 ;
2004-12-07 21:25:53 +03:00
temp = SMB_REALLOC_ARRAY ( queue , print_queue_struct , qalloc ) ;
2001-03-16 08:55:30 +03:00
2005-01-21 03:29:38 +03:00
if ( temp = = NULL ) {
2001-08-09 22:20:43 +04:00
DEBUG ( 0 , ( " cups_queue_get: Not enough memory! " ) ) ;
2005-01-21 03:29:38 +03:00
qcount = 0 ;
2002-07-15 14:35:28 +04:00
SAFE_FREE ( queue ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
queue = temp ;
}
temp = queue + qcount ;
memset ( temp , 0 , sizeof ( print_queue_struct ) ) ;
/*
* Pull the needed attributes from this job . . .
*/
job_id = 0 ;
job_priority = 50 ;
job_status = IPP_JOB_PENDING ;
job_time = 0 ;
job_k_octets = 0 ;
user_name = NULL ;
job_name = NULL ;
2005-01-21 03:29:38 +03:00
while ( attr ! = NULL & & attr - > group_tag = = IPP_TAG_JOB ) {
if ( attr - > name = = NULL ) {
2001-03-16 08:55:30 +03:00
attr = attr - > next ;
break ;
}
if ( strcmp ( attr - > name , " job-id " ) = = 0 & &
attr - > value_tag = = IPP_TAG_INTEGER )
job_id = attr - > values [ 0 ] . integer ;
if ( strcmp ( attr - > name , " job-k-octets " ) = = 0 & &
attr - > value_tag = = IPP_TAG_INTEGER )
job_k_octets = attr - > values [ 0 ] . integer ;
if ( strcmp ( attr - > name , " job-priority " ) = = 0 & &
attr - > value_tag = = IPP_TAG_INTEGER )
job_priority = attr - > values [ 0 ] . integer ;
if ( strcmp ( attr - > name , " job-state " ) = = 0 & &
attr - > value_tag = = IPP_TAG_ENUM )
job_status = ( ipp_jstate_t ) ( attr - > values [ 0 ] . integer ) ;
if ( strcmp ( attr - > name , " time-at-creation " ) = = 0 & &
attr - > value_tag = = IPP_TAG_INTEGER )
job_time = attr - > values [ 0 ] . integer ;
if ( strcmp ( attr - > name , " job-name " ) = = 0 & &
attr - > value_tag = = IPP_TAG_NAME )
job_name = attr - > values [ 0 ] . string . text ;
if ( strcmp ( attr - > name , " job-originating-user-name " ) = = 0 & &
attr - > value_tag = = IPP_TAG_NAME )
user_name = attr - > values [ 0 ] . string . text ;
attr = attr - > next ;
}
/*
* See if we have everything needed . . .
*/
2005-01-21 03:29:38 +03:00
if ( user_name = = NULL | | job_name = = NULL | | job_id = = 0 ) {
if ( attr = = NULL )
break ;
else
continue ;
2001-03-16 08:55:30 +03:00
}
temp - > job = job_id ;
temp - > size = job_k_octets * 1024 ;
temp - > status = job_status = = IPP_JOB_PENDING ? LPQ_QUEUED :
job_status = = IPP_JOB_STOPPED ? LPQ_PAUSED :
job_status = = IPP_JOB_HELD ? LPQ_PAUSED :
LPQ_PRINTING ;
temp - > priority = job_priority ;
temp - > time = job_time ;
2002-03-18 01:40:51 +03:00
strncpy ( temp - > fs_user , user_name , sizeof ( temp - > fs_user ) - 1 ) ;
strncpy ( temp - > fs_file , job_name , sizeof ( temp - > fs_file ) - 1 ) ;
2001-03-16 08:55:30 +03:00
qcount + + ;
if ( attr = = NULL )
2005-01-21 03:29:38 +03:00
break ;
2001-03-16 08:55:30 +03:00
}
ippDelete ( response ) ;
2005-01-21 03:29:38 +03:00
response = NULL ;
2001-03-16 08:55:30 +03:00
/*
* Build an IPP_GET_PRINTER_ATTRIBUTES request , which requires the
* following attributes :
*
* attributes - charset
* attributes - natural - language
* requested - attributes
* printer - uri
*/
2005-01-26 17:46:54 +03:00
request = ippNew ( ) ;
2001-03-16 08:55:30 +03:00
request - > request . op . operation_id = IPP_GET_PRINTER_ATTRIBUTES ;
request - > request . op . request_id = 1 ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
ippAddStrings ( request , IPP_TAG_OPERATION , IPP_TAG_NAME ,
" requested-attributes " ,
( sizeof ( pattrs ) / sizeof ( pattrs [ 0 ] ) ) ,
NULL , pattrs ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI ,
" printer-uri " , NULL , uri ) ;
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " / " ) ) = = NULL ) {
2005-02-12 17:41:00 +03:00
DEBUG ( 0 , ( " Unable to get printer status for %s - %s \n " , printername ,
2001-03-16 08:55:30 +03:00
ippErrorString ( cupsLastError ( ) ) ) ) ;
2001-08-08 20:54:16 +04:00
* q = queue ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
2005-02-12 17:41:00 +03:00
DEBUG ( 0 , ( " Unable to get printer status for %s - %s \n " , printername ,
2001-03-16 08:55:30 +03:00
ippErrorString ( response - > request . status . status_code ) ) ) ;
2001-08-08 20:54:16 +04:00
* q = queue ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Get the current printer status and convert it to the SAMBA values .
*/
2005-01-21 03:29:38 +03:00
if ( ( attr = ippFindAttribute ( response , " printer-state " , IPP_TAG_ENUM ) ) ! = NULL ) {
2001-03-16 08:55:30 +03:00
if ( attr - > values [ 0 ] . integer = = IPP_PRINTER_STOPPED )
status - > status = LPSTAT_STOPPED ;
else
status - > status = LPSTAT_OK ;
}
if ( ( attr = ippFindAttribute ( response , " printer-state-message " ,
IPP_TAG_TEXT ) ) ! = NULL )
fstrcpy ( status - > message , attr - > values [ 0 ] . string . text ) ;
/*
* Return the job queue . . .
*/
* q = queue ;
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return qcount ;
2001-03-16 08:55:30 +03:00
}
/*
* ' cups_queue_pause ( ) ' - Pause a print queue .
*/
2005-01-21 03:29:38 +03:00
static int cups_queue_pause ( int snum )
2001-03-16 08:55:30 +03:00
{
2001-07-23 23:50:36 +04:00
extern userdom_struct current_user_info ;
2005-01-21 03:29:38 +03:00
int ret = 1 ; /* Return value */
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
DEBUG ( 5 , ( " cups_queue_pause(%d) \n " , snum ) ) ;
2001-07-23 23:50:36 +04:00
/*
* Make sure we don ' t ask for passwords . . .
*/
2001-03-16 08:55:30 +03:00
cupsSetPasswordCB ( cups_passwd_cb ) ;
2001-07-23 23:50:36 +04:00
/*
* Try to connect to the server . . .
*/
2001-03-16 08:55:30 +03:00
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
2001-07-23 23:50:36 +04:00
/*
* Build an IPP_PAUSE_PRINTER request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* printer - uri
* requesting - user - name
*/
2001-03-16 08:55:30 +03:00
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_PAUSE_PRINTER ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/printers/%s " ,
PRINTERNAME ( snum ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI , " printer-uri " , NULL , uri ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " requesting-user-name " ,
2001-07-23 23:50:36 +04:00
NULL , current_user_info . unix_name ) ;
2001-03-16 08:55:30 +03:00
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " /admin/ " ) ) ! = NULL ) {
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
DEBUG ( 0 , ( " Unable to pause printer %s - %s \n " , PRINTERNAME ( snum ) ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
} else {
ret = 0 ;
}
} else {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to pause printer %s - %s \n " , PRINTERNAME ( snum ) ,
2005-01-21 03:29:38 +03:00
ippErrorString ( cupsLastError ( ) ) ) ) ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return ret ;
2001-03-16 08:55:30 +03:00
}
/*
* ' cups_queue_resume ( ) ' - Restart a print queue .
*/
2005-01-21 03:29:38 +03:00
static int cups_queue_resume ( int snum )
2001-03-16 08:55:30 +03:00
{
2001-07-24 00:47:55 +04:00
extern userdom_struct current_user_info ;
2005-01-21 03:29:38 +03:00
int ret = 1 ; /* Return value */
http_t * http = NULL ; /* HTTP connection to server */
ipp_t * request = NULL , /* IPP Request */
* response = NULL ; /* IPP Response */
cups_lang_t * language = NULL ; /* Default language */
2001-03-16 08:55:30 +03:00
char uri [ HTTP_MAX_URI ] ; /* printer-uri attribute */
DEBUG ( 5 , ( " cups_queue_resume(%d) \n " , snum ) ) ;
/*
* Make sure we don ' t ask for passwords . . .
*/
cupsSetPasswordCB ( cups_passwd_cb ) ;
/*
* Try to connect to the server . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( http = httpConnect ( cups_server ( ) , ippPort ( ) ) ) = = NULL ) {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to connect to CUPS server %s - %s \n " ,
2004-06-02 18:58:18 +04:00
cups_server ( ) , strerror ( errno ) ) ) ;
2005-01-21 03:29:38 +03:00
goto out ;
2001-03-16 08:55:30 +03:00
}
/*
* Build an IPP_RESUME_PRINTER request , which requires the following
* attributes :
*
* attributes - charset
* attributes - natural - language
* printer - uri
* requesting - user - name
*/
request = ippNew ( ) ;
request - > request . op . operation_id = IPP_RESUME_PRINTER ;
request - > request . op . request_id = 1 ;
language = cupsLangDefault ( ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_CHARSET ,
" attributes-charset " , NULL , cupsLangEncoding ( language ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_LANGUAGE ,
" attributes-natural-language " , NULL , language - > language ) ;
slprintf ( uri , sizeof ( uri ) - 1 , " ipp://localhost/printers/%s " ,
PRINTERNAME ( snum ) ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_URI , " printer-uri " , NULL , uri ) ;
ippAddString ( request , IPP_TAG_OPERATION , IPP_TAG_NAME , " requesting-user-name " ,
2001-07-24 00:47:55 +04:00
NULL , current_user_info . unix_name ) ;
2001-03-16 08:55:30 +03:00
/*
* Do the request and get back a response . . .
*/
2005-01-21 03:29:38 +03:00
if ( ( response = cupsDoRequest ( http , request , " /admin/ " ) ) ! = NULL ) {
if ( response - > request . status . status_code > = IPP_OK_CONFLICT ) {
DEBUG ( 0 , ( " Unable to resume printer %s - %s \n " , PRINTERNAME ( snum ) ,
ippErrorString ( cupsLastError ( ) ) ) ) ;
} else {
ret = 0 ;
}
} else {
2001-03-16 08:55:30 +03:00
DEBUG ( 0 , ( " Unable to resume printer %s - %s \n " , PRINTERNAME ( snum ) ,
2005-01-21 03:29:38 +03:00
ippErrorString ( cupsLastError ( ) ) ) ) ;
2001-03-16 08:55:30 +03:00
}
2005-01-21 03:29:38 +03:00
out :
if ( response )
ippDelete ( response ) ;
if ( language )
cupsLangFree ( language ) ;
if ( http )
httpClose ( http ) ;
return ret ;
2001-03-16 08:55:30 +03:00
}
2004-10-19 21:05:01 +04:00
/*******************************************************************
* CUPS printing interface definitions . . .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct printif cups_printif =
{
PRINT_CUPS ,
cups_queue_get ,
cups_queue_pause ,
cups_queue_resume ,
cups_job_delete ,
cups_job_pause ,
cups_job_resume ,
cups_job_submit ,
} ;
2001-03-16 08:55:30 +03:00
1999-12-17 04:46:15 +03:00
# else
/* this keeps fussy compilers happy */
void print_cups_dummy ( void ) { }
2001-08-23 23:06:20 +04:00
# endif /* HAVE_CUPS */