1*0c4db8c1Sderaadt /* $OpenBSD: lprm.c,v 1.19 2013/11/24 21:32:32 deraadt Exp $ */ 2a7643117Smillert /* $$NetBSD: lprm.c,v 1.9 1999/08/16 03:12:32 simonb Exp $ */ 3ca5d3c4eSmillert 4df930be7Sderaadt /* 5df930be7Sderaadt * Copyright (c) 1983, 1993 6df930be7Sderaadt * The Regents of the University of California. All rights reserved. 7df930be7Sderaadt * 8df930be7Sderaadt * 9df930be7Sderaadt * Redistribution and use in source and binary forms, with or without 10df930be7Sderaadt * modification, are permitted provided that the following conditions 11df930be7Sderaadt * are met: 12df930be7Sderaadt * 1. Redistributions of source code must retain the above copyright 13df930be7Sderaadt * notice, this list of conditions and the following disclaimer. 14df930be7Sderaadt * 2. Redistributions in binary form must reproduce the above copyright 15df930be7Sderaadt * notice, this list of conditions and the following disclaimer in the 16df930be7Sderaadt * documentation and/or other materials provided with the distribution. 1729295d1cSmillert * 3. Neither the name of the University nor the names of its contributors 18df930be7Sderaadt * may be used to endorse or promote products derived from this software 19df930be7Sderaadt * without specific prior written permission. 20df930be7Sderaadt * 21df930be7Sderaadt * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 22df930be7Sderaadt * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 23df930be7Sderaadt * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 24df930be7Sderaadt * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 25df930be7Sderaadt * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 26df930be7Sderaadt * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 27df930be7Sderaadt * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 28df930be7Sderaadt * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 29df930be7Sderaadt * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 30df930be7Sderaadt * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31df930be7Sderaadt * SUCH DAMAGE. 32df930be7Sderaadt */ 33df930be7Sderaadt 34df930be7Sderaadt /* 35df930be7Sderaadt * lprm - remove the current user's spool entry 36df930be7Sderaadt * 37df930be7Sderaadt * lprm [-] [[job #] [user] ...] 38df930be7Sderaadt * 39df930be7Sderaadt * Using information in the lock file, lprm will kill the 40df930be7Sderaadt * currently active daemon (if necessary), remove the associated files, 41a7643117Smillert * and startup a new daemon. Privileged users may remove anyone's spool 42df930be7Sderaadt * entries, otherwise one can only remove their own. 43df930be7Sderaadt */ 44df930be7Sderaadt 45df930be7Sderaadt #include <sys/param.h> 46df930be7Sderaadt 47a7643117Smillert #include <ctype.h> 48df930be7Sderaadt #include <dirent.h> 49a7643117Smillert #include <err.h> 506468ba68Smillert #include <errno.h> 51df930be7Sderaadt #include <pwd.h> 52df930be7Sderaadt #include <stdlib.h> 53df930be7Sderaadt #include <stdio.h> 54df930be7Sderaadt #include <string.h> 55a7643117Smillert #include <syslog.h> 56a7643117Smillert #include <unistd.h> 57a7643117Smillert 58df930be7Sderaadt #include "lp.h" 59df930be7Sderaadt #include "lp.local.h" 60df930be7Sderaadt 61df930be7Sderaadt /* 62df930be7Sderaadt * Stuff for handling job specifications 63df930be7Sderaadt */ 64df930be7Sderaadt char *person; /* name of person doing lprm */ 65df930be7Sderaadt int requ[MAXREQUESTS]; /* job number of spool entries */ 66df930be7Sderaadt int requests; /* # of spool requests */ 67df930be7Sderaadt char *user[MAXUSERS]; /* users to process */ 68df930be7Sderaadt int users; /* # of users in user array */ 69a7643117Smillert volatile sig_atomic_t gotintr; /* set when we receive SIGINT */ 7049848d36Sderaadt static char luser[MAXLOGNAME]; /* buffer for person */ 71df930be7Sderaadt 72a7643117Smillert static __dead void usage(void); 73df930be7Sderaadt 74df930be7Sderaadt int 75a7643117Smillert main(int argc, char **argv) 76df930be7Sderaadt { 779f6ae840Smillert struct passwd *pw; 78a7643117Smillert char *cp; 79a7643117Smillert long l; 80a7643117Smillert int ch; 81df930be7Sderaadt 829f6ae840Smillert /* 832d40ed46Smillert * Simulate setuid daemon w/ PRIV_END called. 849f6ae840Smillert * We don't want lpr to actually be setuid daemon since that 859f6ae840Smillert * requires that the lpr binary be owned by user daemon, which 869f6ae840Smillert * is potentially unsafe. 879f6ae840Smillert */ 889f6ae840Smillert if ((pw = getpwuid(DEFUID)) == NULL) 899f6ae840Smillert errx(1, "daemon uid (%u) not in password file", DEFUID); 909f6ae840Smillert effective_uid = pw->pw_uid; 916468ba68Smillert real_uid = getuid(); 929f6ae840Smillert effective_gid = pw->pw_gid; 936468ba68Smillert real_gid = getgid(); 942d40ed46Smillert setresgid(real_gid, real_gid, effective_gid); 952d40ed46Smillert setresuid(real_uid, real_uid, effective_uid); 96222c0752Sericj 97df930be7Sderaadt gethostname(host, sizeof(host)); 98a7643117Smillert openlog("lprm", 0, LOG_LPR); 999f6ae840Smillert if ((pw = getpwuid(real_uid)) == NULL) 100df930be7Sderaadt fatal("Who are you?"); 1019f6ae840Smillert if (strlen(pw->pw_name) >= sizeof(luser)) 102df930be7Sderaadt fatal("Your name is too long"); 1039f6ae840Smillert strlcpy(luser, pw->pw_name, sizeof(luser)); 104df930be7Sderaadt person = luser; 10502e888b3Smillert while ((ch = getopt(argc, argv, "P:w:-")) != -1) { 106a7643117Smillert switch (ch) { 107a7643117Smillert case '-': 108df930be7Sderaadt users = -1; 109df930be7Sderaadt break; 110a7643117Smillert case 'P': 111a7643117Smillert printer = optarg; 112a7643117Smillert break; 113a7643117Smillert case 'w': 114a7643117Smillert l = strtol(optarg, &cp, 10); 115a7643117Smillert if (*cp != '\0' || l < 0 || l >= INT_MAX) 116a7643117Smillert errx(1, "wait time must be postive integer: %s", 117a7643117Smillert optarg); 118a7643117Smillert wait_time = (u_int)l; 119a7643117Smillert if (wait_time < 30) 120a7643117Smillert warnx("warning: wait time less than 30 seconds"); 121a7643117Smillert break; 122df930be7Sderaadt default: 123df930be7Sderaadt usage(); 124df930be7Sderaadt } 125a7643117Smillert } 126a7643117Smillert argc -= optind; 127a7643117Smillert argv += optind; 128a7643117Smillert 129a7643117Smillert if (printer == NULL && (printer = getenv("PRINTER")) == NULL) 130a7643117Smillert printer = DEFLP; 131a7643117Smillert if (users < 0 && argc != 0) 132df930be7Sderaadt usage(); 133a7643117Smillert while (argc > 0) { 134*0c4db8c1Sderaadt if (isdigit((unsigned char)*argv[0])) { 135df930be7Sderaadt if (requests >= MAXREQUESTS) 136df930be7Sderaadt fatal("Too many requests"); 137a7643117Smillert requ[requests++] = atoi(argv[0]); 138df930be7Sderaadt } else { 139df930be7Sderaadt if (users >= MAXUSERS) 140df930be7Sderaadt fatal("Too many users"); 141a7643117Smillert user[users++] = argv[0]; 142df930be7Sderaadt } 143a7643117Smillert argc--; 144a7643117Smillert argv++; 14562831258Sderaadt } 146df930be7Sderaadt 147df930be7Sderaadt rmjob(); 148df930be7Sderaadt exit(0); 149df930be7Sderaadt } 150df930be7Sderaadt 151a7643117Smillert static __dead void 152a7643117Smillert usage(void) 153df930be7Sderaadt { 154a7643117Smillert extern char *__progname; 155a7643117Smillert 1562761ea90Sjmc fprintf(stderr, "usage: %s [-] [-Pprinter] [[job# ...] [user ...]]\n", 157a7643117Smillert __progname); 158df930be7Sderaadt exit(2); 159df930be7Sderaadt } 160