summaryrefslogtreecommitdiff
path: root/mail.local/mail.local.c
diff options
context:
space:
mode:
Diffstat (limited to 'mail.local/mail.local.c')
-rw-r--r--mail.local/mail.local.c1304
1 files changed, 0 insertions, 1304 deletions
diff --git a/mail.local/mail.local.c b/mail.local/mail.local.c
deleted file mode 100644
index abe1a19..0000000
--- a/mail.local/mail.local.c
+++ /dev/null
@@ -1,1304 +0,0 @@
-/*
- * Copyright (c) 1998 Sendmail, Inc. All rights reserved.
- * Copyright (c) 1990, 1993, 1994
- * The Regents of the University of California. All rights reserved.
- *
- * By using this file, you agree to the terms and conditions set
- * forth in the LICENSE file which can be found at the top level of
- * the sendmail distribution.
- *
- */
-
-#ifndef lint
-static char copyright[] =
-"@(#) Copyright (c) 1990, 1993, 1994\n\
- The Regents of the University of California. All rights reserved.\n";
-#endif /* not lint */
-
-#ifndef lint
-static char sccsid[] = "@(#)mail.local.c 8.83 (Berkeley) 12/17/1998";
-#endif /* not lint */
-
-/*
- * This is not intended to work on System V derived systems
- * such as Solaris or HP-UX, since they use a totally different
- * approach to mailboxes (essentially, they have a setgid program
- * rather than setuid, and they rely on the ability to "give away"
- * files to do their work). IT IS NOT A BUG that this doesn't
- * work on such architectures.
- */
-
-#include <sys/param.h>
-#include <sys/stat.h>
-#include <sys/socket.h>
-#include <sys/file.h>
-
-#include <netinet/in.h>
-
-#include <errno.h>
-#include <fcntl.h>
-#include <netdb.h>
-#include <pwd.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <syslog.h>
-#include <time.h>
-#include <unistd.h>
-#ifdef EX_OK
-# undef EX_OK /* unistd.h may have another use for this */
-#endif
-#include <sysexits.h>
-#include <ctype.h>
-
-#ifdef __STDC__
-#include <stdarg.h>
-#else
-#include <varargs.h>
-#endif
-
-#if (defined(sun) && defined(__svr4__)) || defined(__SVR4)
-# define USE_LOCKF 1
-# define USE_SETEUID 1
-# define _PATH_MAILDIR "/var/mail"
-#endif
-
-#if (defined(sun) && !defined(__svr4__)) && !defined(__SVR4)
-# ifdef __dead
-# undef __dead
-# define __dead
-# endif
-#endif
-
-#if defined(_AIX)
-# define USE_LOCKF 1
-# define USE_SETEUID 1
-# define USE_VSYSLOG 0
-#endif
-
-#if defined(__hpux)
-# define USE_LOCKF 1
-# define USE_SETRESUID 1
-# define USE_VSYSLOG 0
-# ifdef __dead
-# undef __dead
-# define __dead
-# endif
-#endif
-
-#if defined(_CRAY)
-# if !defined(MAXPATHLEN)
-# define MAXPATHLEN PATHSIZE
-# endif
-# define USE_VSYSLOG 0
-# define _PATH_MAILDIR "/usr/spool/mail"
-#endif
-
-#if defined(ultrix)
-# define USE_VSYSLOG 0
-#endif
-
-#if defined(__osf__)
-# define USE_VSYSLOG 0
-#endif
-
-#if defined(NeXT) && !defined(__APPLE__)
-# include <libc.h>
-# define _PATH_MAILDIR "/usr/spool/mail"
-# define __dead /* empty */
-# define S_IRUSR S_IREAD
-# define S_IWUSR S_IWRITE
-#endif
-
-#if defined(IRIX64) || defined(IRIX5) || defined(IRIX6)
-# include <paths.h>
-# define HASSTRERROR 1 /* has strerror(3) */
-#endif
-
-/*
- * If you don't have flock, you could try using lockf instead.
- */
-
-#ifdef USE_LOCKF
-# define flock(a, b) lockf(a, b, 0)
-# define LOCK_EX F_LOCK
-#endif
-
-#ifndef USE_VSYSLOG
-# define USE_VSYSLOG 1
-#endif
-
-#ifndef LOCK_EX
-# include <sys/file.h>
-#endif
-
-#if defined(BSD4_4) || defined(__GLIBC__)
-# include "pathnames.h"
-#endif
-
-#ifndef __P
-# ifdef __STDC__
-# define __P(protos) protos
-# else
-# define __P(protos) ()
-# define const
-# endif
-#endif
-#ifndef __dead
-# if defined(__GNUC__) && (__GNUC__ < 2 || __GNUC_MINOR__ < 5) && !defined(__STRICT_ANSI__)
-# define __dead __volatile
-# else
-# define __dead
-# endif
-#endif
-
-#ifdef BSD4_4
-# define HAS_ST_GEN 1
-#else
-# ifndef _BSD_VA_LIST_
-# define _BSD_VA_LIST_ va_list
-# endif
-#endif
-
-#if defined(BSD4_4) || defined(linux)
-# define HASSNPRINTF 1
-#else
-# ifndef ultrix
-extern FILE *fdopen __P((int, const char *));
-# endif
-#endif
-
-#if SOLARIS >= 20600 || (SOLARIS < 10000 && SOLARIS >= 206)
-# define HASSNPRINTF 1 /* has snprintf starting in 2.6 */
-#endif
-
-#if !HASSNPRINTF
-extern int snprintf __P((char *, size_t, const char *, ...));
-# ifndef _CRAY
-extern int vsnprintf __P((char *, size_t, const char *, ...));
-# endif
-#endif
-
-#if defined(BSD4_4) || defined(__osf__) || defined(__GNU_LIBRARY__)
-# ifndef HASSTRERROR
-# define HASSTRERROR 1
-# endif
-#endif
-
-#if !HASSTRERROR
-extern char *strerror __P((int));
-#endif
-
-/*
- * If you don't have setreuid, and you have saved uids, and you have
- * a seteuid() call that doesn't try to emulate using setuid(), then
- * you can try defining USE_SETEUID.
- */
-#ifdef USE_SETEUID
-# define setreuid(r, e) seteuid(e)
-#endif
-
-/*
- * And of course on hpux you have setresuid()
- */
-#ifdef USE_SETRESUID
-# define setreuid(r, e) setresuid(-1, e, -1)
-#endif
-
-#ifndef _PATH_LOCTMP
-# define _PATH_LOCTMP "/tmp/local.XXXXXX"
-#endif
-#ifndef _PATH_MAILDIR
-# define _PATH_MAILDIR "/var/spool/mail"
-#endif
-
-#ifndef S_ISREG
-# define S_ISREG(mode) (((mode) & _S_IFMT) == S_IFREG)
-#endif
-
-#ifndef MAILER_DAEMON
-# define MAILER_DAEMON "MAILER-DAEMON"
-#endif
-
-int eval = EX_OK; /* sysexits.h error value. */
-int lmtpmode = 0;
-u_char tTdvect[100];
-
-void deliver __P((int, char *));
-void e_to_sys __P((int));
-void notifybiff __P((char *));
-int store __P((char *, int));
-void usage __P((void));
-void vwarn __P((const char *, _BSD_VA_LIST_));
-void lockmbox __P((char *));
-void unlockmbox __P((void));
-void mailerr __P((const char *, const char *, ...));
-
-int
-main(argc, argv)
- int argc;
- char *argv[];
-{
- struct passwd *pw;
- int ch, fd;
- uid_t uid;
- char *from;
- extern char *optarg;
- extern int optind;
- extern void dolmtp __P((void));
-
- /* make sure we have some open file descriptors */
- for (fd = 10; fd < 30; fd++)
- (void) close(fd);
-
- /* use a reasonable umask */
- (void) umask(0077);
-
-#ifdef LOG_MAIL
- openlog("mail.local", 0, LOG_MAIL);
-#else
- openlog("mail.local", 0);
-#endif
-
- from = NULL;
- while ((ch = getopt(argc, argv, "df:r:l")) != EOF)
- switch(ch) {
- case 'd': /* Backward compatible. */
- break;
- case 'f':
- case 'r': /* Backward compatible. */
- if (from != NULL) {
- mailerr(NULL, "multiple -f options");
- usage();
- }
- from = optarg;
- break;
- case 'l':
- lmtpmode++;
- break;
- case '?':
- default:
- usage();
- }
- argc -= optind;
- argv += optind;
-
- if (lmtpmode)
- dolmtp();
-
- if (!*argv)
- usage();
-
- /*
- * If from not specified, use the name from getlogin() if the
- * uid matches, otherwise, use the name from the password file
- * corresponding to the uid.
- */
- uid = getuid();
- if (!from && (!(from = getlogin()) ||
- !(pw = getpwnam(from)) || pw->pw_uid != uid))
- from = (pw = getpwuid(uid)) ? pw->pw_name : "???";
-
- /*
- * There is no way to distinguish the error status of one delivery
- * from the rest of the deliveries. So, if we failed hard on one
- * or more deliveries, but had no failures on any of the others, we
- * return a hard failure. If we failed temporarily on one or more
- * deliveries, we return a temporary failure regardless of the other
- * failures. This results in the delivery being reattempted later
- * at the expense of repeated failures and multiple deliveries.
- */
- for (fd = store(from, 0); *argv; ++argv)
- deliver(fd, *argv);
- exit(eval);
-}
-
-char *
-parseaddr(s)
- char *s;
-{
- char *p;
- int len;
-
- if (*s++ != '<')
- return NULL;
-
- p = s;
-
- /* at-domain-list */
- while (*p == '@') {
- p++;
- if (*p == '[') {
- p++;
- while (isascii(*p) &&
- (isalnum(*p) || *p == '.' ||
- *p == '-' || *p == ':'))
- p++;
- if (*p++ != ']')
- return NULL;
- } else {
- while ((isascii(*p) && isalnum(*p)) ||
- strchr(".-_", *p))
- p++;
- }
- if (*p == ',' && p[1] == '@')
- p++;
- else if (*p == ':' && p[1] != '@')
- p++;
- else
- return NULL;
- }
-
- s = p;
-
- /* local-part */
- if (*p == '\"') {
- p++;
- while (*p && *p != '\"') {
- if (*p == '\\') {
- if (!*++p)
- return NULL;
- }
- p++;
- }
- if (!*p++)
- return NULL;
- } else {
- while (*p && *p != '@' && *p != '>') {
- if (*p == '\\') {
- if (!*++p)
- return NULL;
- } else {
- if (*p <= ' ' || (*p & 128) ||
- strchr("<>()[]\\,;:\"", *p))
- return NULL;
- }
- p++;
- }
- }
-
- /* @domain */
- if (*p == '@') {
- p++;
- if (*p == '[') {
- p++;
- while (isascii(*p) &&
- (isalnum(*p) || *p == '.' ||
- *p == '-' || *p == ':'))
- p++;
- if (*p++ != ']')
- return NULL;
- } else {
- while ((isascii(*p) && isalnum(*p)) ||
- strchr(".-_", *p))
- p++;
- }
- }
-
- if (*p++ != '>')
- return NULL;
- if (*p && *p != ' ')
- return NULL;
- len = p - s - 1;
- if (*s == '\0' || len <= 0)
- {
- s = MAILER_DAEMON;
- len = strlen(s);
- }
-
- p = malloc(len + 1);
- if (p == NULL) {
- printf("421 4.3.0 memory exhausted\r\n");
- exit(EX_TEMPFAIL);
- }
-
- strncpy(p, s, len);
- p[len] = '\0';
- return p;
-}
-
-char *
-process_recipient(addr)
- char *addr;
-{
- if (getpwnam(addr) == NULL) {
- return "550 5.1.1 user unknown";
- }
-
- return NULL;
-}
-
-
-#define RCPT_GROW 30
-
-void
-dolmtp()
-{
- char *return_path = NULL;
- char **rcpt_addr = NULL;
- int rcpt_num = 0;
- int rcpt_alloc = 0;
- char myhostname[1024];
- char buf[4096];
- char *err;
- int msgfd;
- char *p;
- int i;
-
- gethostname(myhostname, sizeof myhostname - 1);
-
- printf("220 %s LMTP ready\r\n", myhostname);
- for (;;) {
- fflush(stdout);
- if (fgets(buf, sizeof(buf)-1, stdin) == NULL) {
- exit(EX_OK);
- }
- p = buf + strlen(buf) - 1;
- if (p >= buf && *p == '\n')
- *p-- = '\0';
- if (p >= buf && *p == '\r')
- *p-- = '\0';
-
- switch (buf[0]) {
-
- case 'd':
- case 'D':
- if (strcasecmp(buf, "data") == 0) {
- if (rcpt_num == 0) {
- printf("503 5.5.1 No recipients\r\n");
- continue;
- }
- msgfd = store(return_path, rcpt_num);
- if (msgfd == -1)
- continue;
-
- for (i = 0; i < rcpt_num; i++) {
- p = strchr(rcpt_addr[i], '+');
- if (p != NULL)
- *p++ = '\0';
- deliver(msgfd, rcpt_addr[i]);
- }
- close(msgfd);
- goto rset;
- }
- goto syntaxerr;
-
- case 'l':
- case 'L':
- if (strncasecmp(buf, "lhlo ", 5) == 0) {
- printf("250-%s\r\n250-8BITMIME\r\n250-ENHANCEDSTATUSCODES\r\n250 PIPELINING\r\n",
- myhostname);
- continue;
- }
- goto syntaxerr;
-
- case 'm':
- case 'M':
- if (strncasecmp(buf, "mail ", 5) == 0) {
- if (return_path != NULL) {
- printf("503 5.5.1 Nested MAIL command\r\n");
- continue;
- }
- if (strncasecmp(buf+5, "from:", 5) != 0 ||
- ((return_path = parseaddr(buf+10)) == NULL)) {
- printf("501 5.5.4 Syntax error in parameters\r\n");
- continue;
- }
- printf("250 2.5.0 ok\r\n");
- continue;
- }
- goto syntaxerr;
-
- case 'n':
- case 'N':
- if (strcasecmp(buf, "noop") == 0) {
- printf("250 2.0.0 ok\r\n");
- continue;
- }
- goto syntaxerr;
-
- case 'q':
- case 'Q':
- if (strcasecmp(buf, "quit") == 0) {
- printf("221 2.0.0 bye\r\n");
- exit(EX_OK);
- }
- goto syntaxerr;
-
- case 'r':
- case 'R':
- if (strncasecmp(buf, "rcpt ", 5) == 0) {
- if (return_path == NULL) {
- printf("503 5.5.1 Need MAIL command\r\n");
- continue;
- }
- if (rcpt_num >= rcpt_alloc) {
- rcpt_alloc += RCPT_GROW;
- rcpt_addr = (char **)
- realloc((char *)rcpt_addr,
- rcpt_alloc * sizeof(char **));
- if (rcpt_addr == NULL) {
- printf("421 4.3.0 memory exhausted\r\n");
- exit(EX_TEMPFAIL);
- }
- }
- if (strncasecmp(buf+5, "to:", 3) != 0 ||
- ((rcpt_addr[rcpt_num] = parseaddr(buf+8)) == NULL)) {
- printf("501 5.5.4 Syntax error in parameters\r\n");
- continue;
- }
- if ((err = process_recipient(rcpt_addr[rcpt_num])) != NULL) {
- printf("%s\r\n", err);
- continue;
- }
- rcpt_num++;
- printf("250 2.1.5 ok\r\n");
- continue;
- }
- else if (strcasecmp(buf, "rset") == 0) {
- printf("250 2.0.0 ok\r\n");
-
- rset:
- while (rcpt_num) {
- free(rcpt_addr[--rcpt_num]);
- }
- if (return_path != NULL)
- free(return_path);
- return_path = NULL;
- continue;
- }
- goto syntaxerr;
-
- case 'v':
- case 'V':
- if (strncasecmp(buf, "vrfy ", 5) == 0) {
- printf("252 2.3.3 try RCPT to attempt delivery\r\n");
- continue;
- }
- goto syntaxerr;
-
- default:
- syntaxerr:
- printf("500 5.5.2 Syntax error\r\n");
- continue;
- }
- }
-}
-
-int
-store(from, lmtprcpts)
- char *from;
- int lmtprcpts;
-{
- FILE *fp = NULL;
- time_t tval;
- int fd, eline;
- char line[2048];
- char tmpbuf[sizeof _PATH_LOCTMP + 1];
-
- strcpy(tmpbuf, _PATH_LOCTMP);
- if ((fd = mkstemp(tmpbuf)) == -1 || (fp = fdopen(fd, "w+")) == NULL) {
- if (lmtprcpts) {
- printf("451 4.3.0 unable to open temporary file\r\n");
- return -1;
- } else {
- mailerr("451 4.3.0", "unable to open temporary file");
- exit(eval);
- }
- }
- (void)unlink(tmpbuf);
-
- if (lmtpmode) {
- printf("354 go ahead\r\n");
- fflush(stdout);
- }
-
- (void)time(&tval);
- (void)fprintf(fp, "From %s %s", from, ctime(&tval));
-
- line[0] = '\0';
- for (eline = 1; fgets(line, sizeof(line), stdin);) {
- size_t line_len = strlen(line);
-
- if (line_len >= 2 &&
- line[line_len - 2] == '\r' &&
- line[line_len - 1] == '\n') {
- strcpy(line + line_len - 2, "\n");
- }
- if (lmtprcpts && line[0] == '.') {
- char *src = line + 1, *dest = line;
-
- if (line[1] == '\n')
- goto lmtpdot;
- while (*src != '\0')
- *dest++ = *src++;
- *dest = '\0';
- }
- if (line[0] == '\n')
- eline = 1;
- else {
- if (eline && line[0] == 'F' &&
- !memcmp(line, "From ", 5))
- (void)putc('>', fp);
- eline = 0;
- }
- (void)fprintf(fp, "%s", line);
- if (ferror(fp)) {
- if (lmtprcpts) {
- while (lmtprcpts--) {
- printf("451 4.3.0 temporary file write error\r\n");
- }
- fclose(fp);
- return -1;
- } else {
- mailerr("451 4.3.0",
- "temporary file write error");
- fclose(fp);
- exit(eval);
- }
- }
- }
-
- if (lmtprcpts) {
- /* Got a premature EOF -- toss message and exit */
- exit(EX_OK);
- }
-
- /* If message not newline terminated, need an extra. */
- if (strchr(line, '\n') == NULL)
- (void)putc('\n', fp);
-
- lmtpdot:
-
- /* Output a newline; note, empty messages are allowed. */
- (void)putc('\n', fp);
-
- if (fflush(fp) == EOF || ferror(fp)) {
- if (lmtprcpts) {
- while (lmtprcpts--) {
- printf("451 4.3.0 temporary file write error\r\n");
- }
- fclose(fp);
- return -1;
- } else {
- mailerr("451 4.3.0", "temporary file write error");
- fclose(fp);
- exit(eval);
- }
- }
- return (fd);
-}
-
-void
-deliver(fd, name)
- int fd;
- char *name;
-{
- struct stat fsb, sb;
- struct passwd *pw;
- int mbfd, nr, nw, off;
- char *p;
- char biffmsg[100], buf[8*1024], path[MAXPATHLEN];
- off_t curoff;
- extern char *quad_to_string();
-
- /*
- * Disallow delivery to unknown names -- special mailboxes can be
- * handled in the sendmail aliases file.
- */
- if ((pw = getpwnam(name)) == NULL) {
- if (eval != EX_TEMPFAIL)
- eval = EX_UNAVAILABLE;
- if (lmtpmode) {
- if (eval == EX_TEMPFAIL) {
- printf("451 4.3.0 cannot lookup name: %s\r\n", name);
- } else {
- printf("550 5.1.1 unknown name: %s\r\n", name);
- }
- }
- else {
- char *errcode = NULL;
-
- if (eval == EX_TEMPFAIL)
- errcode = "451 4.3.0";
- else
- errcode = "550 5.1.1";
- mailerr(errcode, "unknown name: %s", name);
- }
- return;
- }
- endpwent();
-
- /*
- * Keep name reasonably short to avoid buffer overruns.
- * This isn't necessary on BSD because of the proper
- * definition of snprintf(), but it can cause problems
- * on other systems.
- * Also, clear out any bogus characters.
- */
-
- if (strlen(name) > 40)
- name[40] = '\0';
- for (p = name; *p != '\0'; p++)
- {
- if (!isascii(*p))
- *p &= 0x7f;
- else if (!isprint(*p))
- *p = '.';
- }
-
- (void)snprintf(path, sizeof(path), "%s/%s", _PATH_MAILDIR, name);
-
- /*
- * If the mailbox is linked or a symlink, fail. There's an obvious
- * race here, that the file was replaced with a symbolic link after
- * the lstat returned, but before the open. We attempt to detect
- * this by comparing the original stat information and information
- * returned by an fstat of the file descriptor returned by the open.
- *
- * NB: this is a symptom of a larger problem, that the mail spooling
- * directory is writeable by the wrong users. If that directory is
- * writeable, system security is compromised for other reasons, and
- * it cannot be fixed here.
- *
- * If we created the mailbox, set the owner/group. If that fails,
- * just return. Another process may have already opened it, so we
- * can't unlink it. Historically, binmail set the owner/group at
- * each mail delivery. We no longer do this, assuming that if the
- * ownership or permissions were changed there was a reason.
- *
- * XXX
- * open(2) should support flock'ing the file.
- */
-tryagain:
- lockmbox(path);
- if (lstat(path, &sb) < 0) {
- mbfd = open(path,
- O_APPEND|O_CREAT|O_EXCL|O_WRONLY, S_IRUSR|S_IWUSR);
- if (lstat(path, &sb) < 0)
- {
- eval = EX_CANTCREAT;
- mailerr("550 5.2.0",
- "%s: lstat: file changed after open", path);
- goto err1;
- }
- else
- sb.st_uid = pw->pw_uid;
- if (mbfd == -1) {
- if (errno == EEXIST)
- goto tryagain;
- } else if (fchown(mbfd, pw->pw_uid, pw->pw_gid)) {
- mailerr("451 4.3.0", "chown %u.%u: %s",
- pw->pw_uid, pw->pw_gid, name);
- goto err1;
- }
- } else if (sb.st_nlink != 1 || !S_ISREG(sb.st_mode)) {
- mailerr("550 5.2.0", "%s: irregular file", path);
- goto err0;
- } else if (sb.st_uid != pw->pw_uid) {
- eval = EX_CANTCREAT;
- mailerr("550 5.2.0", "%s: wrong ownership (%d)",
- path, sb.st_uid);
- goto err0;
- } else {
- mbfd = open(path, O_APPEND|O_WRONLY, 0);
- }
-
- if (mbfd == -1) {
- mailerr("450 4.2.0", "%s: %s", path, strerror(errno));
- goto err0;
- } else if (fstat(mbfd, &fsb) < 0 ||
- fsb.st_nlink != 1 ||
- sb.st_nlink != 1 ||
- !S_ISREG(fsb.st_mode) ||
- sb.st_dev != fsb.st_dev ||
- sb.st_ino != fsb.st_ino ||
-#if HAS_ST_GEN && 0 /* AFS returns random values for st_gen */
- sb.st_gen != fsb.st_gen ||
-#endif
- sb.st_uid != fsb.st_uid) {
- eval = EX_TEMPFAIL;
- mailerr("550 5.2.0", "%s: fstat: file changed after open",
- path);
- goto err1;
- }
-
- /* Wait until we can get a lock on the file. */
- if (flock(mbfd, LOCK_EX)) {
- mailerr("450 4.2.0", "%s: %s", path, strerror(errno));
- goto err1;
- }
-
- /* Get the starting offset of the new message for biff. */
- curoff = lseek(mbfd, (off_t)0, SEEK_END);
- if (sizeof curoff > sizeof(long))
- (void)snprintf(biffmsg, sizeof(biffmsg), "%s@%s\n",
- name, quad_to_string(curoff));
- else
- (void)snprintf(biffmsg, sizeof(biffmsg), "%s@%ld\n",
- name, curoff);
-
- /* Copy the message into the file. */
- if (lseek(fd, (off_t)0, SEEK_SET) == (off_t)-1) {
- mailerr("450 4.2.0", "temporary file: %s",
- strerror(errno));
- goto err1;
- }
- if (setreuid(0, pw->pw_uid) < 0) {
- mailerr("450 4.2.0", "setreuid(0, %d): %s (r=%d, e=%d)",
- pw->pw_uid, strerror(errno), getuid(), geteuid());
- goto err1;
- }
-#ifdef DEBUG
- printf("new euid = %d\n", geteuid());
-#endif
- while ((nr = read(fd, buf, sizeof(buf))) > 0)
- for (off = 0; off < nr; off += nw)
- if ((nw = write(mbfd, buf + off, nr - off)) < 0) {
- mailerr("450 4.2.0", "%s: %s",
- path, strerror(errno));
- goto err3;
- }
- if (nr < 0) {
- mailerr("450 4.2.0", "temporary file: %s",
- strerror(errno));
- goto err3;
- }
-
- /* Flush to disk, don't wait for update. */
- if (fsync(mbfd)) {
- mailerr("450 4.2.0", "%s: %s", path, strerror(errno));
-err3:
- if (setreuid(0, 0) < 0) {
-#if 0
- /* already printed an error above for this recipient */
- e_to_sys(errno);
- mailerr("450 4.2.0", "setreuid(0, 0): %s",
- strerror(errno));
-#endif
- }
-#ifdef DEBUG
- printf("reset euid = %d\n", geteuid());
-#endif
- (void)ftruncate(mbfd, curoff);
-err1: (void)close(mbfd);
-err0: unlockmbox();
- return;
- }
-
- /* Close and check -- NFS doesn't write until the close. */
- if (close(mbfd)) {
- mailerr("450 4.2.0", "%s: %s", path, strerror(errno));
- truncate(path, curoff);
- } else
- notifybiff(biffmsg);
-
- if (setreuid(0, 0) < 0) {
- mailerr("450 4.2.0", "setreuid(0, 0): %s",
- strerror(errno));
- goto err0;
- }
-#ifdef DEBUG
- printf("reset euid = %d\n", geteuid());
-#endif
- unlockmbox();
- if (lmtpmode) {
- printf("250 2.1.5 %s OK\r\n", name);
- }
-}
-
-/*
- * user.lock files are necessary for compatibility with other
- * systems, e.g., when the mail spool file is NFS exported.
- * Alas, mailbox locking is more than just a local matter.
- * EPA 11/94.
- */
-
-char lockname[MAXPATHLEN];
-int locked = 0;
-
-void
-lockmbox(path)
- char *path;
-{
- int statfailed = 0;
-
- if (locked)
- return;
- if (strlen(path) + 6 > sizeof lockname)
- return;
- snprintf(lockname, sizeof lockname, "%s.lock", path);
- for (;; sleep(5)) {
- int fd;
- struct stat st;
- time_t now;
-
- fd = open(lockname, O_WRONLY|O_EXCL|O_CREAT, 0);
- if (fd >= 0) {
- /* defeat lock checking programs which test pid */
- write(fd, "0", 2);
- locked = 1;
- close(fd);
- return;
- }
- if (stat(lockname, &st) < 0) {
- if (statfailed++ > 5)
- return;
- continue;
- }
- statfailed = 0;
- time(&now);
- if (now < st.st_ctime + 300)
- continue;
- unlink(lockname);
- }
-}
-
-void
-unlockmbox()
-{
- if (!locked)
- return;
- unlink(lockname);
- locked = 0;
-}
-
-void
-notifybiff(msg)
- char *msg;
-{
- static struct sockaddr_in addr;
- static int f = -1;
- struct hostent *hp;
- struct servent *sp;
- int len;
-
- if (addr.sin_family == 0) {
- /* Be silent if biff service not available. */
- if ((sp = getservbyname("biff", "udp")) == NULL)
- return;
- if ((hp = gethostbyname("localhost")) == NULL) {
- return;
- }
- addr.sin_family = hp->h_addrtype;
- memcpy(&addr.sin_addr, hp->h_addr, hp->h_length);
- addr.sin_port = sp->s_port;
- }
- if (f < 0 && (f = socket(AF_INET, SOCK_DGRAM, 0)) == -1) {
- return;
- }
- len = strlen(msg) + 1;
- (void) sendto(f, msg, len, 0, (struct sockaddr *)&addr, sizeof(addr));
-}
-
-void
-usage()
-{
- eval = EX_USAGE;
- mailerr(NULL, "usage: mail.local [-l] [-f from] user ...");
- exit(eval);
-}
-
-void
-#ifdef __STDC__
-mailerr(const char *hdr, const char *fmt, ...)
-#else
-mailerr(hdr, fmt, va_alist)
- const char *hdr;
- const char *fmt;
- va_dcl
-#endif
-{
- va_list ap;
-
-#ifdef __STDC__
- va_start(ap, fmt);
-#else
- va_start(ap);
-#endif
- if (lmtpmode)
- {
- if (hdr != NULL)
- printf("%s ", hdr);
- vprintf(fmt, ap);
- printf("\r\n");
- }
- else
- {
- e_to_sys(errno);
- vwarn(fmt, ap);
- }
-}
-
-void
-vwarn(fmt, ap)
- const char *fmt;
- _BSD_VA_LIST_ ap;
-{
- /*
- * Log the message to stderr.
- *
- * Don't use LOG_PERROR as an openlog() flag to do this,
- * it's not portable enough.
- */
- if (eval != EX_USAGE)
- (void)fprintf(stderr, "mail.local: ");
- (void)vfprintf(stderr, fmt, ap);
- (void)fprintf(stderr, "\n");
-
-#if USE_VSYSLOG
- /* Log the message to syslog. */
- vsyslog(LOG_ERR, fmt, ap);
-#else
- {
- char fmtbuf[10240];
-
- (void) vsnprintf(fmtbuf, sizeof fmtbuf, fmt, ap);
- syslog(LOG_ERR, "%s", fmtbuf);
- }
-#endif
-}
-
-/*
- * e_to_sys --
- * Guess which errno's are temporary. Gag me.
- */
-void
-e_to_sys(num)
- int num;
-{
- /* Temporary failures override hard errors. */
- if (eval == EX_TEMPFAIL)
- return;
-
- switch(num) { /* Hopefully temporary errors. */
-#ifdef EAGAIN
- case EAGAIN: /* Resource temporarily unavailable */
-#endif
-#ifdef EDQUOT
- case EDQUOT: /* Disc quota exceeded */
-#endif
-#ifdef EBUSY
- case EBUSY: /* Device busy */
-#endif
-#ifdef EPROCLIM
- case EPROCLIM: /* Too many processes */
-#endif
-#ifdef EUSERS
- case EUSERS: /* Too many users */
-#endif
-#ifdef ECONNABORTED
- case ECONNABORTED: /* Software caused connection abort */
-#endif
-#ifdef ECONNREFUSED
- case ECONNREFUSED: /* Connection refused */
-#endif
-#ifdef ECONNRESET
- case ECONNRESET: /* Connection reset by peer */
-#endif
-#ifdef EDEADLK
- case EDEADLK: /* Resource deadlock avoided */
-#endif
-#ifdef EFBIG
- case EFBIG: /* File too large */
-#endif
-#ifdef EHOSTDOWN
- case EHOSTDOWN: /* Host is down */
-#endif
-#ifdef EHOSTUNREACH
- case EHOSTUNREACH: /* No route to host */
-#endif
-#ifdef EMFILE
- case EMFILE: /* Too many open files */
-#endif
-#ifdef ENETDOWN
- case ENETDOWN: /* Network is down */
-#endif
-#ifdef ENETRESET
- case ENETRESET: /* Network dropped connection on reset */
-#endif
-#ifdef ENETUNREACH
- case ENETUNREACH: /* Network is unreachable */
-#endif
-#ifdef ENFILE
- case ENFILE: /* Too many open files in system */
-#endif
-#ifdef ENOBUFS
- case ENOBUFS: /* No buffer space available */
-#endif
-#ifdef ENOMEM
- case ENOMEM: /* Cannot allocate memory */
-#endif
-#ifdef ENOSPC
- case ENOSPC: /* No space left on device */
-#endif
-#ifdef EROFS
- case EROFS: /* Read-only file system */
-#endif
-#ifdef ESTALE
- case ESTALE: /* Stale NFS file handle */
-#endif
-#ifdef ETIMEDOUT
- case ETIMEDOUT: /* Connection timed out */
-#endif
-#if defined(EWOULDBLOCK) && EWOULDBLOCK != EAGAIN && EWOULDBLOCK != EDEADLK
- case EWOULDBLOCK: /* Operation would block. */
-#endif
- eval = EX_TEMPFAIL;
- break;
- default:
- eval = EX_UNAVAILABLE;
- break;
- }
-}
-
-#if !HASSTRERROR
-
-char *
-strerror(eno)
- int eno;
-{
- extern int sys_nerr;
- extern char *sys_errlist[];
- static char ebuf[60];
-
- if (eno >= 0 && eno < sys_nerr)
- return sys_errlist[eno];
- (void) sprintf(ebuf, "Error %d", eno);
- return ebuf;
-}
-
-#endif /* !HASSTRERROR */
-
-#if defined(ultrix) || defined(_CRAY)
-
-/*
- * Copyright (c) 1987, 1993
- * The Regents of the University of California. All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed by the University of
- * California, Berkeley and its contributors.
- * 4. Neither the name of the University nor the names of its contributors
- * may be used to endorse or promote products derived from this software
- * without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
- * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- */
-
-#if defined(LIBC_SCCS) && !defined(lint)
-static char sccsid[] = "@(#)mktemp.c 8.1 (Berkeley) 6/4/93";
-#endif /* LIBC_SCCS and not lint */
-
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <errno.h>
-#include <stdio.h>
-#include <ctype.h>
-
-static int _gettemp();
-
-mkstemp(path)
- char *path;
-{
- int fd;
-
- return (_gettemp(path, &fd) ? fd : -1);
-}
-
-/*
-char *
-mktemp(path)
- char *path;
-{
- return(_gettemp(path, (int *)NULL) ? path : (char *)NULL);
-}
-*/
-
-static
-_gettemp(path, doopen)
- char *path;
- register int *doopen;
-{
- extern int errno;
- register char *start, *trv;
- struct stat sbuf;
- u_int pid;
-
- pid = getpid();
- for (trv = path; *trv; ++trv); /* extra X's get set to 0's */
- while (*--trv == 'X') {
- *trv = (pid % 10) + '0';
- pid /= 10;
- }
-
- /*
- * check the target directory; if you have six X's and it
- * doesn't exist this runs for a *very* long time.
- */
- for (start = trv + 1;; --trv) {
- if (trv <= path)
- break;
- if (*trv == '/') {
- *trv = '\0';
- if (stat(path, &sbuf) < 0)
- return(0);
- if (!S_ISDIR(sbuf.st_mode)) {
- errno = ENOTDIR;
- return(0);
- }
- *trv = '/';
- break;
- }
- }
-
- for (;;) {
- if (doopen) {
- if ((*doopen =
- open(path, O_CREAT|O_EXCL|O_RDWR, 0600)) >= 0)
- return(1);
- if (errno != EEXIST)
- return(0);
- }
- else if (stat(path, &sbuf) < 0)
- return(errno == ENOENT ? 1 : 0);
-
- /* tricky little algorithm for backward compatibility */
- for (trv = start;;) {
- if (!*trv)
- return(0);
- if (*trv == 'z')
- *trv++ = 'a';
- else {
- if (isascii(*trv) && isdigit(*trv))
- *trv = 'a';
- else
- ++*trv;
- break;
- }
- }
- }
- /*NOTREACHED*/
-}
-
-#endif /* ultrix */