/at-3.1.13/at.c

# · C · 961 lines · 671 code · 167 blank · 123 comment · 201 complexity · dc983574c91e9987b94abce3b7e5858b MD5 · raw file

  1. /*
  2. * at.c : Put file into atd queue
  3. * Copyright (C) 1993, 1994, 1995, 1996, 1997 Thomas Koenig
  4. * Copyright (C) 2002, 2005 Ryan Murray
  5. *
  6. * Atrun & Atq modifications
  7. * Copyright (C) 1993 David Parsons
  8. *
  9. * This program is free software; you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License as published by
  11. * the Free Software Foundation; either version 2 of the License, or
  12. * (at your option) any later version.
  13. *
  14. * This program is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. * GNU General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU General Public License
  20. * along with this program; if not, write to the Free Software
  21. * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  22. */
  23. #ifdef HAVE_CONFIG_H
  24. #include "config.h"
  25. #endif
  26. /* System Headers */
  27. #include <sys/types.h>
  28. #include <sys/stat.h>
  29. #ifdef HAVE_SYS_WAIT_H
  30. #include <sys/wait.h>
  31. #endif
  32. #include <ctype.h>
  33. #ifdef HAVE_DIRENT_H
  34. #include <dirent.h>
  35. #elif HAVE_SYS_DIRENT_H
  36. #include <sys/dirent.h>
  37. #elif HAVE_SYS_DIR_H
  38. #include <sys/dir.h>
  39. #endif
  40. #ifdef HAVE_ERRNO_H
  41. #include <errno.h>
  42. #endif
  43. #ifdef HAVE_FCNTL_H
  44. #include <fcntl.h>
  45. #elif defined(HAVE_SYS_FCNTL_H)
  46. #include <sys/fcntl.h>
  47. #endif
  48. #include <pwd.h>
  49. #include <grp.h>
  50. #include <signal.h>
  51. #include <stddef.h>
  52. #include <stdio.h>
  53. #include <stdlib.h>
  54. #include <string.h>
  55. #include <sys/time.h>
  56. #include <time.h>
  57. #ifdef HAVE_UNISTD_H
  58. #include <unistd.h>
  59. #endif
  60. /* Local headers */
  61. #include "at.h"
  62. #include "panic.h"
  63. #include "parsetime.h"
  64. #include "perm.h"
  65. #include "posixtm.h"
  66. #include "privs.h"
  67. /* Macros */
  68. #ifndef ATJOB_MX
  69. #define ATJOB_MX 255
  70. #endif
  71. #define ALARMC 10 /* Number of seconds to wait for timeout */
  72. #define SIZE 255
  73. #define TIMEFORMAT_POSIX "%a %b %e %T %Y"
  74. #define TIMESIZE 50
  75. #define DEFAULT_QUEUE 'a'
  76. #define BATCH_QUEUE 'b'
  77. enum {
  78. ATQ, BATCH, ATRM, AT, CAT
  79. }; /* what program we want to run */
  80. /* Global variables */
  81. uid_t real_uid, effective_uid;
  82. gid_t real_gid, effective_gid;
  83. uid_t daemon_uid = (uid_t) - 3;
  84. gid_t daemon_gid = (gid_t) - 3;
  85. /* File scope variables */
  86. char *no_export[] =
  87. {
  88. "TERM", "DISPLAY", "_", "SHELLOPTS", "BASH_VERSINFO", "EUID", "GROUPS", "PPID", "UID"
  89. };
  90. static int send_mail = 0;
  91. /* External variables */
  92. extern char **environ;
  93. int fcreated;
  94. char *namep;
  95. char atfile[] = ATJOB_DIR "/12345678901234";
  96. char *atinput = (char *) 0; /* where to get input from */
  97. char atqueue = 0; /* which queue to examine for jobs (atq) */
  98. char atverify = 0; /* verify time instead of queuing job */
  99. /* Function declarations */
  100. static void sigc(int signo);
  101. static void alarmc(int signo);
  102. static char *cwdname(void);
  103. static void writefile(time_t runtimer, char queue);
  104. static void list_jobs(void);
  105. /* Signal catching functions */
  106. static RETSIGTYPE
  107. sigc(int signo)
  108. {
  109. /* If the user presses ^C, remove the spool file and exit
  110. */
  111. if (fcreated) {
  112. PRIV_START
  113. /*
  114. We need the unprivileged uid here since the file is owned by the real
  115. (not effective) uid.
  116. */
  117. unlink(atfile);
  118. PRIV_END
  119. }
  120. exit(EXIT_FAILURE);
  121. }
  122. static void
  123. alarmc(int signo)
  124. {
  125. /* Time out after some seconds
  126. */
  127. panic("File locking timed out");
  128. }
  129. /* Local functions */
  130. static char *
  131. cwdname(void)
  132. {
  133. /* Read in the current directory; the name will be overwritten on
  134. * subsequent calls.
  135. */
  136. static char *ptr = NULL;
  137. static size_t size = SIZE;
  138. if (ptr == NULL)
  139. ptr = (char *) mymalloc(size);
  140. while (1) {
  141. if (ptr == NULL)
  142. panic("Out of memory");
  143. if (getcwd(ptr, size - 1) != NULL)
  144. return ptr;
  145. if (errno != ERANGE)
  146. perr("Cannot get current working directory");
  147. free(ptr);
  148. size += SIZE;
  149. ptr = (char *) mymalloc(size);
  150. }
  151. }
  152. static long
  153. nextjob()
  154. {
  155. long jobno;
  156. FILE *fid;
  157. jobno = 0;
  158. fid = fopen(LFILE, "r+");
  159. if (fid != NULL) {
  160. fscanf(fid, "%5lx", &jobno);
  161. rewind(fid);
  162. } else {
  163. fid = fopen(LFILE, "w");
  164. if (fid == NULL)
  165. return EOF;
  166. }
  167. jobno = (1 + jobno) % 0xfffff; /* 2^20 jobs enough? */
  168. fprintf(fid, "%05lx\n", jobno);
  169. fclose(fid);
  170. return jobno;
  171. }
  172. static void
  173. writefile(time_t runtimer, char queue)
  174. {
  175. /* This does most of the work if at or batch are invoked for writing a job.
  176. */
  177. long jobno;
  178. char *ap, *ppos, *mailname;
  179. struct passwd *pass_entry;
  180. struct stat statbuf;
  181. int fd, lockdes, fd2;
  182. FILE *fp, *fpin;
  183. struct sigaction act;
  184. char **atenv;
  185. int ch;
  186. mode_t cmask;
  187. struct flock lock;
  188. struct tm *runtime;
  189. char timestr[TIMESIZE];
  190. pid_t pid;
  191. int istty;
  192. int kill_errno;
  193. int rc;
  194. int mailsize = 128;
  195. struct timeval tv;
  196. struct timezone tz;
  197. long int i;
  198. gettimeofday(&tv, &tz);
  199. srandom(getpid()+tv.tv_usec);
  200. /* Install the signal handler for SIGINT; terminate after removing the
  201. * spool file if necessary
  202. */
  203. memset(&act, 0, sizeof act);
  204. act.sa_handler = sigc;
  205. sigemptyset(&(act.sa_mask));
  206. act.sa_flags = 0;
  207. sigaction(SIGINT, &act, NULL);
  208. ppos = atfile + strlen(ATJOB_DIR) + 1;
  209. #ifdef _SC_LOGIN_NAME_MAX
  210. errno = 0;
  211. rc = sysconf(_SC_LOGIN_NAME_MAX);
  212. if (rc > 0)
  213. mailsize = rc;
  214. #else
  215. # ifdef LOGIN_NAME_MAX
  216. mailsize = LOGIN_NAME_MAX;
  217. # endif
  218. #endif
  219. /* Loop over all possible file names for running something at this
  220. * particular time, see if a file is there; the first empty slot at any
  221. * particular time is used. Lock the file LFILE first to make sure
  222. * we're alone when doing this.
  223. */
  224. PRIV_START
  225. if ((lockdes = open(LFILE, O_WRONLY)) < 0)
  226. perr("Cannot open lockfile " LFILE);
  227. lock.l_type = F_WRLCK;
  228. lock.l_whence = SEEK_SET;
  229. lock.l_start = 0;
  230. lock.l_len = 0;
  231. act.sa_handler = alarmc;
  232. sigemptyset(&(act.sa_mask));
  233. act.sa_flags = 0;
  234. /* Set an alarm so a timeout occurs after ALARMC seconds, in case
  235. * something is seriously broken.
  236. */
  237. sigaction(SIGALRM, &act, NULL);
  238. alarm(ALARMC);
  239. fcntl(lockdes, F_SETLKW, &lock);
  240. alarm(0);
  241. if ((jobno = nextjob()) == EOF)
  242. perr("Cannot generate job number");
  243. (void)snprintf(ppos, sizeof(atfile) - (ppos - atfile),
  244. "%c%5lx%8lx", queue, jobno, (unsigned long) (runtimer / 60));
  245. for (ap = ppos; *ap != '\0'; ap++)
  246. if (*ap == ' ')
  247. *ap = '0';
  248. /*if (stat(atfile, &statbuf) != 0) {
  249. if (errno != ENOENT)
  250. perr("Cannot access " ATJOB_DIR);
  251. } else {
  252. perr("atjob file already exists; bailing");
  253. }
  254. */
  255. /* Create the file. The x bit is only going to be set after it has
  256. * been completely written out, to make sure it is not executed in the
  257. * meantime. To make sure they do not get deleted, turn off their r
  258. * bit. Yes, this is a kluge.
  259. */
  260. cmask = umask(S_IRUSR | S_IWUSR | S_IXUSR);
  261. if ((seteuid(effective_uid)) < 0)
  262. perr("Error in seteuid: %s", errno);
  263. if ((fd = open(atfile, O_CREAT | O_EXCL | O_TRUNC | O_WRONLY, S_IRUSR)) == -1)
  264. perr("Cannot create atjob file %.500s", atfile);
  265. if ((fd2 = dup(fd)) < 0)
  266. perr("Error in dup() of job file");
  267. if (fchown(fd2, real_uid, real_gid) != 0)
  268. perr("Cannot give real_uid and real_gid the file");
  269. PRIV_END
  270. /* We've successfully created the file; let's set the flag so it
  271. * gets removed in case of an interrupt or error.
  272. */
  273. fcreated = 1;
  274. /* Now we can release the lock, so other people can access it
  275. */
  276. lock.l_type = F_UNLCK;
  277. lock.l_whence = SEEK_SET;
  278. lock.l_start = 0;
  279. lock.l_len = 0;
  280. fcntl(lockdes, F_SETLKW, &lock);
  281. close(lockdes);
  282. if ((fp = fdopen(fd, "w")) == NULL)
  283. panic("Cannot reopen atjob file");
  284. /* Get the userid to mail to, first by trying getlogin(), which reads
  285. * /var/run/utmp, then from LOGNAME, finally from getpwuid().
  286. */
  287. mailname = getlogin();
  288. if (mailname == NULL)
  289. mailname = getenv("LOGNAME");
  290. if (mailname == NULL || mailname[0] == '\0' || getpwnam(mailname) == NULL) {
  291. pass_entry = getpwuid(real_uid);
  292. if (pass_entry != NULL)
  293. mailname = pass_entry->pw_name;
  294. }
  295. if ((mailname == NULL) || (mailname[0] == '\0')
  296. || (strlen(mailname) > mailsize) ) {
  297. panic("Cannot find username to mail output to");
  298. }
  299. if (atinput != (char *) NULL) {
  300. fpin = freopen(atinput, "r", stdin);
  301. if (fpin == NULL)
  302. perr("Cannot open input file %.500s", atinput);
  303. }
  304. fprintf(fp, "#!/bin/sh\n# atrun uid=%d gid=%d\n# mail %s %d\n",
  305. real_uid, real_gid, mailname, send_mail);
  306. /* Write out the umask at the time of invocation
  307. */
  308. fprintf(fp, "umask %lo\n", (unsigned long) cmask);
  309. /* Write out the environment. Anything that may look like a
  310. * special character to the shell is quoted, except for \n, which is
  311. * done with a pair of ""'s. Dont't export the no_export list (such
  312. * as TERM or DISPLAY) because we don't want these.
  313. */
  314. for (atenv = environ; *atenv != NULL; atenv++) {
  315. int export = 1;
  316. char *eqp;
  317. eqp = strchr(*atenv, '=');
  318. if (ap == NULL)
  319. eqp = *atenv;
  320. else {
  321. unsigned int i;
  322. for (i = 0; i < sizeof(no_export) / sizeof(no_export[0]); i++) {
  323. export = export
  324. && ((((size_t) (eqp - *atenv)) != strlen(no_export[i]))
  325. ||(strncmp(*atenv, no_export[i],(size_t) (eqp - *atenv)) != 0)
  326. );
  327. }
  328. eqp++;
  329. }
  330. if (export) {
  331. fwrite(*atenv, sizeof(char), eqp - *atenv, fp);
  332. for (ap = eqp; *ap != '\0'; ap++) {
  333. if (*ap == '\n')
  334. fprintf(fp, "\"\n\"");
  335. else {
  336. if (!isalnum(*ap)) {
  337. switch (*ap) {
  338. case '%':
  339. case '/':
  340. case '{':
  341. case '[':
  342. case ']':
  343. case '=':
  344. case '}':
  345. case '@':
  346. case '+':
  347. case '#':
  348. case ',':
  349. case '.':
  350. case ':':
  351. case '-':
  352. case '_':
  353. break;
  354. default:
  355. fputc('\\', fp);
  356. break;
  357. }
  358. }
  359. fputc(*ap, fp);
  360. }
  361. }
  362. fputs("; export ", fp);
  363. fwrite(*atenv, sizeof(char), eqp - *atenv - 1, fp);
  364. fputc('\n', fp);
  365. }
  366. }
  367. /* Cd to the directory at the time and write out all the
  368. * commands the user supplies from stdin.
  369. */
  370. fprintf(fp, "cd ");
  371. for (ap = cwdname(); *ap != '\0'; ap++) {
  372. if (*ap == '\n')
  373. fprintf(fp, "\"\n\"");
  374. else {
  375. if (*ap != '/' && !isalnum(*ap))
  376. fputc('\\', fp);
  377. fputc(*ap, fp);
  378. }
  379. }
  380. /* Test cd's exit status: die if the original directory has been
  381. * removed, become unreadable or whatever
  382. */
  383. fprintf(fp, " || {\n\t echo 'Execution directory "
  384. "inaccessible' >&2\n\t exit 1\n}\n");
  385. i = random();
  386. fprintf(fp, "${SHELL:-/bin/sh} << \'marcinDELIMITER%08lx\'\n", i);
  387. istty = isatty(fileno(stdin));
  388. if (istty) {
  389. fprintf(stderr, "at> ");
  390. fflush(stderr);
  391. }
  392. while ((ch = getchar()) != EOF) {
  393. fputc(ch, fp);
  394. if (ch == '\n' && istty) {
  395. fprintf(stderr, "at> ");
  396. fflush(stderr);
  397. }
  398. }
  399. if (istty) {
  400. fprintf(stderr, "<EOT>\n");
  401. }
  402. fprintf(fp, "\n");
  403. fprintf(fp, "marcinDELIMITER%08lx\n", i);
  404. if (ferror(fp))
  405. panic("Output error");
  406. if (ferror(stdin))
  407. panic("Input error");
  408. fclose(fp);
  409. /* Set the x bit so that we're ready to start executing
  410. */
  411. if (fchmod(fd2, S_IRUSR | S_IWUSR | S_IXUSR) < 0)
  412. perr("Cannot give away file");
  413. close(fd2);
  414. runtime = localtime(&runtimer);
  415. strftime(timestr, TIMESIZE, TIMEFORMAT_POSIX, runtime);
  416. fprintf(stderr, "job %ld at %s\n", jobno, timestr);
  417. /* Signal atd, if present. Usual precautions taken... */
  418. fd = open(PIDFILE, O_RDONLY);
  419. if (fd == -1) {
  420. fprintf(stderr, "Can't open " PIDFILE " to signal atd. No atd running?\n");
  421. return;
  422. }
  423. if (fstat(fd, &statbuf) == -1)
  424. return;
  425. if ((statbuf.st_uid != 0) || !S_ISREG(statbuf.st_mode) ||
  426. (statbuf.st_mode & (S_IWGRP | S_IWOTH)))
  427. return;
  428. fp = fdopen(fd, "r");
  429. if (fp == NULL)
  430. return;
  431. if (fscanf(fp, "%d", &pid) != 1)
  432. return;
  433. kill_errno = 0;
  434. PRIV_START
  435. if (kill(pid, SIGHUP) == -1)
  436. kill_errno = errno;
  437. PRIV_END
  438. switch (kill_errno) {
  439. case 0:
  440. break;
  441. case EINVAL:
  442. panic("kill returned EINVAL");
  443. break;
  444. case EPERM:
  445. fprintf(stderr,"Can't signal atd (permission denied)\n");
  446. break;
  447. case ESRCH:
  448. fprintf(stderr, "Warning: at daemon not running\n");
  449. break;
  450. default:
  451. panic("kill returned impossible error number");
  452. break;
  453. }
  454. return;
  455. }
  456. static void
  457. list_jobs(void)
  458. {
  459. /* List all a user's jobs in the queue, by looping through ATJOB_DIR,
  460. * or everybody's if we are root
  461. */
  462. DIR *spool;
  463. struct dirent *dirent;
  464. struct stat buf;
  465. struct tm *runtime;
  466. unsigned long ctm;
  467. char queue;
  468. long jobno;
  469. time_t runtimer;
  470. char timestr[TIMESIZE];
  471. struct passwd *pwd;
  472. PRIV_START
  473. if (chdir(ATJOB_DIR) != 0)
  474. perr("Cannot change to " ATJOB_DIR);
  475. if ((spool = opendir(".")) == NULL)
  476. perr("Cannot open " ATJOB_DIR);
  477. /* Loop over every file in the directory
  478. */
  479. while ((dirent = readdir(spool)) != NULL) {
  480. if (stat(dirent->d_name, &buf) != 0)
  481. perr("Cannot stat in " ATJOB_DIR);
  482. /* See it's a regular file and is the user's */
  483. if (!S_ISREG(buf.st_mode)
  484. || ((buf.st_uid != real_uid) && !(real_uid == 0))
  485. || atverify)
  486. continue;
  487. if (sscanf(dirent->d_name, "%c%5lx%8lx", &queue, &jobno, &ctm) != 3)
  488. continue;
  489. if (atqueue && (queue != atqueue))
  490. continue;
  491. runtimer = 60 * (time_t) ctm;
  492. runtime = localtime(&runtimer);
  493. strftime(timestr, TIMESIZE, TIMEFORMAT_POSIX, runtime);
  494. if ((pwd = getpwuid(buf.st_uid)))
  495. printf("%ld\t%s %c %s\n", jobno, timestr, queue, pwd->pw_name);
  496. else
  497. printf("%ld\t%s %c\n", jobno, timestr, queue);
  498. }
  499. PRIV_END
  500. }
  501. static int
  502. process_jobs(int argc, char **argv, int what)
  503. {
  504. /* Delete every argument (job - ID) given
  505. */
  506. int i;
  507. struct stat buf;
  508. DIR *spool;
  509. struct dirent *dirent;
  510. unsigned long ctm;
  511. char queue;
  512. long jobno;
  513. int rc = EXIT_SUCCESS;
  514. int done;
  515. for (i = optind; i < argc; i++) {
  516. done = 0;
  517. PRIV_START
  518. if (chdir(ATJOB_DIR) != 0)
  519. perr("Cannot change to " ATJOB_DIR);
  520. if ((spool = opendir(".")) == NULL)
  521. perr("Cannot open " ATJOB_DIR);
  522. PRIV_END
  523. /* Loop over every file in the directory
  524. */
  525. while ((dirent = readdir(spool)) != NULL) {
  526. PRIV_START
  527. if (stat(dirent->d_name, &buf) != 0)
  528. perr("Cannot stat in " ATJOB_DIR);
  529. PRIV_END
  530. if (sscanf(dirent->d_name, "%c%5lx%8lx", &queue, &jobno, &ctm) != 3)
  531. continue;
  532. if (atoi(argv[i]) == jobno) {
  533. if ((buf.st_uid != real_uid) && !(real_uid == 0)) {
  534. fprintf(stderr, "%s: Not owner\n", argv[i]);
  535. exit(EXIT_FAILURE);
  536. }
  537. switch (what) {
  538. case ATRM:
  539. /*
  540. We need the unprivileged uid here since the file is owned by the real
  541. (not effective) uid.
  542. */
  543. PRIV_START
  544. if (queue == '=') {
  545. fprintf(stderr, "Warning: deleting running job\n");
  546. }
  547. if (unlink(dirent->d_name) != 0) {
  548. perr("Cannot unlink %.500s", dirent->d_name);
  549. rc = EXIT_FAILURE;
  550. }
  551. PRIV_END
  552. done = 1;
  553. break;
  554. case CAT:
  555. {
  556. FILE *fp;
  557. int ch;
  558. PRIV_START
  559. fp = fopen(dirent->d_name, "r");
  560. if (fp) {
  561. while ((ch = getc(fp)) != EOF) {
  562. putchar(ch);
  563. }
  564. done = 1;
  565. }
  566. else {
  567. perr("Cannot open %.500s", dirent->d_name);
  568. rc = EXIT_FAILURE;
  569. }
  570. PRIV_END
  571. }
  572. break;
  573. default:
  574. fprintf(stderr,
  575. "Internal error, process_jobs = %d\n", what);
  576. exit(EXIT_FAILURE);
  577. break;
  578. }
  579. }
  580. }
  581. closedir(spool);
  582. if (done != 1) {
  583. fprintf(stderr, "Cannot find jobid %s\n", argv[i] );
  584. rc = EXIT_FAILURE;
  585. }
  586. }
  587. return rc;
  588. } /* delete_jobs */
  589. /* Global functions */
  590. void *
  591. mymalloc(size_t n)
  592. {
  593. void *p;
  594. if ((p = malloc(n)) == (void *) 0) {
  595. fprintf(stderr, "Virtual memory exhausted\n");
  596. exit(EXIT_FAILURE);
  597. }
  598. return p;
  599. }
  600. int
  601. main(int argc, char **argv)
  602. {
  603. int c;
  604. char queue = DEFAULT_QUEUE;
  605. char queue_set = 0;
  606. char *pgm;
  607. int program = AT; /* our default program */
  608. char *options = "q:f:MmvlrdhVct:"; /* default options for at */
  609. int disp_version = 0;
  610. time_t timer = 0;
  611. struct passwd *pwe;
  612. struct group *ge;
  613. RELINQUISH_PRIVS
  614. if ((pwe = getpwnam(DAEMON_USERNAME)) == NULL)
  615. perr("Cannot get uid for " DAEMON_USERNAME);
  616. daemon_uid = pwe->pw_uid;
  617. if ((ge = getgrnam(DAEMON_GROUPNAME)) == NULL)
  618. perr("Cannot get gid for " DAEMON_GROUPNAME);
  619. daemon_gid = ge->gr_gid;
  620. /* Eat any leading paths
  621. */
  622. if ((pgm = strrchr(argv[0], '/')) == NULL)
  623. pgm = argv[0];
  624. else
  625. pgm++;
  626. namep = pgm;
  627. /* find out what this program is supposed to do
  628. */
  629. if (strcmp(pgm, "atq") == 0) {
  630. program = ATQ;
  631. options = "hq:V";
  632. } else if (strcmp(pgm, "atrm") == 0) {
  633. program = ATRM;
  634. options = "hV";
  635. }
  636. /* process whatever options we can process
  637. */
  638. opterr = 1;
  639. while ((c = getopt(argc, argv, options)) != EOF)
  640. switch (c) {
  641. case 'h':
  642. usage();
  643. exit (0);
  644. case 'v': /* verify time settings */
  645. atverify = 1;
  646. break;
  647. case 'm': /* send mail when job is complete */
  648. send_mail = 1;
  649. break;
  650. case 'M': /* don't send mail, even when job failed */
  651. send_mail = -1;
  652. break;
  653. case 'f':
  654. atinput = optarg;
  655. break;
  656. case 'q': /* specify queue */
  657. if (strlen(optarg) > 1)
  658. usage();
  659. atqueue = queue = *optarg;
  660. if (!(islower(queue) || isupper(queue)) & (queue != '='))
  661. usage();
  662. queue_set = 1;
  663. break;
  664. case 'r':
  665. case 'd':
  666. if (program != AT)
  667. usage();
  668. program = ATRM;
  669. options = "V";
  670. break;
  671. case 'l':
  672. if (program != AT)
  673. usage();
  674. program = ATQ;
  675. options = "q:V";
  676. break;
  677. case 'b':
  678. if (program != AT)
  679. usage();
  680. program = BATCH;
  681. options = "";
  682. break;
  683. case 'V':
  684. disp_version = 1;
  685. break;
  686. case 'c':
  687. program = CAT;
  688. options = "";
  689. break;
  690. case 't':
  691. if (!posixtime(&timer, optarg, PDS_LEADING_YEAR | PDS_CENTURY | PDS_SECONDS)) {
  692. fprintf(stderr, "invalid date format: %s\n", optarg);
  693. exit(EXIT_FAILURE);
  694. }
  695. /* drop seconds */
  696. timer -= timer % 60;
  697. break;
  698. default:
  699. usage();
  700. break;
  701. }
  702. /* end of options eating
  703. */
  704. if (disp_version) {
  705. fprintf(stderr, "at version " VERSION "\n");
  706. if (argc == 2)
  707. exit(EXIT_SUCCESS);
  708. }
  709. /* select our program
  710. */
  711. if (!check_permission()) {
  712. fprintf(stderr, "You do not have permission to use %.100s.\n", namep);
  713. exit(EXIT_FAILURE);
  714. }
  715. switch (program) {
  716. int i;
  717. case ATQ:
  718. REDUCE_PRIV(daemon_uid, daemon_gid)
  719. list_jobs();
  720. break;
  721. case ATRM:
  722. REDUCE_PRIV(daemon_uid, daemon_gid)
  723. if (argc > optind) {
  724. for (i = optind; i < argc ; i++ )
  725. if (strspn(argv[i],"0123456789") != strlen(argv[i])) {
  726. fprintf(stderr,"at: unknown jobid: %s\n", argv[i]);
  727. exit(EXIT_FAILURE);
  728. }
  729. return process_jobs(argc, argv, ATRM);
  730. }
  731. else
  732. usage();
  733. break;
  734. case CAT:
  735. if (argc > optind) {
  736. for (i = optind; i < argc ; i++ )
  737. if (strspn(argv[i],"0123456789") != strlen(argv[i])) {
  738. fprintf(stderr,"at: unknown jobid: %s", argv[i]);
  739. exit(EXIT_FAILURE);
  740. }
  741. return process_jobs(argc, argv, CAT);
  742. }
  743. else
  744. usage();
  745. break;
  746. case AT:
  747. if (argc > optind) {
  748. if (timer != 0) {
  749. fprintf(stderr, "Cannot give time twice.\n");
  750. exit(EXIT_FAILURE);
  751. }
  752. timer = parsetime(time(0), argc - optind, argv + optind);
  753. }
  754. if (timer == 0) {
  755. fprintf(stderr, "Garbled time\n");
  756. exit(EXIT_FAILURE);
  757. }
  758. if (atverify) {
  759. struct tm *tm = localtime(&timer);
  760. fprintf(stderr, "%s\n", asctime(tm));
  761. }
  762. /* POSIX.2 allows the shell specified by the user's SHELL environment
  763. variable, the login shell from the user's password database entry,
  764. or /bin/sh to be the command interpreter that processes the at-job.
  765. It also alows a warning diagnostic to be printed. Because of the
  766. possible variance, we always output the diagnostic. */
  767. //fprintf(stderr, "warning: commands will be executed using /bin/sh\n");
  768. writefile(timer, queue);
  769. break;
  770. case BATCH:
  771. if (queue_set)
  772. queue = toupper(queue);
  773. else
  774. queue = BATCH_QUEUE;
  775. if (argc > optind) {
  776. if (timer != 0) {
  777. fprintf(stderr, "Cannot give time twice.\n");
  778. exit(EXIT_FAILURE);
  779. }
  780. timer = parsetime(time(0), argc, argv);
  781. } else if (timer == 0)
  782. timer = time(NULL);
  783. if (atverify) {
  784. struct tm *tm = localtime(&timer);
  785. fprintf(stderr, "%s\n", asctime(tm));
  786. }
  787. writefile(timer, queue);
  788. break;
  789. default:
  790. panic("Internal error");
  791. break;
  792. }
  793. exit(EXIT_SUCCESS);
  794. }