1996-05-04 07:50:46 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1996-05-04 07:50:46 +00:00
printcap parsing
1998-01-22 13:27:43 +00:00
Copyright ( C ) Karl Auer 1993 - 1998
1996-05-04 07:50:46 +00:00
Re - working by Martin Kiff , 1994
Re - written again by Andrew Tridgell
1997-12-03 05:08:07 +00:00
Modified for SVID support by Norm Jacobs , 1997
1999-12-13 13:27:58 +00:00
Modified for CUPS support by Michael Sweet , 1999
1996-05-04 07:50:46 +00: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 .
*/
/*
* Parse printcap file .
*
* This module does exactly one thing - it looks into the printcap file
* and tells callers if a specified string appears as a printer name .
*
* The way this module looks at the printcap file is very simplistic .
* Only the local printcap file is inspected ( no searching of NIS
* databases etc ) .
*
* There are assumed to be one or more printer names per record , held
* as a set of sub - fields separated by vertical bar symbols ( ' | ' ) in the
* first field of the record . The field separator is assumed to be a colon
* ' : ' and the record separator a newline .
*
* Lines ending with a backspace ' \ ' are assumed to flag that the following
* line is a continuation line so that a set of lines can be read as one
* printcap entry .
*
* A line stating with a hash ' # ' is assumed to be a comment and is ignored
* Comments are discarded before the record is strung together from the
* set of continuation lines .
*
* Opening a pipe for " lpc status " and reading that would probably
* be pretty effective . Code to do this already exists in the freely
* distributable PCNFS server code .
1997-12-03 05:08:07 +00:00
*
* Modified to call SVID / XPG4 support if printcap name is set to " lpstat "
* in smb . conf under Solaris .
1999-12-13 13:27:58 +00:00
*
* Modified to call CUPS support if printcap name is set to " cups "
* in smb . conf .
1996-05-04 07:50:46 +00:00
*/
# include "includes.h"
# include "smb.h"
# ifdef AIX
/* ******************************************
Extend for AIX system and qconfig file
from ' boulard @ univ - rennes1 . fr
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int strlocate ( char * xpLine , char * xpS )
{
1999-12-13 13:27:58 +00:00
int iS , iL , iRet ;
1996-05-04 07:50:46 +00:00
char * p ;
iS = strlen ( xpS ) ;
iL = strlen ( xpLine ) ;
iRet = 0 ;
p = xpLine ;
while ( iL > = iS )
{
if ( strncmp ( p , xpS , iS ) = = 0 ) { iRet = 1 ; break ; } ;
p + + ;
iL - - ;
}
/*DEBUG(3,(" strlocate %s in line '%s',ret=%d\n",xpS,xpLine,iRet));*/
return ( iRet ) ;
}
/* ******************************************************************* */
/* * Scan qconfig and search all virtual printer (device printer) * */
/* ******************************************************************* */
1999-12-13 13:27:58 +00:00
static void ScanQconfig_fn ( char * psz , void ( * fn ) ( char * , char * ) )
1996-05-04 07:50:46 +00:00
{
1999-12-13 13:27:58 +00:00
int iEtat ;
2001-09-10 12:46:42 +00:00
XFILE * pfile ;
1996-05-04 07:50:46 +00:00
char * line , * p ;
pstring name , comment ;
line = NULL ;
* name = 0 ;
* comment = 0 ;
2001-09-10 12:46:42 +00:00
if ( ( pfile = x_fopen ( psz , O_RDONLY , 0 ) ) = = NULL )
1996-05-04 07:50:46 +00:00
{
DEBUG ( 0 , ( " Unable to open qconfig file %s for read! \n " , psz ) ) ;
return ;
}
iEtat = 0 ;
/* scan qconfig file for searching <printername>: */
2002-07-15 10:35:28 +00:00
for ( ; ( line = fgets_slash ( NULL , sizeof ( pstring ) , pfile ) ) ; safe_free ( line ) )
1996-05-04 07:50:46 +00:00
{
if ( * line = = ' * ' | | * line = = 0 )
continue ;
switch ( iEtat )
{
case 0 : /* locate an entry */
if ( * line = = ' \t ' | | * line = = ' ' ) continue ;
2001-07-04 07:36:09 +00:00
if ( ( p = strchr_m ( line , ' : ' ) ) )
1996-05-04 07:50:46 +00:00
{
* p = ' \0 ' ;
p = strtok ( line , " : " ) ;
if ( strcmp ( p , " bsh " ) ! = 0 )
{
1998-05-12 00:55:32 +00:00
pstrcpy ( name , p ) ;
1996-05-04 07:50:46 +00:00
iEtat = 1 ;
continue ;
}
}
break ;
case 1 : /* scanning device stanza */
if ( * line = = ' * ' | | * line = = 0 ) continue ;
if ( * line ! = ' \t ' & & * line ! = ' ' )
{
/* name is found without stanza device */
/* probably a good printer ??? */
fn ( name , comment ) ;
iEtat = 0 ;
continue ;
}
if ( strlocate ( line , " backend " ) )
{
/* it's a device, not a virtual printer*/
iEtat = 0 ;
}
else if ( strlocate ( line , " device " ) )
{
/* it's a good virtual printer */
fn ( name , comment ) ;
iEtat = 0 ;
continue ;
}
break ;
}
}
2001-09-10 12:46:42 +00:00
x_fclose ( pfile ) ;
1996-05-04 07:50:46 +00:00
}
/* Scan qconfig file and locate de printername */
static BOOL ScanQconfig ( char * psz , char * pszPrintername )
{
int iLg , iEtat ;
2001-09-10 12:46:42 +00:00
XFILE * pfile ;
1996-05-04 07:50:46 +00:00
char * pName ;
char * line ;
pName = NULL ;
line = NULL ;
if ( ( pszPrintername ! = NULL ) & & ( ( iLg = strlen ( pszPrintername ) ) > 0 ) )
pName = malloc ( iLg + 10 ) ;
if ( pName = = NULL )
{
DEBUG ( 0 , ( " Unable to allocate memory for printer %s \n " , pszPrintername ) ) ;
return ( False ) ;
}
2001-09-10 12:46:42 +00:00
if ( ( pfile = x_fopen ( psz , O_RDONLY , 0 ) ) = = NULL )
1996-05-04 07:50:46 +00:00
{
DEBUG ( 0 , ( " Unable to open qconfig file %s for read! \n " , psz ) ) ;
2002-07-15 10:35:28 +00:00
SAFE_FREE ( pName ) ;
1996-05-04 07:50:46 +00:00
return ( False ) ;
}
1998-05-12 00:55:32 +00:00
slprintf ( pName , iLg + 9 , " %s: " , pszPrintername ) ;
1996-05-04 07:50:46 +00:00
iLg = strlen ( pName ) ;
/*DEBUG(3,( " Looking for entry %s\n",pName));*/
iEtat = 0 ;
/* scan qconfig file for searching <printername>: */
2002-07-15 10:35:28 +00:00
for ( ; ( line = fgets_slash ( NULL , sizeof ( pstring ) , pfile ) ) ; safe_free ( line ) )
1996-05-04 07:50:46 +00:00
{
if ( * line = = ' * ' | | * line = = 0 )
continue ;
switch ( iEtat )
{
case 0 : /* scanning entry */
if ( strncmp ( line , pName , iLg ) = = 0 )
{
iEtat = 1 ;
continue ;
}
break ;
case 1 : /* scanning device stanza */
if ( * line = = ' * ' | | * line = = 0 ) continue ;
if ( * line ! = ' \t ' & & * line ! = ' ' )
{
/* name is found without stanza device */
/* probably a good printer ??? */
free ( line ) ;
2002-07-15 10:35:28 +00:00
SAFE_FREE ( pName ) ;
1996-05-04 07:50:46 +00:00
fclose ( pfile ) ;
return ( True ) ;
}
if ( strlocate ( line , " backend " ) )
{
/* it's a device, not a virtual printer*/
iEtat = 0 ;
}
else if ( strlocate ( line , " device " ) )
{
/* it's a good virtual printer */
free ( line ) ;
2002-07-15 10:35:28 +00:00
SAFE_FREE ( pName ) ;
1996-05-04 07:50:46 +00:00
fclose ( pfile ) ;
return ( True ) ;
}
break ;
}
}
free ( pName ) ;
2001-09-10 12:46:42 +00:00
x_fclose ( pfile ) ;
1996-05-04 07:50:46 +00:00
return ( False ) ;
}
1998-07-29 03:08:05 +00:00
# endif /* AIX */
1996-05-04 07:50:46 +00:00
/***************************************************************************
Scan printcap file pszPrintcapname for a printer called pszPrintername .
Return True if found , else False . Returns False on error , too , after logging
the error at level 0. For generality , the printcap name may be passed - if
2002-08-17 17:00:51 +00:00
passed as NULL , the configuration will be queried for the name .
1996-05-04 07:50:46 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-08-17 17:00:51 +00:00
BOOL pcap_printername_ok ( const char * pszPrintername , const char * pszPrintcapname )
1996-05-04 07:50:46 +00:00
{
char * line = NULL ;
2002-08-17 17:00:51 +00:00
const char * psz ;
1996-05-04 07:50:46 +00:00
char * p , * q ;
2001-09-10 12:46:42 +00:00
XFILE * pfile ;
1996-05-04 07:50:46 +00:00
if ( pszPrintername = = NULL | | pszPrintername [ 0 ] = = ' \0 ' )
{
DEBUG ( 0 , ( " Attempt to locate null printername! Internal error? \n " ) ) ;
return ( False ) ;
}
/* only go looking if no printcap name supplied */
if ( ( psz = pszPrintcapname ) = = NULL | | psz [ 0 ] = = ' \0 ' )
if ( ( ( psz = lp_printcapname ( ) ) = = NULL ) | | ( psz [ 0 ] = = ' \0 ' ) )
{
DEBUG ( 0 , ( " No printcap file name configured! \n " ) ) ;
return ( False ) ;
}
1997-12-03 05:08:07 +00:00
2001-08-23 19:06:20 +00:00
# ifdef HAVE_CUPS
1999-12-13 13:27:58 +00:00
if ( strequal ( psz , " cups " ) )
return ( cups_printername_ok ( pszPrintername ) ) ;
2001-08-23 19:06:20 +00:00
# endif /* HAVE_CUPS */
1999-12-13 13:27:58 +00:00
1997-12-03 05:08:07 +00:00
# ifdef SYSV
if ( strequal ( psz , " lpstat " ) )
return ( sysv_printername_ok ( pszPrintername ) ) ;
# endif
1996-05-04 07:50:46 +00:00
# ifdef AIX
1998-01-16 06:29:38 +00:00
if ( strlocate ( psz , " /qconfig " ) )
1996-05-04 07:50:46 +00:00
return ( ScanQconfig ( psz , pszPrintername ) ) ;
# endif
1997-12-03 05:08:07 +00:00
2001-09-10 12:46:42 +00:00
if ( ( pfile = x_fopen ( psz , O_RDONLY , 0 ) ) = = NULL )
1996-05-04 07:50:46 +00:00
{
DEBUG ( 0 , ( " Unable to open printcap file %s for read! \n " , psz ) ) ;
return ( False ) ;
}
2002-07-15 10:35:28 +00:00
for ( ; ( line = fgets_slash ( NULL , sizeof ( pstring ) , pfile ) ) ; safe_free ( line ) )
1996-05-04 07:50:46 +00:00
{
if ( * line = = ' # ' | | * line = = 0 )
continue ;
/* now we have a real printer line - cut it off at the first : */
2001-07-04 07:36:09 +00:00
p = strchr_m ( line , ' : ' ) ;
1996-05-04 07:50:46 +00:00
if ( p ) * p = 0 ;
/* now just check if the name is in the list */
/* NOTE: I avoid strtok as the fn calling this one may be using it */
for ( p = line ; p ; p = q )
{
2001-07-04 07:36:09 +00:00
if ( ( q = strchr_m ( p , ' | ' ) ) ) * q + + = 0 ;
1996-05-04 07:50:46 +00:00
if ( strequal ( p , pszPrintername ) )
{
2002-07-15 10:35:28 +00:00
SAFE_FREE ( line ) ;
2001-09-10 12:46:42 +00:00
x_fclose ( pfile ) ;
1996-05-04 07:50:46 +00:00
return ( True ) ;
}
p = q ;
}
}
2001-09-10 12:46:42 +00:00
x_fclose ( pfile ) ;
1996-05-04 07:50:46 +00:00
return ( False ) ;
}
/***************************************************************************
run a function on each printer name in the printcap file . The function is
2001-01-17 18:47:46 +00:00
passed the primary name and the comment ( if possible ) . Note the fn ( ) takes
strings in DOS codepage . This means the xxx_printer_fn ( ) calls must be fixed
to return DOS codepage . FIXME ! ! JRA .
1996-05-04 07:50:46 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-04-13 19:24:06 +00:00
void pcap_printer_fn ( void ( * fn ) ( char * , char * ) )
1996-05-04 07:50:46 +00:00
{
pstring name , comment ;
char * line ;
char * psz ;
char * p , * q ;
2001-09-10 12:46:42 +00:00
XFILE * pfile ;
1996-05-04 07:50:46 +00:00
/* only go looking if no printcap name supplied */
if ( ( ( psz = lp_printcapname ( ) ) = = NULL ) | | ( psz [ 0 ] = = ' \0 ' ) )
{
DEBUG ( 0 , ( " No printcap file name configured! \n " ) ) ;
return ;
}
2001-08-23 19:06:20 +00:00
# ifdef HAVE_CUPS
1999-12-13 13:27:58 +00:00
if ( strequal ( psz , " cups " ) ) {
cups_printer_fn ( fn ) ;
return ;
}
2001-08-23 19:06:20 +00:00
# endif /* HAVE_CUPS */
1999-12-13 13:27:58 +00:00
1997-12-03 05:08:07 +00:00
# ifdef SYSV
if ( strequal ( psz , " lpstat " ) ) {
sysv_printer_fn ( fn ) ;
return ;
}
# endif
1996-05-04 07:50:46 +00:00
# ifdef AIX
1998-01-16 06:29:38 +00:00
if ( strlocate ( psz , " /qconfig " ) )
1996-05-04 07:50:46 +00:00
{
ScanQconfig_fn ( psz , fn ) ;
return ;
}
# endif
1997-12-03 05:08:07 +00:00
2001-09-10 12:46:42 +00:00
if ( ( pfile = x_fopen ( psz , O_RDONLY , 0 ) ) = = NULL )
1996-05-04 07:50:46 +00:00
{
DEBUG ( 0 , ( " Unable to open printcap file %s for read! \n " , psz ) ) ;
return ;
}
2002-07-15 10:35:28 +00:00
for ( ; ( line = fgets_slash ( NULL , sizeof ( pstring ) , pfile ) ) ; safe_free ( line ) )
1996-05-04 07:50:46 +00:00
{
if ( * line = = ' # ' | | * line = = 0 )
continue ;
/* now we have a real printer line - cut it off at the first : */
2001-07-04 07:36:09 +00:00
p = strchr_m ( line , ' : ' ) ;
1996-05-04 07:50:46 +00:00
if ( p ) * p = 0 ;
/* now find the most likely printer name and comment
this is pure guesswork , but it ' s better than nothing */
* name = 0 ;
* comment = 0 ;
for ( p = line ; p ; p = q )
{
BOOL has_punctuation ;
2001-07-04 07:36:09 +00:00
if ( ( q = strchr_m ( p , ' | ' ) ) ) * q + + = 0 ;
1996-05-04 07:50:46 +00:00
2001-07-04 07:36:09 +00:00
has_punctuation = ( strchr_m ( p , ' ' ) | | strchr_m ( p , ' \t ' ) | | strchr_m ( p , ' ( ' ) | | strchr_m ( p , ' ) ' ) ) ;
1996-05-04 07:50:46 +00:00
if ( strlen ( p ) > strlen ( comment ) & & has_punctuation )
{
StrnCpy ( comment , p , sizeof ( comment ) - 1 ) ;
continue ;
}
1998-03-20 13:37:46 +00:00
if ( strlen ( p ) < = MAXPRINTERLEN & & strlen ( p ) > strlen ( name ) & & ! has_punctuation )
1996-05-04 07:50:46 +00:00
{
1997-09-26 18:55:29 +00:00
if ( ! * comment ) pstrcpy ( comment , name ) ;
pstrcpy ( name , p ) ;
1996-05-04 07:50:46 +00:00
continue ;
}
2001-07-04 07:36:09 +00:00
if ( ! strchr_m ( comment , ' ' ) & &
1996-05-04 07:50:46 +00:00
strlen ( p ) > strlen ( comment ) )
{
StrnCpy ( comment , p , sizeof ( comment ) - 1 ) ;
continue ;
}
}
comment [ 60 ] = 0 ;
1998-03-20 13:37:46 +00:00
name [ MAXPRINTERLEN ] = 0 ;
1996-05-04 07:50:46 +00:00
2000-05-01 05:42:39 +00:00
if ( * name )
1996-05-04 07:50:46 +00:00
fn ( name , comment ) ;
}
2001-09-10 12:46:42 +00:00
x_fclose ( pfile ) ;
1996-05-04 07:50:46 +00:00
}