10b561052SJoerg Wunsch /* 20b561052SJoerg Wunsch * Copyright (c) 1983, 1993 30b561052SJoerg Wunsch * The Regents of the University of California. All rights reserved. 40b561052SJoerg Wunsch * 50b561052SJoerg Wunsch * Redistribution and use in source and binary forms, with or without 60b561052SJoerg Wunsch * modification, are permitted provided that the following conditions 70b561052SJoerg Wunsch * are met: 80b561052SJoerg Wunsch * 1. Redistributions of source code must retain the above copyright 90b561052SJoerg Wunsch * notice, this list of conditions and the following disclaimer. 100b561052SJoerg Wunsch * 2. Redistributions in binary form must reproduce the above copyright 110b561052SJoerg Wunsch * notice, this list of conditions and the following disclaimer in the 120b561052SJoerg Wunsch * documentation and/or other materials provided with the distribution. 130b561052SJoerg Wunsch * 3. All advertising materials mentioning features or use of this software 140b561052SJoerg Wunsch * must display the following acknowledgement: 150b561052SJoerg Wunsch * This product includes software developed by the University of 160b561052SJoerg Wunsch * California, Berkeley and its contributors. 170b561052SJoerg Wunsch * 4. Neither the name of the University nor the names of its contributors 180b561052SJoerg Wunsch * may be used to endorse or promote products derived from this software 190b561052SJoerg Wunsch * without specific prior written permission. 200b561052SJoerg Wunsch * 210b561052SJoerg Wunsch * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 220b561052SJoerg Wunsch * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 230b561052SJoerg Wunsch * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 240b561052SJoerg Wunsch * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 250b561052SJoerg Wunsch * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 260b561052SJoerg Wunsch * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 270b561052SJoerg Wunsch * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 280b561052SJoerg Wunsch * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 290b561052SJoerg Wunsch * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 300b561052SJoerg Wunsch * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 310b561052SJoerg Wunsch * SUCH DAMAGE. 320b561052SJoerg Wunsch */ 330b561052SJoerg Wunsch 341f589b47SGarance A Drosehn #if 0 35b7fd8699SGarance A Drosehn #ifndef lint 365458e2f4SJoerg Wunsch static char sccsid[] = "@(#)displayq.c 8.4 (Berkeley) 4/28/95"; 37b7fd8699SGarance A Drosehn #endif /* not lint */ 381f589b47SGarance A Drosehn #endif 39b7fd8699SGarance A Drosehn 401f589b47SGarance A Drosehn #include "lp.cdefs.h" /* A cross-platform version of <sys/cdefs.h> */ 411f589b47SGarance A Drosehn __FBSDID("$FreeBSD$"); 420b561052SJoerg Wunsch 430b561052SJoerg Wunsch #include <sys/param.h> 440b561052SJoerg Wunsch #include <sys/stat.h> 450b561052SJoerg Wunsch 464a1a0dbeSGarrett Wollman #include <ctype.h> 470b561052SJoerg Wunsch #include <dirent.h> 484a1a0dbeSGarrett Wollman #include <errno.h> 494a1a0dbeSGarrett Wollman #include <fcntl.h> 504a1a0dbeSGarrett Wollman #include <signal.h> 510b561052SJoerg Wunsch #include <stdio.h> 520b561052SJoerg Wunsch #include <stdlib.h> 530b561052SJoerg Wunsch #include <string.h> 544a1a0dbeSGarrett Wollman #define psignal foil_gcc_psignal 554a1a0dbeSGarrett Wollman #define sys_siglist foil_gcc_siglist 564a1a0dbeSGarrett Wollman #include <unistd.h> 574a1a0dbeSGarrett Wollman #undef psignal 584a1a0dbeSGarrett Wollman #undef sys_siglist 594a1a0dbeSGarrett Wollman 600b561052SJoerg Wunsch #include "lp.h" 610b561052SJoerg Wunsch #include "lp.local.h" 620b561052SJoerg Wunsch #include "pathnames.h" 630b561052SJoerg Wunsch 640b561052SJoerg Wunsch /* 650b561052SJoerg Wunsch * Routines to display the state of the queue. 660b561052SJoerg Wunsch */ 670b561052SJoerg Wunsch #define JOBCOL 40 /* column for job # in -l format */ 680b561052SJoerg Wunsch #define OWNCOL 7 /* start of Owner column in normal */ 690b561052SJoerg Wunsch #define SIZCOL 62 /* start of Size column in normal */ 700b561052SJoerg Wunsch 710b561052SJoerg Wunsch /* 7246bfa198SGarance A Drosehn * isprint() takes a parameter of 'int', but expect values in the range 7346bfa198SGarance A Drosehn * of unsigned char. Define a wrapper which takes a value of type 'char', 7446bfa198SGarance A Drosehn * whether signed or unsigned, and ensure it ends up in the right range. 7546bfa198SGarance A Drosehn */ 7646bfa198SGarance A Drosehn #define isprintch(Anychar) isprint((u_char)(Anychar)) 7746bfa198SGarance A Drosehn 7846bfa198SGarance A Drosehn /* 790b561052SJoerg Wunsch * Stuff for handling job specifications 800b561052SJoerg Wunsch */ 81360d4ad5SWarner Losh extern uid_t uid, euid; 82360d4ad5SWarner Losh 830b561052SJoerg Wunsch static int col; /* column on screen */ 847b7fb4bbSKris Kennaway static char current[MAXNAMLEN+1]; /* current file being printed */ 857b7fb4bbSKris Kennaway static char file[MAXNAMLEN+1]; /* print file name */ 860b561052SJoerg Wunsch static int first; /* first file in ``files'' column? */ 870b561052SJoerg Wunsch static int garbage; /* # of garbage cf files */ 880b561052SJoerg Wunsch static int lflag; /* long output option */ 890b561052SJoerg Wunsch static int rank; /* order to be printed (-1=none, 0=active) */ 900b561052SJoerg Wunsch static long totsize; /* total print job size in bytes */ 910b561052SJoerg Wunsch 92ba7a1ad7SGarance A Drosehn static const char *head0 = "Rank Owner Job Files"; 93ba7a1ad7SGarance A Drosehn static const char *head1 = "Total Size\n"; 940b561052SJoerg Wunsch 95ba7a1ad7SGarance A Drosehn static void alarmhandler(int _signo); 9646bfa198SGarance A Drosehn static void filtered_write(char *_obuffer, int _wlen, FILE *_wstream); 97ba7a1ad7SGarance A Drosehn static void warn(const struct printer *_pp); 98334a9508SJoerg Wunsch 990b561052SJoerg Wunsch /* 1000b561052SJoerg Wunsch * Display the current state of the queue. Format = 1 if long format. 1010b561052SJoerg Wunsch */ 1020b561052SJoerg Wunsch void 103ba7a1ad7SGarance A Drosehn displayq(struct printer *pp, int format) 1040b561052SJoerg Wunsch { 10530b4b758SGarance A Drosehn register struct jobqueue *q; 106360d4ad5SWarner Losh register int i, nitems, fd, ret; 1077b7fb4bbSKris Kennaway char *cp, *endp; 10830b4b758SGarance A Drosehn struct jobqueue **queue; 1090b561052SJoerg Wunsch struct stat statb; 1100b561052SJoerg Wunsch FILE *fp; 111334a9508SJoerg Wunsch void (*savealrm)(int); 1120b561052SJoerg Wunsch 1130b561052SJoerg Wunsch lflag = format; 1140b561052SJoerg Wunsch totsize = 0; 1150b561052SJoerg Wunsch rank = -1; 1164a1a0dbeSGarrett Wollman 1174a1a0dbeSGarrett Wollman if ((cp = checkremote(pp))) { 1180b561052SJoerg Wunsch printf("Warning: %s\n", cp); 1194a1a0dbeSGarrett Wollman free(cp); 1204a1a0dbeSGarrett Wollman } 1210b561052SJoerg Wunsch 1220b561052SJoerg Wunsch /* 1230b561052SJoerg Wunsch * Print out local queue 1240b561052SJoerg Wunsch * Find all the control files in the spooling directory 1250b561052SJoerg Wunsch */ 126360d4ad5SWarner Losh seteuid(euid); 1274a1a0dbeSGarrett Wollman if (chdir(pp->spool_dir) < 0) 1284a1a0dbeSGarrett Wollman fatal(pp, "cannot chdir to spooling directory: %s", 1294a1a0dbeSGarrett Wollman strerror(errno)); 130360d4ad5SWarner Losh seteuid(uid); 1314a1a0dbeSGarrett Wollman if ((nitems = getq(pp, &queue)) < 0) 1324a1a0dbeSGarrett Wollman fatal(pp, "cannot examine spooling area\n"); 133360d4ad5SWarner Losh seteuid(euid); 1344a1a0dbeSGarrett Wollman ret = stat(pp->lock_file, &statb); 135360d4ad5SWarner Losh seteuid(uid); 136360d4ad5SWarner Losh if (ret >= 0) { 1374a1a0dbeSGarrett Wollman if (statb.st_mode & LFM_PRINT_DIS) { 1384a1a0dbeSGarrett Wollman if (pp->remote) 139cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 1404a1a0dbeSGarrett Wollman printf("Warning: %s is down: ", pp->printer); 141360d4ad5SWarner Losh seteuid(euid); 1424a1a0dbeSGarrett Wollman fd = open(pp->status_file, O_RDONLY|O_SHLOCK); 143360d4ad5SWarner Losh seteuid(uid); 1440b561052SJoerg Wunsch if (fd >= 0) { 1450b561052SJoerg Wunsch while ((i = read(fd, line, sizeof(line))) > 0) 1460b561052SJoerg Wunsch (void) fwrite(line, 1, i, stdout); 1470b561052SJoerg Wunsch (void) close(fd); /* unlocks as well */ 1480b561052SJoerg Wunsch } else 1490b561052SJoerg Wunsch putchar('\n'); 1500b561052SJoerg Wunsch } 1514a1a0dbeSGarrett Wollman if (statb.st_mode & LFM_QUEUE_DIS) { 1524a1a0dbeSGarrett Wollman if (pp->remote) 153cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 1544a1a0dbeSGarrett Wollman printf("Warning: %s queue is turned off\n", 1554a1a0dbeSGarrett Wollman pp->printer); 1560b561052SJoerg Wunsch } 1570b561052SJoerg Wunsch } 1580b561052SJoerg Wunsch 1590b561052SJoerg Wunsch if (nitems) { 160360d4ad5SWarner Losh seteuid(euid); 1614a1a0dbeSGarrett Wollman fp = fopen(pp->lock_file, "r"); 162360d4ad5SWarner Losh seteuid(uid); 1630b561052SJoerg Wunsch if (fp == NULL) 1644a1a0dbeSGarrett Wollman warn(pp); 1650b561052SJoerg Wunsch else { 1660b561052SJoerg Wunsch /* get daemon pid */ 1670b561052SJoerg Wunsch cp = current; 1687b7fb4bbSKris Kennaway endp = cp + sizeof(current) - 1; 1697b7fb4bbSKris Kennaway while ((i = getc(fp)) != EOF && i != '\n') { 1707b7fb4bbSKris Kennaway if (cp < endp) 1715458e2f4SJoerg Wunsch *cp++ = i; 1727b7fb4bbSKris Kennaway } 1730b561052SJoerg Wunsch *cp = '\0'; 1740b561052SJoerg Wunsch i = atoi(current); 175360d4ad5SWarner Losh if (i <= 0) { 176360d4ad5SWarner Losh ret = -1; 177360d4ad5SWarner Losh } else { 178360d4ad5SWarner Losh seteuid(euid); 179360d4ad5SWarner Losh ret = kill(i, 0); 180360d4ad5SWarner Losh seteuid(uid); 181360d4ad5SWarner Losh } 182360d4ad5SWarner Losh if (ret < 0) { 1834a1a0dbeSGarrett Wollman warn(pp); 184360d4ad5SWarner Losh } else { 1850b561052SJoerg Wunsch /* read current file name */ 1860b561052SJoerg Wunsch cp = current; 1877b7fb4bbSKris Kennaway endp = cp + sizeof(current) - 1; 1887b7fb4bbSKris Kennaway while ((i = getc(fp)) != EOF && i != '\n') { 1897b7fb4bbSKris Kennaway if (cp < endp) 1905458e2f4SJoerg Wunsch *cp++ = i; 1917b7fb4bbSKris Kennaway } 1920b561052SJoerg Wunsch *cp = '\0'; 1930b561052SJoerg Wunsch /* 1940b561052SJoerg Wunsch * Print the status file. 1950b561052SJoerg Wunsch */ 1964a1a0dbeSGarrett Wollman if (pp->remote) 197cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 198360d4ad5SWarner Losh seteuid(euid); 1994a1a0dbeSGarrett Wollman fd = open(pp->status_file, O_RDONLY|O_SHLOCK); 200360d4ad5SWarner Losh seteuid(uid); 2010b561052SJoerg Wunsch if (fd >= 0) { 2024a1a0dbeSGarrett Wollman while ((i = read(fd, line, 2034a1a0dbeSGarrett Wollman sizeof(line))) > 0) 2044a1a0dbeSGarrett Wollman fwrite(line, 1, i, stdout); 2054a1a0dbeSGarrett Wollman close(fd); /* unlocks as well */ 2060b561052SJoerg Wunsch } else 2070b561052SJoerg Wunsch putchar('\n'); 2080b561052SJoerg Wunsch } 2090b561052SJoerg Wunsch (void) fclose(fp); 2100b561052SJoerg Wunsch } 2110b561052SJoerg Wunsch /* 2120b561052SJoerg Wunsch * Now, examine the control files and print out the jobs to 2130b561052SJoerg Wunsch * be done for each user. 2140b561052SJoerg Wunsch */ 2150b561052SJoerg Wunsch if (!lflag) 2160b561052SJoerg Wunsch header(); 2170b561052SJoerg Wunsch for (i = 0; i < nitems; i++) { 2180b561052SJoerg Wunsch q = queue[i]; 21930b4b758SGarance A Drosehn inform(pp, q->job_cfname); 2200b561052SJoerg Wunsch free(q); 2210b561052SJoerg Wunsch } 2220b561052SJoerg Wunsch free(queue); 2230b561052SJoerg Wunsch } 2244a1a0dbeSGarrett Wollman if (!pp->remote) { 2250b561052SJoerg Wunsch if (nitems == 0) 2260b561052SJoerg Wunsch puts("no entries"); 2270b561052SJoerg Wunsch return; 2280b561052SJoerg Wunsch } 2290b561052SJoerg Wunsch 2300b561052SJoerg Wunsch /* 2310b561052SJoerg Wunsch * Print foreign queue 2320b561052SJoerg Wunsch * Note that a file in transit may show up in either queue. 2330b561052SJoerg Wunsch */ 2340b561052SJoerg Wunsch if (nitems) 2350b561052SJoerg Wunsch putchar('\n'); 2364a1a0dbeSGarrett Wollman (void) snprintf(line, sizeof(line), "%c%s", format ? '\4' : '\3', 2374a1a0dbeSGarrett Wollman pp->remote_queue); 2380b561052SJoerg Wunsch cp = line; 2396ee8b269SWarner Losh for (i = 0; i < requests && cp-line+10 < sizeof(line) - 1; i++) { 2400b561052SJoerg Wunsch cp += strlen(cp); 2410b561052SJoerg Wunsch (void) sprintf(cp, " %d", requ[i]); 2420b561052SJoerg Wunsch } 243d583a7c3SWarner Losh for (i = 0; i < users && cp - line + 1 + strlen(user[i]) < 2446ee8b269SWarner Losh sizeof(line) - 1; i++) { 2450b561052SJoerg Wunsch cp += strlen(cp); 2460b561052SJoerg Wunsch *cp++ = ' '; 2470b561052SJoerg Wunsch (void) strcpy(cp, user[i]); 2480b561052SJoerg Wunsch } 2490b561052SJoerg Wunsch strcat(line, "\n"); 250334a9508SJoerg Wunsch savealrm = signal(SIGALRM, alarmhandler); 2514a1a0dbeSGarrett Wollman alarm(pp->conn_timeout); 2524a1a0dbeSGarrett Wollman fd = getport(pp, pp->remote_host, 0); 2536d0727f4SJoerg Wunsch alarm(0); 254334a9508SJoerg Wunsch (void)signal(SIGALRM, savealrm); 2550b561052SJoerg Wunsch if (fd < 0) { 256cc3fd56fSGarance A Drosehn if (from_host != local_host) 257cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 2584a1a0dbeSGarrett Wollman printf("connection to %s is down\n", pp->remote_host); 2590b561052SJoerg Wunsch } 2600b561052SJoerg Wunsch else { 2610b561052SJoerg Wunsch i = strlen(line); 2620b561052SJoerg Wunsch if (write(fd, line, i) != i) 2634a1a0dbeSGarrett Wollman fatal(pp, "Lost connection"); 2640b561052SJoerg Wunsch while ((i = read(fd, line, sizeof(line))) > 0) 26546bfa198SGarance A Drosehn filtered_write(line, i, stdout); 26646bfa198SGarance A Drosehn filtered_write(NULL, -1, stdout); 2670b561052SJoerg Wunsch (void) close(fd); 2680b561052SJoerg Wunsch } 2690b561052SJoerg Wunsch } 2700b561052SJoerg Wunsch 2710b561052SJoerg Wunsch /* 27246bfa198SGarance A Drosehn * The lpq-info read from remote hosts may contain unprintable characters, 27346bfa198SGarance A Drosehn * or carriage-returns instead of line-feeds. Clean those up before echoing 27446bfa198SGarance A Drosehn * the lpq-info line(s) to stdout. The info may also be missing any kind of 27546bfa198SGarance A Drosehn * end-of-line character. This also turns CRLF and LFCR into a plain LF. 27646bfa198SGarance A Drosehn * 27746bfa198SGarance A Drosehn * This routine may be called multiple times to process a single set of 27846bfa198SGarance A Drosehn * information, and after a set is finished this routine must be called 27946bfa198SGarance A Drosehn * one extra time with NULL specified as the buffer address. 28046bfa198SGarance A Drosehn */ 28146bfa198SGarance A Drosehn static void 28246bfa198SGarance A Drosehn filtered_write(char *wbuffer, int wlen, FILE *wstream) 28346bfa198SGarance A Drosehn { 28446bfa198SGarance A Drosehn static char lastchar, savedchar; 28546bfa198SGarance A Drosehn char *chkptr, *dest_end, *dest_ch, *nxtptr, *w_end; 28646bfa198SGarance A Drosehn int destlen; 28746bfa198SGarance A Drosehn char destbuf[BUFSIZ]; 28846bfa198SGarance A Drosehn 28946bfa198SGarance A Drosehn if (wbuffer == NULL) { 29046bfa198SGarance A Drosehn if (savedchar != '\0') { 29146bfa198SGarance A Drosehn if (savedchar == '\r') 29246bfa198SGarance A Drosehn savedchar = '\n'; 29346bfa198SGarance A Drosehn fputc(savedchar, wstream); 29446bfa198SGarance A Drosehn lastchar = savedchar; 29546bfa198SGarance A Drosehn savedchar = '\0'; 29646bfa198SGarance A Drosehn } 29746bfa198SGarance A Drosehn if (lastchar != '\0' && lastchar != '\n') 29846bfa198SGarance A Drosehn fputc('\n', wstream); 29946bfa198SGarance A Drosehn lastchar = '\0'; 30046bfa198SGarance A Drosehn return; 30146bfa198SGarance A Drosehn } 30246bfa198SGarance A Drosehn 30346bfa198SGarance A Drosehn dest_ch = &destbuf[0]; 30446bfa198SGarance A Drosehn dest_end = dest_ch + sizeof(destbuf); 30546bfa198SGarance A Drosehn chkptr = wbuffer; 30646bfa198SGarance A Drosehn w_end = wbuffer + wlen; 30746bfa198SGarance A Drosehn lastchar = '\0'; 30846bfa198SGarance A Drosehn if (savedchar != '\0') { 30946bfa198SGarance A Drosehn chkptr = &savedchar; 31046bfa198SGarance A Drosehn nxtptr = wbuffer; 31146bfa198SGarance A Drosehn } else 31246bfa198SGarance A Drosehn nxtptr = chkptr + 1; 31346bfa198SGarance A Drosehn 31446bfa198SGarance A Drosehn while (chkptr < w_end) { 31546bfa198SGarance A Drosehn if (nxtptr < w_end) { 31646bfa198SGarance A Drosehn if ((*chkptr == '\r' && *nxtptr == '\n') || 31746bfa198SGarance A Drosehn (*chkptr == '\n' && *nxtptr == '\r')) { 31846bfa198SGarance A Drosehn *dest_ch++ = '\n'; 31946bfa198SGarance A Drosehn /* want to skip past that second character */ 32046bfa198SGarance A Drosehn nxtptr++; 32146bfa198SGarance A Drosehn goto check_next; 32246bfa198SGarance A Drosehn } 32346bfa198SGarance A Drosehn } else { 32446bfa198SGarance A Drosehn /* This is the last byte in the buffer given on this 32546bfa198SGarance A Drosehn * call, so check if it could be the first-byte of a 32646bfa198SGarance A Drosehn * significant two-byte sequence. If it is, then 32746bfa198SGarance A Drosehn * don't write it out now, but save for checking in 32846bfa198SGarance A Drosehn * the next call. 32946bfa198SGarance A Drosehn */ 33046bfa198SGarance A Drosehn savedchar = '\0'; 33146bfa198SGarance A Drosehn if (*chkptr == '\r' || *chkptr == '\n') { 33246bfa198SGarance A Drosehn savedchar = *chkptr; 33346bfa198SGarance A Drosehn break; 33446bfa198SGarance A Drosehn } 33546bfa198SGarance A Drosehn } 33646bfa198SGarance A Drosehn if (*chkptr == '\r') 33746bfa198SGarance A Drosehn *dest_ch++ = '\n'; 33846bfa198SGarance A Drosehn #if 0 /* XXX - don't translate unprintable characters (yet) */ 33946bfa198SGarance A Drosehn else if (*chkptr != '\t' && *chkptr != '\n' && 34046bfa198SGarance A Drosehn !isprintch(*chkptr)) 34146bfa198SGarance A Drosehn *dest_ch++ = '?'; 34246bfa198SGarance A Drosehn #endif 34346bfa198SGarance A Drosehn else 34446bfa198SGarance A Drosehn *dest_ch++ = *chkptr; 34546bfa198SGarance A Drosehn 34646bfa198SGarance A Drosehn check_next: 34746bfa198SGarance A Drosehn chkptr = nxtptr; 34846bfa198SGarance A Drosehn nxtptr = chkptr + 1; 34946bfa198SGarance A Drosehn if (dest_ch >= dest_end) { 35046bfa198SGarance A Drosehn destlen = dest_ch - &destbuf[0]; 35146bfa198SGarance A Drosehn fwrite(destbuf, 1, destlen, wstream); 35246bfa198SGarance A Drosehn lastchar = destbuf[destlen - 1]; 35346bfa198SGarance A Drosehn dest_ch = &destbuf[0]; 35446bfa198SGarance A Drosehn } 35546bfa198SGarance A Drosehn } 35646bfa198SGarance A Drosehn destlen = dest_ch - &destbuf[0]; 35746bfa198SGarance A Drosehn if (destlen > 0) { 35846bfa198SGarance A Drosehn fwrite(destbuf, 1, destlen, wstream); 35946bfa198SGarance A Drosehn lastchar = destbuf[destlen - 1]; 36046bfa198SGarance A Drosehn } 36146bfa198SGarance A Drosehn } 36246bfa198SGarance A Drosehn 36346bfa198SGarance A Drosehn /* 3640b561052SJoerg Wunsch * Print a warning message if there is no daemon present. 3650b561052SJoerg Wunsch */ 36636d0e2a3SJoerg Wunsch static void 367ba7a1ad7SGarance A Drosehn warn(const struct printer *pp) 3680b561052SJoerg Wunsch { 3694a1a0dbeSGarrett Wollman if (pp->remote) 370cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 3710b561052SJoerg Wunsch puts("Warning: no daemon present"); 3720b561052SJoerg Wunsch current[0] = '\0'; 3730b561052SJoerg Wunsch } 3740b561052SJoerg Wunsch 3750b561052SJoerg Wunsch /* 3760b561052SJoerg Wunsch * Print the header for the short listing format 3770b561052SJoerg Wunsch */ 3780b561052SJoerg Wunsch void 379ba7a1ad7SGarance A Drosehn header(void) 3800b561052SJoerg Wunsch { 3816d39e1b7SGarance A Drosehn printf("%s", head0); 3820b561052SJoerg Wunsch col = strlen(head0)+1; 3830b561052SJoerg Wunsch blankfill(SIZCOL); 3846d39e1b7SGarance A Drosehn printf("%s", head1); 3850b561052SJoerg Wunsch } 3860b561052SJoerg Wunsch 3870b561052SJoerg Wunsch void 388ba7a1ad7SGarance A Drosehn inform(const struct printer *pp, char *cf) 3890b561052SJoerg Wunsch { 390c547dbe8SGarance A Drosehn int copycnt, jnum; 391ec6b8da5SGarance A Drosehn char savedname[MAXPATHLEN+1]; 3920b561052SJoerg Wunsch FILE *cfp; 3930b561052SJoerg Wunsch 3940b561052SJoerg Wunsch /* 3950b561052SJoerg Wunsch * There's a chance the control file has gone away 3960b561052SJoerg Wunsch * in the meantime; if this is the case just keep going 3970b561052SJoerg Wunsch */ 398360d4ad5SWarner Losh seteuid(euid); 3990b561052SJoerg Wunsch if ((cfp = fopen(cf, "r")) == NULL) 4000b561052SJoerg Wunsch return; 401360d4ad5SWarner Losh seteuid(uid); 4020b561052SJoerg Wunsch 4030b561052SJoerg Wunsch if (rank < 0) 4040b561052SJoerg Wunsch rank = 0; 4054a1a0dbeSGarrett Wollman if (pp->remote || garbage || strcmp(cf, current)) 4060b561052SJoerg Wunsch rank++; 407ec6b8da5SGarance A Drosehn 408ec6b8da5SGarance A Drosehn /* 409ec6b8da5SGarance A Drosehn * The cf-file may include commands to print more than one datafile 410ec6b8da5SGarance A Drosehn * from the user. For each datafile, the cf-file contains at least 411ec6b8da5SGarance A Drosehn * one line which starts with some format-specifier ('a'-'z'), and 412ec6b8da5SGarance A Drosehn * a second line ('N'ame) which indicates the original name the user 413ec6b8da5SGarance A Drosehn * specified for that file. There can be multiple format-spec lines 414ec6b8da5SGarance A Drosehn * for a single Name-line, if the user requested multiple copies of 415ec6b8da5SGarance A Drosehn * that file. Standard lpr puts the format-spec line(s) before the 416ec6b8da5SGarance A Drosehn * Name-line, while lprNG puts the Name-line before the format-spec 417ec6b8da5SGarance A Drosehn * line(s). This section needs to handle the lines in either order. 418ec6b8da5SGarance A Drosehn */ 419ec6b8da5SGarance A Drosehn copycnt = 0; 420ec6b8da5SGarance A Drosehn file[0] = '\0'; 421ec6b8da5SGarance A Drosehn savedname[0] = '\0'; 422c547dbe8SGarance A Drosehn jnum = calc_jobnum(cf, NULL); 4230b561052SJoerg Wunsch while (getline(cfp)) { 4240b561052SJoerg Wunsch switch (line[0]) { 4250b561052SJoerg Wunsch case 'P': /* Was this file specified in the user's list? */ 4260b561052SJoerg Wunsch if (!inlist(line+1, cf)) { 4270b561052SJoerg Wunsch fclose(cfp); 4280b561052SJoerg Wunsch return; 4290b561052SJoerg Wunsch } 4300b561052SJoerg Wunsch if (lflag) { 4310b561052SJoerg Wunsch printf("\n%s: ", line+1); 4320b561052SJoerg Wunsch col = strlen(line+1) + 2; 4330b561052SJoerg Wunsch prank(rank); 4340b561052SJoerg Wunsch blankfill(JOBCOL); 4350b561052SJoerg Wunsch printf(" [job %s]\n", cf+3); 4360b561052SJoerg Wunsch } else { 4370b561052SJoerg Wunsch col = 0; 4380b561052SJoerg Wunsch prank(rank); 4390b561052SJoerg Wunsch blankfill(OWNCOL); 440c547dbe8SGarance A Drosehn printf("%-10s %-3d ", line+1, jnum); 4410b561052SJoerg Wunsch col += 16; 4420b561052SJoerg Wunsch first = 1; 4430b561052SJoerg Wunsch } 4440b561052SJoerg Wunsch continue; 4450b561052SJoerg Wunsch default: /* some format specifer and file name? */ 4460b561052SJoerg Wunsch if (line[0] < 'a' || line[0] > 'z') 447ec6b8da5SGarance A Drosehn break; 448ec6b8da5SGarance A Drosehn if (copycnt == 0 || strcmp(file, line+1) != 0) { 4495d7321f6SGarance A Drosehn strlcpy(file, line + 1, sizeof(file)); 450d583a7c3SWarner Losh } 451ec6b8da5SGarance A Drosehn copycnt++; 452ec6b8da5SGarance A Drosehn /* 453ec6b8da5SGarance A Drosehn * deliberately 'continue' to another getline(), so 454ec6b8da5SGarance A Drosehn * all format-spec lines for this datafile are read 455ec6b8da5SGarance A Drosehn * in and counted before calling show() 456ec6b8da5SGarance A Drosehn */ 4570b561052SJoerg Wunsch continue; 4580b561052SJoerg Wunsch case 'N': 4595d7321f6SGarance A Drosehn strlcpy(savedname, line + 1, sizeof(savedname)); 460ec6b8da5SGarance A Drosehn break; 461ec6b8da5SGarance A Drosehn } 462ec6b8da5SGarance A Drosehn if ((file[0] != '\0') && (savedname[0] != '\0')) { 463ec6b8da5SGarance A Drosehn show(savedname, file, copycnt); 464ec6b8da5SGarance A Drosehn copycnt = 0; 4650b561052SJoerg Wunsch file[0] = '\0'; 466ec6b8da5SGarance A Drosehn savedname[0] = '\0'; 4670b561052SJoerg Wunsch } 4680b561052SJoerg Wunsch } 4690b561052SJoerg Wunsch fclose(cfp); 470ec6b8da5SGarance A Drosehn /* check for a file which hasn't been shown yet */ 471ec6b8da5SGarance A Drosehn if (file[0] != '\0') { 472ec6b8da5SGarance A Drosehn if (savedname[0] == '\0') { 473ec6b8da5SGarance A Drosehn /* a safeguard in case the N-ame line is missing */ 4745d7321f6SGarance A Drosehn strlcpy(savedname, file, sizeof(savedname)); 475ec6b8da5SGarance A Drosehn } 476ec6b8da5SGarance A Drosehn show(savedname, file, copycnt); 477ec6b8da5SGarance A Drosehn } 4780b561052SJoerg Wunsch if (!lflag) { 4790b561052SJoerg Wunsch blankfill(SIZCOL); 4800b561052SJoerg Wunsch printf("%ld bytes\n", totsize); 4810b561052SJoerg Wunsch totsize = 0; 4820b561052SJoerg Wunsch } 4830b561052SJoerg Wunsch } 4840b561052SJoerg Wunsch 4850b561052SJoerg Wunsch int 486ba7a1ad7SGarance A Drosehn inlist(char *uname, char *cfile) 4870b561052SJoerg Wunsch { 488c547dbe8SGarance A Drosehn int *r, jnum; 489c547dbe8SGarance A Drosehn char **u; 490c547dbe8SGarance A Drosehn const char *cfhost; 4910b561052SJoerg Wunsch 4920b561052SJoerg Wunsch if (users == 0 && requests == 0) 4930b561052SJoerg Wunsch return(1); 4940b561052SJoerg Wunsch /* 4950b561052SJoerg Wunsch * Check to see if it's in the user list 4960b561052SJoerg Wunsch */ 4970b561052SJoerg Wunsch for (u = user; u < &user[users]; u++) 498ba7a1ad7SGarance A Drosehn if (!strcmp(*u, uname)) 4990b561052SJoerg Wunsch return(1); 5000b561052SJoerg Wunsch /* 5010b561052SJoerg Wunsch * Check the request list 5020b561052SJoerg Wunsch */ 503c547dbe8SGarance A Drosehn jnum = calc_jobnum(cfile, &cfhost); 5040b561052SJoerg Wunsch for (r = requ; r < &requ[requests]; r++) 505c547dbe8SGarance A Drosehn if (*r == jnum && !strcmp(cfhost, from_host)) 5060b561052SJoerg Wunsch return(1); 5070b561052SJoerg Wunsch return(0); 5080b561052SJoerg Wunsch } 5090b561052SJoerg Wunsch 5100b561052SJoerg Wunsch void 511ba7a1ad7SGarance A Drosehn show(const char *nfile, const char *datafile, int copies) 5120b561052SJoerg Wunsch { 5130b561052SJoerg Wunsch if (strcmp(nfile, " ") == 0) 5140b561052SJoerg Wunsch nfile = "(standard input)"; 5150b561052SJoerg Wunsch if (lflag) 516ba7a1ad7SGarance A Drosehn ldump(nfile, datafile, copies); 5170b561052SJoerg Wunsch else 518ba7a1ad7SGarance A Drosehn dump(nfile, datafile, copies); 5190b561052SJoerg Wunsch } 5200b561052SJoerg Wunsch 5210b561052SJoerg Wunsch /* 5220b561052SJoerg Wunsch * Fill the line with blanks to the specified column 5230b561052SJoerg Wunsch */ 5240b561052SJoerg Wunsch void 525ba7a1ad7SGarance A Drosehn blankfill(int tocol) 5260b561052SJoerg Wunsch { 527ba7a1ad7SGarance A Drosehn while (col++ < tocol) 5280b561052SJoerg Wunsch putchar(' '); 5290b561052SJoerg Wunsch } 5300b561052SJoerg Wunsch 5310b561052SJoerg Wunsch /* 5320b561052SJoerg Wunsch * Give the abbreviated dump of the file names 5330b561052SJoerg Wunsch */ 5340b561052SJoerg Wunsch void 535ba7a1ad7SGarance A Drosehn dump(const char *nfile, const char *datafile, int copies) 5360b561052SJoerg Wunsch { 5370b561052SJoerg Wunsch struct stat lbuf; 538be794da7SGarance A Drosehn const char etctmpl[] = ", ..."; 539be794da7SGarance A Drosehn char etc[sizeof(etctmpl)]; 540be794da7SGarance A Drosehn char *lastsep; 541be794da7SGarance A Drosehn short fill, nlen; 542be794da7SGarance A Drosehn short rem, remetc; 5430b561052SJoerg Wunsch 5440b561052SJoerg Wunsch /* 545be794da7SGarance A Drosehn * Print as many filenames as will fit 546be794da7SGarance A Drosehn * (leaving room for the 'total size' field) 5470b561052SJoerg Wunsch */ 5480b561052SJoerg Wunsch fill = first ? 0 : 2; /* fill space for ``, '' */ 549be794da7SGarance A Drosehn nlen = strlen(nfile); 550be794da7SGarance A Drosehn rem = SIZCOL - 1 - col; 551be794da7SGarance A Drosehn if (nlen + fill > rem) { 552be794da7SGarance A Drosehn if (first) { 553be794da7SGarance A Drosehn /* print the right-most part of the name */ 554be794da7SGarance A Drosehn printf("...%s ", &nfile[3+nlen-rem]); 555be794da7SGarance A Drosehn col = SIZCOL; 556be794da7SGarance A Drosehn } else if (rem > 0) { 557be794da7SGarance A Drosehn /* fit as much of the etc-string as we can */ 558be794da7SGarance A Drosehn remetc = rem; 559be794da7SGarance A Drosehn if (rem > strlen(etctmpl)) 560be794da7SGarance A Drosehn remetc = strlen(etctmpl); 561be794da7SGarance A Drosehn etc[0] = '\0'; 562be794da7SGarance A Drosehn strncat(etc, etctmpl, remetc); 5636d39e1b7SGarance A Drosehn printf("%s", etc); 564be794da7SGarance A Drosehn col += remetc; 565be794da7SGarance A Drosehn rem -= remetc; 566be794da7SGarance A Drosehn /* room for the last segment of this filename? */ 567be794da7SGarance A Drosehn lastsep = strrchr(nfile, '/'); 568be794da7SGarance A Drosehn if ((lastsep != NULL) && (rem > strlen(lastsep))) { 569be794da7SGarance A Drosehn /* print the right-most part of this name */ 570be794da7SGarance A Drosehn printf("%s", lastsep); 571be794da7SGarance A Drosehn col += strlen(lastsep); 572be794da7SGarance A Drosehn } else { 573be794da7SGarance A Drosehn /* do not pack any more names in here */ 5740b561052SJoerg Wunsch blankfill(SIZCOL); 5750b561052SJoerg Wunsch } 576be794da7SGarance A Drosehn } 5770b561052SJoerg Wunsch } else { 578be794da7SGarance A Drosehn if (!first) 5790b561052SJoerg Wunsch printf(", "); 5800b561052SJoerg Wunsch printf("%s", nfile); 581be794da7SGarance A Drosehn col += nlen + fill; 5820b561052SJoerg Wunsch } 583be794da7SGarance A Drosehn first = 0; 584be794da7SGarance A Drosehn 585360d4ad5SWarner Losh seteuid(euid); 586ba7a1ad7SGarance A Drosehn if (*datafile && !stat(datafile, &lbuf)) 5870b561052SJoerg Wunsch totsize += copies * lbuf.st_size; 588360d4ad5SWarner Losh seteuid(uid); 5890b561052SJoerg Wunsch } 5900b561052SJoerg Wunsch 5910b561052SJoerg Wunsch /* 5920b561052SJoerg Wunsch * Print the long info about the file 5930b561052SJoerg Wunsch */ 5940b561052SJoerg Wunsch void 595ba7a1ad7SGarance A Drosehn ldump(const char *nfile, const char *datafile, int copies) 5960b561052SJoerg Wunsch { 5970b561052SJoerg Wunsch struct stat lbuf; 5980b561052SJoerg Wunsch 5990b561052SJoerg Wunsch putchar('\t'); 6000b561052SJoerg Wunsch if (copies > 1) 6010b561052SJoerg Wunsch printf("%-2d copies of %-19s", copies, nfile); 6020b561052SJoerg Wunsch else 6030b561052SJoerg Wunsch printf("%-32s", nfile); 604ba7a1ad7SGarance A Drosehn if (*datafile && !stat(datafile, &lbuf)) 605ef1c4c53SJohn Birrell printf(" %qd bytes", (long long) lbuf.st_size); 6060b561052SJoerg Wunsch else 6070b561052SJoerg Wunsch printf(" ??? bytes"); 6080b561052SJoerg Wunsch putchar('\n'); 6090b561052SJoerg Wunsch } 6100b561052SJoerg Wunsch 6110b561052SJoerg Wunsch /* 6120b561052SJoerg Wunsch * Print the job's rank in the queue, 6130b561052SJoerg Wunsch * update col for screen management 6140b561052SJoerg Wunsch */ 6150b561052SJoerg Wunsch void 616ba7a1ad7SGarance A Drosehn prank(int n) 6170b561052SJoerg Wunsch { 6180b561052SJoerg Wunsch char rline[100]; 619ba7a1ad7SGarance A Drosehn static const char *r[] = { 6200b561052SJoerg Wunsch "th", "st", "nd", "rd", "th", "th", "th", "th", "th", "th" 6210b561052SJoerg Wunsch }; 6220b561052SJoerg Wunsch 6230b561052SJoerg Wunsch if (n == 0) { 6240b561052SJoerg Wunsch printf("active"); 6250b561052SJoerg Wunsch col += 6; 6260b561052SJoerg Wunsch return; 6270b561052SJoerg Wunsch } 6280b561052SJoerg Wunsch if ((n/10)%10 == 1) 6290b561052SJoerg Wunsch (void)snprintf(rline, sizeof(rline), "%dth", n); 6300b561052SJoerg Wunsch else 6310b561052SJoerg Wunsch (void)snprintf(rline, sizeof(rline), "%d%s", n, r[n%10]); 6320b561052SJoerg Wunsch col += strlen(rline); 6330b561052SJoerg Wunsch printf("%s", rline); 6340b561052SJoerg Wunsch } 635334a9508SJoerg Wunsch 636334a9508SJoerg Wunsch void 637ba7a1ad7SGarance A Drosehn alarmhandler(int signo __unused) 638334a9508SJoerg Wunsch { 639ba7a1ad7SGarance A Drosehn /* the signal is ignored */ 640ba7a1ad7SGarance A Drosehn /* (the '__unused' is just to avoid a compile-time warning) */ 641334a9508SJoerg Wunsch } 642