1995-12-02 17:30:23 +00:00
|
|
|
|
/*
|
1998-08-03 05:56:20 +00:00
|
|
|
|
* Copyright (c) 1998 Sendmail, Inc. All rights reserved.
|
|
|
|
|
* Copyright (c) 1992 Eric P. Allman. All rights reserved.
|
1995-12-02 17:30:23 +00:00
|
|
|
|
* Copyright (c) 1992, 1993
|
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
|
*
|
1998-08-03 05:56:20 +00:00
|
|
|
|
* 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.
|
1995-12-02 17:30:23 +00:00
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifndef lint
|
1999-01-12 12:38:06 +00:00
|
|
|
|
static char sccsid[] = "@(#)makemap.c 8.71 (Berkeley) 11/29/1998";
|
1995-12-02 17:30:23 +00:00
|
|
|
|
#endif /* not lint */
|
|
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <sys/errno.h>
|
|
|
|
|
#ifndef ISC_UNIX
|
|
|
|
|
# include <sys/file.h>
|
|
|
|
|
#endif
|
1997-06-27 14:53:01 +00:00
|
|
|
|
#include "sendmail.h"
|
1998-08-03 05:56:20 +00:00
|
|
|
|
#include "pathnames.h"
|
1995-12-02 17:30:23 +00:00
|
|
|
|
|
|
|
|
|
#ifdef NDBM
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# include <ndbm.h>
|
1995-12-02 17:30:23 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef NEWDB
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# include <db.h>
|
|
|
|
|
# ifndef DB_VERSION_MAJOR
|
|
|
|
|
# define DB_VERSION_MAJOR 1
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
enum type { T_DBM, T_BTREE, T_HASH, T_ERR, T_UNKNOWN };
|
|
|
|
|
|
|
|
|
|
union dbent
|
|
|
|
|
{
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
datum dbm;
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
DBT db;
|
|
|
|
|
#endif
|
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
char *data;
|
|
|
|
|
size_t size;
|
|
|
|
|
} xx;
|
|
|
|
|
};
|
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
uid_t RealUid;
|
|
|
|
|
gid_t RealGid;
|
|
|
|
|
char *RealUserName;
|
|
|
|
|
uid_t RunAsUid;
|
|
|
|
|
uid_t RunAsGid;
|
|
|
|
|
char *RunAsUserName;
|
|
|
|
|
int Verbose = 2;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
bool DontInitGroups = FALSE;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
long DontBlameSendmail = DBS_SAFE;
|
1997-06-27 14:53:01 +00:00
|
|
|
|
u_char tTdvect[100];
|
1999-01-12 12:38:06 +00:00
|
|
|
|
uid_t TrustedUid = 0;
|
1997-06-27 14:53:01 +00:00
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
#define BUFSIZE 1024
|
|
|
|
|
|
1998-08-03 05:56:20 +00:00
|
|
|
|
int
|
1995-12-02 17:30:23 +00:00
|
|
|
|
main(argc, argv)
|
|
|
|
|
int argc;
|
|
|
|
|
char **argv;
|
|
|
|
|
{
|
|
|
|
|
char *progname;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
char *cfile;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
bool inclnull = FALSE;
|
|
|
|
|
bool notrunc = FALSE;
|
|
|
|
|
bool allowreplace = FALSE;
|
|
|
|
|
bool allowdups = FALSE;
|
|
|
|
|
bool verbose = FALSE;
|
|
|
|
|
bool foldcase = TRUE;
|
|
|
|
|
int exitstat;
|
|
|
|
|
int opt;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
char *typename = NULL;
|
|
|
|
|
char *mapname = NULL;
|
|
|
|
|
char *ext = NULL;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
int lineno;
|
|
|
|
|
int st;
|
|
|
|
|
int mode;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
int putflags = 0;
|
|
|
|
|
#ifdef NEWDB
|
1996-11-18 02:26:51 +00:00
|
|
|
|
long dbcachesize = 1024 * 1024;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
enum type type;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if !O_EXLOCK
|
1995-12-02 17:30:23 +00:00
|
|
|
|
int fd;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#endif
|
1998-08-03 05:56:20 +00:00
|
|
|
|
int sff = SFF_ROOTOK|SFF_REGONLY;
|
|
|
|
|
struct passwd *pw;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
union
|
|
|
|
|
{
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
DBM *dbm;
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
DB *db;
|
|
|
|
|
#endif
|
|
|
|
|
void *dbx;
|
|
|
|
|
} dbp;
|
|
|
|
|
union dbent key, val;
|
|
|
|
|
#ifdef NEWDB
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1995-12-02 17:30:23 +00:00
|
|
|
|
BTREEINFO bti;
|
1996-11-18 02:26:51 +00:00
|
|
|
|
HASHINFO hinfo;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
DB_INFO dbinfo;
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
#endif
|
|
|
|
|
char ibuf[BUFSIZE];
|
|
|
|
|
char fbuf[MAXNAME];
|
1997-06-27 14:53:01 +00:00
|
|
|
|
char dbuf[MAXNAME];
|
1998-08-03 05:56:20 +00:00
|
|
|
|
#ifdef NDBM
|
1997-06-27 14:53:01 +00:00
|
|
|
|
char pbuf[MAXNAME];
|
1998-08-03 05:56:20 +00:00
|
|
|
|
#endif
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if _FFR_TRUSTED_USER
|
1998-08-03 05:56:20 +00:00
|
|
|
|
FILE *cfp;
|
|
|
|
|
char buf[MAXLINE];
|
|
|
|
|
#endif
|
1997-06-27 14:53:01 +00:00
|
|
|
|
static char rnamebuf[MAXNAME]; /* holds RealUserName */
|
1998-08-03 05:56:20 +00:00
|
|
|
|
struct stat std;
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
struct stat stp;
|
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
extern char *optarg;
|
|
|
|
|
extern int optind;
|
|
|
|
|
|
|
|
|
|
progname = argv[0];
|
1998-08-03 05:56:20 +00:00
|
|
|
|
cfile = _PATH_SENDMAILCF;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
RunAsUid = RealUid = getuid();
|
|
|
|
|
RunAsGid = RealGid = getgid();
|
|
|
|
|
pw = getpwuid(RealUid);
|
1997-08-04 05:00:07 +00:00
|
|
|
|
if (pw != NULL)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
if (strlen(pw->pw_name) > MAXNAME - 1)
|
|
|
|
|
pw->pw_name[MAXNAME] = 0;
|
|
|
|
|
sprintf(rnamebuf, "%s", pw->pw_name);
|
|
|
|
|
}
|
|
|
|
|
else
|
1998-08-03 05:56:20 +00:00
|
|
|
|
sprintf(rnamebuf, "Unknown UID %d", (int) RealUid);
|
1997-06-27 14:53:01 +00:00
|
|
|
|
RunAsUserName = RealUserName = rnamebuf;
|
|
|
|
|
|
|
|
|
|
#if _FFR_NEW_MAKEMAP_FLAGS
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#define OPTIONS "C:Nc:dflorsv"
|
1996-11-18 02:26:51 +00:00
|
|
|
|
#else
|
1998-08-03 05:56:20 +00:00
|
|
|
|
#define OPTIONS "C:Ndforsv"
|
1996-11-18 02:26:51 +00:00
|
|
|
|
#endif
|
1998-08-04 12:53:30 +00:00
|
|
|
|
while ((opt = getopt(argc, argv, OPTIONS)) != -1)
|
1995-12-02 17:30:23 +00:00
|
|
|
|
{
|
|
|
|
|
switch (opt)
|
|
|
|
|
{
|
1998-08-03 05:56:20 +00:00
|
|
|
|
case 'C':
|
|
|
|
|
cfile = optarg;
|
|
|
|
|
break;
|
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
case 'N':
|
|
|
|
|
inclnull = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
#if _FFR_NEW_MAKEMAP_FLAGS
|
1996-11-18 02:26:51 +00:00
|
|
|
|
case 'c':
|
1999-01-12 12:38:06 +00:00
|
|
|
|
# ifdef NEWDB
|
1996-11-18 02:26:51 +00:00
|
|
|
|
dbcachesize = atol(optarg);
|
1999-01-12 12:38:06 +00:00
|
|
|
|
# endif
|
1996-11-18 02:26:51 +00:00
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
case 'd':
|
|
|
|
|
allowdups = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 'f':
|
|
|
|
|
foldcase = FALSE;
|
|
|
|
|
break;
|
|
|
|
|
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if _FFR_NEW_MAKEMAP_FLAGS
|
|
|
|
|
case 'l':
|
|
|
|
|
# ifdef NDBM
|
|
|
|
|
printf("dbm\n");
|
|
|
|
|
# endif
|
|
|
|
|
# ifdef NEWDB
|
|
|
|
|
printf("hash\n");
|
|
|
|
|
printf("btree\n");
|
|
|
|
|
# endif
|
|
|
|
|
exit(EX_OK);
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
case 'o':
|
|
|
|
|
notrunc = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 'r':
|
|
|
|
|
allowreplace = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
case 's':
|
1998-08-03 05:56:20 +00:00
|
|
|
|
DontBlameSendmail |= DBS_MAPINUNSAFEDIRPATH|DBS_WRITEMAPTOHARDLINK|DBS_WRITEMAPTOSYMLINK|DBS_LINKEDMAPINWRITABLEDIR;
|
1997-06-27 14:53:01 +00:00
|
|
|
|
break;
|
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
case 'v':
|
|
|
|
|
verbose = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
type = T_ERR;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (!bitset(DBS_WRITEMAPTOSYMLINK, DontBlameSendmail))
|
|
|
|
|
sff |= SFF_NOSLINK;
|
|
|
|
|
if (!bitset(DBS_WRITEMAPTOHARDLINK, DontBlameSendmail))
|
|
|
|
|
sff |= SFF_NOHLINK;
|
|
|
|
|
if (!bitset(DBS_LINKEDMAPINWRITABLEDIR, DontBlameSendmail))
|
|
|
|
|
sff |= SFF_NOWLINK;
|
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
argc -= optind;
|
|
|
|
|
argv += optind;
|
|
|
|
|
if (argc != 2)
|
|
|
|
|
type = T_ERR;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
typename = argv[0];
|
|
|
|
|
mapname = argv[1];
|
|
|
|
|
ext = NULL;
|
|
|
|
|
|
|
|
|
|
if (strcmp(typename, "dbm") == 0)
|
|
|
|
|
{
|
|
|
|
|
type = T_DBM;
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(typename, "btree") == 0)
|
|
|
|
|
{
|
|
|
|
|
type = T_BTREE;
|
|
|
|
|
ext = ".db";
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(typename, "hash") == 0)
|
|
|
|
|
{
|
|
|
|
|
type = T_HASH;
|
|
|
|
|
ext = ".db";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
type = T_UNKNOWN;
|
|
|
|
|
}
|
|
|
|
|
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if _FFR_TRUSTED_USER
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if ((cfp = fopen(cfile, "r")) == NULL)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "mailstats: ");
|
|
|
|
|
perror(cfile);
|
|
|
|
|
exit(EX_NOINPUT);
|
|
|
|
|
}
|
|
|
|
|
while (fgets(buf, sizeof(buf), cfp) != NULL)
|
|
|
|
|
{
|
|
|
|
|
register char *b;
|
|
|
|
|
|
|
|
|
|
if ((b = strchr(buf, '\n')) != NULL)
|
|
|
|
|
*b = '\0';
|
|
|
|
|
|
|
|
|
|
b = buf;
|
|
|
|
|
switch (*b++)
|
|
|
|
|
{
|
|
|
|
|
case 'O': /* option */
|
1999-01-12 12:38:06 +00:00
|
|
|
|
if (strncasecmp(b, " TrustedUser", 12) == 0 &&
|
|
|
|
|
!(isascii(b[12]) && isalnum(b[12])))
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
b = strchr(b, '=');
|
|
|
|
|
if (b == NULL)
|
|
|
|
|
continue;
|
|
|
|
|
while (isascii(*++b) && isspace(*b))
|
|
|
|
|
continue;
|
|
|
|
|
if (isascii(*b) && isdigit(*b))
|
1999-01-12 12:38:06 +00:00
|
|
|
|
TrustedUid = atoi(b);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
register struct passwd *pw;
|
|
|
|
|
|
1999-01-12 12:38:06 +00:00
|
|
|
|
TrustedUid = 0;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
pw = getpwnam(b);
|
|
|
|
|
if (pw == NULL)
|
|
|
|
|
fprintf(stderr,
|
1999-01-12 12:38:06 +00:00
|
|
|
|
"TrustedUser: unknown user %s\n", b);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
else
|
1999-01-12 12:38:06 +00:00
|
|
|
|
TrustedUid = pw->pw_uid;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# ifdef UID_MAX
|
1999-01-12 12:38:06 +00:00
|
|
|
|
if (TrustedUid > UID_MAX)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
1999-01-12 12:38:06 +00:00
|
|
|
|
syserr("TrustedUser: uid value (%ld) > UID_MAX (%ld)",
|
|
|
|
|
TrustedUid, UID_MAX);
|
|
|
|
|
TrustedUid = 0;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
}
|
|
|
|
|
# endif
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
(void) fclose(cfp);
|
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case T_ERR:
|
1997-06-27 14:53:01 +00:00
|
|
|
|
#if _FFR_NEW_MAKEMAP_FLAGS
|
|
|
|
|
fprintf(stderr,
|
1999-01-12 12:38:06 +00:00
|
|
|
|
"Usage: %s [-N] [-c cachesize] [-d] [-f] [-l] [-o] [-r] [-s] [-v] type mapname\n",
|
1997-06-27 14:53:01 +00:00
|
|
|
|
progname);
|
1996-11-18 02:26:51 +00:00
|
|
|
|
#else
|
1998-08-03 05:56:20 +00:00
|
|
|
|
fprintf(stderr, "Usage: %s [-N] [-d] [-f] [-o] [-r] [-s] [-v] type mapname\n", progname);
|
1996-11-18 02:26:51 +00:00
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
exit(EX_USAGE);
|
|
|
|
|
|
|
|
|
|
case T_UNKNOWN:
|
|
|
|
|
fprintf(stderr, "%s: Unknown database type %s\n",
|
|
|
|
|
progname, typename);
|
|
|
|
|
exit(EX_USAGE);
|
|
|
|
|
|
|
|
|
|
#ifndef NDBM
|
|
|
|
|
case T_DBM:
|
|
|
|
|
#endif
|
|
|
|
|
#ifndef NEWDB
|
|
|
|
|
case T_BTREE:
|
|
|
|
|
case T_HASH:
|
|
|
|
|
#endif
|
|
|
|
|
fprintf(stderr, "%s: Type %s not supported in this version\n",
|
|
|
|
|
progname, typename);
|
|
|
|
|
exit(EX_UNAVAILABLE);
|
|
|
|
|
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
case T_BTREE:
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1995-12-02 17:30:23 +00:00
|
|
|
|
bzero(&bti, sizeof bti);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
bzero(&dbinfo, sizeof dbinfo);
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
if (allowdups)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1995-12-02 17:30:23 +00:00
|
|
|
|
bti.flags |= R_DUP;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
dbinfo.flags |= DB_DUP;
|
|
|
|
|
# endif
|
|
|
|
|
}
|
1996-10-24 04:51:14 +00:00
|
|
|
|
if (allowdups || allowreplace)
|
|
|
|
|
putflags = 0;
|
|
|
|
|
else
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-10-24 04:51:14 +00:00
|
|
|
|
putflags = R_NOOVERWRITE;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
putflags = DB_NOOVERWRITE;
|
|
|
|
|
# endif
|
|
|
|
|
}
|
1995-12-02 17:30:23 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_HASH:
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-12-03 06:07:35 +00:00
|
|
|
|
bzero(&hinfo, sizeof hinfo);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
bzero(&dbinfo, sizeof dbinfo);
|
|
|
|
|
# endif
|
1996-10-24 04:51:14 +00:00
|
|
|
|
if (allowreplace)
|
|
|
|
|
putflags = 0;
|
|
|
|
|
else
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-10-24 04:51:14 +00:00
|
|
|
|
putflags = R_NOOVERWRITE;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
putflags = DB_NOOVERWRITE;
|
|
|
|
|
# endif
|
|
|
|
|
}
|
1996-10-24 04:51:14 +00:00
|
|
|
|
break;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
#endif
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
case T_DBM:
|
|
|
|
|
if (allowdups)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: Type %s does not support -d (allow dups)\n",
|
|
|
|
|
progname, typename);
|
|
|
|
|
exit(EX_UNAVAILABLE);
|
|
|
|
|
}
|
1996-10-24 04:51:14 +00:00
|
|
|
|
if (allowreplace)
|
|
|
|
|
putflags = DBM_REPLACE;
|
|
|
|
|
else
|
|
|
|
|
putflags = DBM_INSERT;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
break;
|
1996-10-24 04:51:14 +00:00
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** Adjust file names.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
if (ext != NULL)
|
|
|
|
|
{
|
|
|
|
|
int el, fl;
|
|
|
|
|
|
|
|
|
|
el = strlen(ext);
|
|
|
|
|
fl = strlen(mapname);
|
1997-06-27 14:53:01 +00:00
|
|
|
|
if (el + fl + 1 >= sizeof fbuf)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: file name too long", mapname);
|
|
|
|
|
exit(EX_USAGE);
|
|
|
|
|
}
|
1995-12-02 17:30:23 +00:00
|
|
|
|
if (fl < el || strcmp(&mapname[fl - el], ext) != 0)
|
|
|
|
|
{
|
|
|
|
|
strcpy(fbuf, mapname);
|
|
|
|
|
strcat(fbuf, ext);
|
|
|
|
|
mapname = fbuf;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
if (!notrunc)
|
|
|
|
|
sff |= SFF_CREAT;
|
1997-08-04 05:00:07 +00:00
|
|
|
|
switch (type)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
case T_BTREE:
|
|
|
|
|
case T_HASH:
|
|
|
|
|
if (strlen(mapname) >= sizeof dbuf)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: map name too long\n", mapname);
|
|
|
|
|
exit(EX_USAGE);
|
|
|
|
|
}
|
|
|
|
|
strcpy(dbuf, mapname);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if ((st = safefile(dbuf, RealUid, RealGid, RealUserName,
|
1997-08-04 05:00:07 +00:00
|
|
|
|
sff, S_IWUSR, &std)) != 0)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: could not create: %s\n",
|
|
|
|
|
dbuf, errstring(st));
|
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
case T_DBM:
|
|
|
|
|
if (strlen(mapname) + 5 > sizeof dbuf)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: map name too long\n", mapname);
|
|
|
|
|
exit(EX_USAGE);
|
|
|
|
|
}
|
|
|
|
|
sprintf(dbuf, "%s.dir", mapname);
|
1997-08-04 05:00:07 +00:00
|
|
|
|
if ((st = safefile(dbuf, RealUid, RealGid, RealUserName,
|
1998-08-03 05:56:20 +00:00
|
|
|
|
sff, S_IWUSR, &std)) != 0)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: could not create: %s\n",
|
|
|
|
|
dbuf, errstring(st));
|
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
|
|
|
|
sprintf(pbuf, "%s.pag", mapname);
|
1997-08-04 05:00:07 +00:00
|
|
|
|
if ((st = safefile(pbuf, RealUid, RealGid, RealUserName,
|
1998-08-03 05:56:20 +00:00
|
|
|
|
sff, S_IWUSR, &stp)) != 0)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: could not create: %s\n",
|
|
|
|
|
pbuf, errstring(st));
|
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
default:
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: internal error: type %d\n",
|
|
|
|
|
progname,
|
|
|
|
|
type);
|
|
|
|
|
exit(EX_SOFTWARE);
|
|
|
|
|
}
|
1995-12-02 17:30:23 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** Create the database.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
mode = O_RDWR;
|
|
|
|
|
if (!notrunc)
|
|
|
|
|
mode |= O_CREAT|O_TRUNC;
|
1997-06-27 14:53:01 +00:00
|
|
|
|
#if O_EXLOCK
|
1995-12-02 17:30:23 +00:00
|
|
|
|
mode |= O_EXLOCK;
|
|
|
|
|
#else
|
|
|
|
|
/* pre-lock the database */
|
1998-08-03 05:56:20 +00:00
|
|
|
|
fd = safeopen(dbuf, mode & ~O_TRUNC, 0644, sff);
|
1995-12-02 17:30:23 +00:00
|
|
|
|
if (fd < 0)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: cannot create type %s map %s\n",
|
|
|
|
|
progname, typename, mapname);
|
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
case T_DBM:
|
|
|
|
|
dbp.dbm = dbm_open(mapname, mode, 0644);
|
1997-11-10 01:58:17 +00:00
|
|
|
|
if (dbp.dbm != NULL &&
|
|
|
|
|
dbm_dirfno(dbp.dbm) == dbm_pagfno(dbp.dbm))
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "dbm map %s: cannot run with GDBM\n",
|
|
|
|
|
mapname);
|
|
|
|
|
dbm_close(dbp.dbm);
|
|
|
|
|
exit(EX_CONFIG);
|
|
|
|
|
}
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (dbp.dbm != NULL &&
|
|
|
|
|
(filechanged(dbuf, dbm_dirfno(dbp.dbm), &std) ||
|
|
|
|
|
filechanged(pbuf, dbm_pagfno(dbp.dbm), &stp)))
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"dbm map %s: file changed after open\n",
|
|
|
|
|
mapname);
|
|
|
|
|
dbm_close(dbp.dbm);
|
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if _FFR_TRUSTED_USER
|
|
|
|
|
if (geteuid() == 0 && TrustedUid != 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
1999-01-12 12:38:06 +00:00
|
|
|
|
if (fchown(dbm_dirfno(dbp.dbm), TrustedUid, -1) < 0 ||
|
|
|
|
|
fchown(dbm_pagfno(dbp.dbm), TrustedUid, -1) < 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"WARNING: ownership change on %s failed: %s",
|
|
|
|
|
mapname, errstring(errno));
|
|
|
|
|
}
|
|
|
|
|
}
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#endif
|
1998-08-03 05:56:20 +00:00
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
case T_HASH:
|
1996-11-18 02:26:51 +00:00
|
|
|
|
/* tweak some parameters for performance */
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-11-18 02:26:51 +00:00
|
|
|
|
hinfo.nelem = 4096;
|
|
|
|
|
hinfo.cachesize = dbcachesize;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
dbinfo.h_nelem = 4096;
|
|
|
|
|
dbinfo.db_cachesize = dbcachesize;
|
|
|
|
|
# endif
|
1997-06-27 14:53:01 +00:00
|
|
|
|
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-11-18 02:26:51 +00:00
|
|
|
|
dbp.db = dbopen(mapname, mode, 0644, DB_HASH, &hinfo);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
{
|
|
|
|
|
int flags = 0;
|
|
|
|
|
|
|
|
|
|
if (bitset(O_CREAT, mode))
|
|
|
|
|
flags |= DB_CREATE;
|
|
|
|
|
if (bitset(O_TRUNC, mode))
|
|
|
|
|
flags |= DB_TRUNCATE;
|
|
|
|
|
|
|
|
|
|
dbp.db = NULL;
|
|
|
|
|
errno = db_open(mapname, DB_HASH, flags, 0644,
|
|
|
|
|
NULL, &dbinfo, &dbp.db);
|
|
|
|
|
}
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
if (dbp.db != NULL)
|
|
|
|
|
{
|
1998-08-03 05:56:20 +00:00
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
|
|
|
|
fd = dbp.db->fd(dbp.db);
|
|
|
|
|
# else
|
|
|
|
|
fd = -1;
|
|
|
|
|
errno = dbp.db->fd(dbp.db, &fd);
|
|
|
|
|
# endif
|
|
|
|
|
if (filechanged(dbuf, fd, &std))
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"db map %s: file changed after open\n",
|
|
|
|
|
mapname);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1997-06-27 14:53:01 +00:00
|
|
|
|
dbp.db->close(dbp.db);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
errno = dbp.db->close(dbp.db, 0);
|
|
|
|
|
# endif
|
1997-06-27 14:53:01 +00:00
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
1995-12-02 17:30:23 +00:00
|
|
|
|
(void) (*dbp.db->sync)(dbp.db, 0);
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if _FFR_TRUSTED_USER
|
|
|
|
|
if (geteuid() == 0 && TrustedUid != 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
1999-01-12 12:38:06 +00:00
|
|
|
|
if (fchown(fd, TrustedUid, -1) < 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"WARNING: ownership change on %s failed: %s",
|
|
|
|
|
mapname, errstring(errno));
|
|
|
|
|
}
|
|
|
|
|
}
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_BTREE:
|
1996-11-18 02:26:51 +00:00
|
|
|
|
/* tweak some parameters for performance */
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-11-18 02:26:51 +00:00
|
|
|
|
bti.cachesize = dbcachesize;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
dbinfo.db_cachesize = dbcachesize;
|
|
|
|
|
# endif
|
1996-11-18 02:26:51 +00:00
|
|
|
|
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1995-12-02 17:30:23 +00:00
|
|
|
|
dbp.db = dbopen(mapname, mode, 0644, DB_BTREE, &bti);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
{
|
|
|
|
|
int flags = 0;
|
|
|
|
|
|
|
|
|
|
if (bitset(O_CREAT, mode))
|
|
|
|
|
flags |= DB_CREATE;
|
|
|
|
|
if (bitset(O_TRUNC, mode))
|
|
|
|
|
flags |= DB_TRUNCATE;
|
|
|
|
|
|
|
|
|
|
dbp.db = NULL;
|
|
|
|
|
errno = db_open(mapname, DB_BTREE, flags, 0644,
|
|
|
|
|
NULL, &dbinfo, &dbp.db);
|
|
|
|
|
}
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
if (dbp.db != NULL)
|
|
|
|
|
{
|
1998-08-03 05:56:20 +00:00
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
|
|
|
|
fd = dbp.db->fd(dbp.db);
|
|
|
|
|
# else
|
|
|
|
|
fd = -1;
|
|
|
|
|
errno = dbp.db->fd(dbp.db, &fd);
|
|
|
|
|
# endif
|
|
|
|
|
if (filechanged(dbuf, fd, &std))
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"db map %s: file changed after open\n",
|
|
|
|
|
mapname);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1997-06-27 14:53:01 +00:00
|
|
|
|
dbp.db->close(dbp.db);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
errno = dbp.db->close(dbp.db, 0);
|
|
|
|
|
# endif
|
1997-06-27 14:53:01 +00:00
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
1995-12-02 17:30:23 +00:00
|
|
|
|
(void) (*dbp.db->sync)(dbp.db, 0);
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if _FFR_TRUSTED_USER
|
|
|
|
|
if (geteuid() == 0 && TrustedUid != 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
1999-01-12 12:38:06 +00:00
|
|
|
|
if (fchown(fd, TrustedUid, -1) < 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"WARNING: ownership change on %s failed: %s",
|
|
|
|
|
mapname, errstring(errno));
|
|
|
|
|
}
|
|
|
|
|
}
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
default:
|
1997-06-27 14:53:01 +00:00
|
|
|
|
fprintf(stderr, "%s: internal error: type %d\n",
|
|
|
|
|
progname, type);
|
1995-12-02 17:30:23 +00:00
|
|
|
|
exit(EX_SOFTWARE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (dbp.dbx == NULL)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: cannot open type %s map %s\n",
|
|
|
|
|
progname, typename, mapname);
|
|
|
|
|
exit(EX_CANTCREAT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** Copy the data
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
lineno = 0;
|
|
|
|
|
exitstat = EX_OK;
|
|
|
|
|
while (fgets(ibuf, sizeof ibuf, stdin) != NULL)
|
|
|
|
|
{
|
|
|
|
|
register char *p;
|
|
|
|
|
|
|
|
|
|
lineno++;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** Parse the line.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
p = strchr(ibuf, '\n');
|
|
|
|
|
if (p != NULL)
|
|
|
|
|
*p = '\0';
|
|
|
|
|
else if (!feof(stdin))
|
|
|
|
|
{
|
1998-08-03 05:56:20 +00:00
|
|
|
|
fprintf(stderr, "%s: %s: line %d: line too long (%ld bytes max)\n",
|
|
|
|
|
progname, mapname, lineno, (long) sizeof ibuf);
|
1995-12-02 17:30:23 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ibuf[0] == '\0' || ibuf[0] == '#')
|
|
|
|
|
continue;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (isascii(ibuf[0]) && isspace(ibuf[0]))
|
1995-12-02 17:30:23 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: %s: line %d: syntax error (leading space)\n",
|
|
|
|
|
progname, mapname, lineno);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
1998-08-03 05:56:20 +00:00
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
if (type == T_HASH || type == T_BTREE)
|
|
|
|
|
{
|
|
|
|
|
bzero(&key.db, sizeof key.db);
|
|
|
|
|
bzero(&val.db, sizeof val.db);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
1995-12-02 17:30:23 +00:00
|
|
|
|
key.xx.data = ibuf;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
for (p = ibuf; *p != '\0' && !(isascii(*p) && isspace(*p)); p++)
|
1995-12-02 17:30:23 +00:00
|
|
|
|
{
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (foldcase && isascii(*p) && isupper(*p))
|
1995-12-02 17:30:23 +00:00
|
|
|
|
*p = tolower(*p);
|
|
|
|
|
}
|
|
|
|
|
key.xx.size = p - key.xx.data;
|
|
|
|
|
if (inclnull)
|
|
|
|
|
key.xx.size++;
|
|
|
|
|
if (*p != '\0')
|
|
|
|
|
*p++ = '\0';
|
1998-08-03 05:56:20 +00:00
|
|
|
|
while (isascii(*p) && isspace(*p))
|
1995-12-02 17:30:23 +00:00
|
|
|
|
p++;
|
|
|
|
|
if (*p == '\0')
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: %s: line %d: no RHS for LHS %s\n",
|
|
|
|
|
progname, mapname, lineno, key.xx.data);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
val.xx.data = p;
|
|
|
|
|
val.xx.size = strlen(p);
|
|
|
|
|
if (inclnull)
|
|
|
|
|
val.xx.size++;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** Do the database insert.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
if (verbose)
|
|
|
|
|
{
|
|
|
|
|
printf("key=`%s', val=`%s'\n", key.xx.data, val.xx.data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
case T_DBM:
|
1996-10-24 04:51:14 +00:00
|
|
|
|
st = dbm_store(dbp.dbm, key.dbm, val.dbm, putflags);
|
1995-12-02 17:30:23 +00:00
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
case T_BTREE:
|
|
|
|
|
case T_HASH:
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1996-10-24 04:51:14 +00:00
|
|
|
|
st = (*dbp.db->put)(dbp.db, &key.db, &val.db, putflags);
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
errno = (*dbp.db->put)(dbp.db, NULL, &key.db,
|
|
|
|
|
&val.db, putflags);
|
|
|
|
|
switch (errno)
|
|
|
|
|
{
|
|
|
|
|
case DB_KEYEXIST:
|
|
|
|
|
st = 1;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
|
st = 0;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
st = -1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
break;
|
|
|
|
|
#endif
|
1998-08-03 05:56:20 +00:00
|
|
|
|
default:
|
|
|
|
|
break;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (st < 0)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: %s: line %d: key %s: put error\n",
|
|
|
|
|
progname, mapname, lineno, key.xx.data);
|
|
|
|
|
perror(mapname);
|
|
|
|
|
exitstat = EX_IOERR;
|
|
|
|
|
}
|
|
|
|
|
else if (st > 0)
|
|
|
|
|
{
|
1997-06-27 14:53:01 +00:00
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: %s: line %d: key %s: duplicate key\n",
|
1995-12-02 17:30:23 +00:00
|
|
|
|
progname, mapname, lineno, key.xx.data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** Now close the database.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
#ifdef NDBM
|
|
|
|
|
case T_DBM:
|
|
|
|
|
dbm_close(dbp.dbm);
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef NEWDB
|
|
|
|
|
case T_HASH:
|
|
|
|
|
case T_BTREE:
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# if DB_VERSION_MAJOR < 2
|
1995-12-02 17:30:23 +00:00
|
|
|
|
if ((*dbp.db->close)(dbp.db) < 0)
|
1998-08-03 05:56:20 +00:00
|
|
|
|
# else
|
|
|
|
|
if ((errno = (*dbp.db->close)(dbp.db, 0)) != 0)
|
|
|
|
|
# endif
|
1995-12-02 17:30:23 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: %s: error on close\n",
|
|
|
|
|
progname, mapname);
|
|
|
|
|
perror(mapname);
|
|
|
|
|
exitstat = EX_IOERR;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
1998-08-03 05:56:20 +00:00
|
|
|
|
default:
|
|
|
|
|
break;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
#if !O_EXLOCK
|
1995-12-02 17:30:23 +00:00
|
|
|
|
/* release locks */
|
|
|
|
|
close(fd);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
exit (exitstat);
|
|
|
|
|
}
|
|
|
|
|
/*
|
|
|
|
|
** LOCKFILE -- lock a file using flock or (shudder) fcntl locking
|
|
|
|
|
**
|
|
|
|
|
** Parameters:
|
|
|
|
|
** fd -- the file descriptor of the file.
|
1997-06-27 14:53:01 +00:00
|
|
|
|
** filename -- the file name (for error messages).
|
|
|
|
|
** ext -- the filename extension.
|
|
|
|
|
** type -- type of the lock. Bits can be:
|
|
|
|
|
** LOCK_EX -- exclusive lock.
|
|
|
|
|
** LOCK_NB -- non-blocking.
|
1995-12-02 17:30:23 +00:00
|
|
|
|
**
|
|
|
|
|
** Returns:
|
|
|
|
|
** TRUE if the lock was acquired.
|
|
|
|
|
** FALSE otherwise.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
bool
|
1997-06-27 14:53:01 +00:00
|
|
|
|
lockfile(fd, filename, ext, type)
|
1995-12-02 17:30:23 +00:00
|
|
|
|
int fd;
|
1997-06-27 14:53:01 +00:00
|
|
|
|
char *filename;
|
|
|
|
|
char *ext;
|
|
|
|
|
int type;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
{
|
|
|
|
|
# if !HASFLOCK
|
|
|
|
|
int action;
|
|
|
|
|
struct flock lfd;
|
|
|
|
|
extern int errno;
|
|
|
|
|
|
|
|
|
|
bzero(&lfd, sizeof lfd);
|
1997-06-27 14:53:01 +00:00
|
|
|
|
if (bitset(LOCK_UN, type))
|
|
|
|
|
lfd.l_type = F_UNLCK;
|
|
|
|
|
else if (bitset(LOCK_EX, type))
|
|
|
|
|
lfd.l_type = F_WRLCK;
|
|
|
|
|
else
|
|
|
|
|
lfd.l_type = F_RDLCK;
|
|
|
|
|
if (bitset(LOCK_NB, type))
|
|
|
|
|
action = F_SETLK;
|
|
|
|
|
else
|
|
|
|
|
action = F_SETLKW;
|
1995-12-02 17:30:23 +00:00
|
|
|
|
|
|
|
|
|
if (fcntl(fd, action, &lfd) >= 0)
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
** On SunOS, if you are testing using -oQ/tmp/mqueue or
|
|
|
|
|
** -oA/tmp/aliases or anything like that, and /tmp is mounted
|
|
|
|
|
** as type "tmp" (that is, served from swap space), the
|
|
|
|
|
** previous fcntl will fail with "Invalid argument" errors.
|
|
|
|
|
** Since this is fairly common during testing, we will assume
|
|
|
|
|
** that this indicates that the lock is successfully grabbed.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
if (errno == EINVAL)
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
# else /* HASFLOCK */
|
|
|
|
|
|
1997-06-27 14:53:01 +00:00
|
|
|
|
if (flock(fd, type) >= 0)
|
1995-12-02 17:30:23 +00:00
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
# endif
|
|
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
1997-06-27 14:53:01 +00:00
|
|
|
|
|
1998-08-03 05:56:20 +00:00
|
|
|
|
/*VARARGS1*/
|
1997-06-27 14:53:01 +00:00
|
|
|
|
void
|
|
|
|
|
#ifdef __STDC__
|
|
|
|
|
message(const char *msg, ...)
|
|
|
|
|
#else
|
|
|
|
|
message(msg, va_alist)
|
|
|
|
|
const char *msg;
|
|
|
|
|
va_dcl
|
|
|
|
|
#endif
|
|
|
|
|
{
|
|
|
|
|
const char *m;
|
|
|
|
|
VA_LOCAL_DECL
|
|
|
|
|
|
|
|
|
|
m = msg;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (isascii(m[0]) && isdigit(m[0]) &&
|
|
|
|
|
isascii(m[1]) && isdigit(m[1]) &&
|
|
|
|
|
isascii(m[2]) && isdigit(m[2]) && m[3] == ' ')
|
1997-06-27 14:53:01 +00:00
|
|
|
|
m += 4;
|
|
|
|
|
VA_START(msg);
|
|
|
|
|
vfprintf(stderr, m, ap);
|
|
|
|
|
VA_END;
|
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
|
}
|
|
|
|
|
|
1998-08-03 05:56:20 +00:00
|
|
|
|
/*VARARGS1*/
|
1997-06-27 14:53:01 +00:00
|
|
|
|
void
|
|
|
|
|
#ifdef __STDC__
|
|
|
|
|
syserr(const char *msg, ...)
|
|
|
|
|
#else
|
|
|
|
|
syserr(msg, va_alist)
|
|
|
|
|
const char *msg;
|
|
|
|
|
va_dcl
|
|
|
|
|
#endif
|
|
|
|
|
{
|
|
|
|
|
const char *m;
|
|
|
|
|
VA_LOCAL_DECL
|
|
|
|
|
|
|
|
|
|
m = msg;
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (isascii(m[0]) && isdigit(m[0]) &&
|
|
|
|
|
isascii(m[1]) && isdigit(m[1]) &&
|
|
|
|
|
isascii(m[2]) && isdigit(m[2]) && m[3] == ' ')
|
1997-06-27 14:53:01 +00:00
|
|
|
|
m += 4;
|
|
|
|
|
VA_START(msg);
|
|
|
|
|
vfprintf(stderr, m, ap);
|
|
|
|
|
VA_END;
|
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
|
errstring(err)
|
|
|
|
|
int err;
|
|
|
|
|
{
|
1999-01-12 12:38:06 +00:00
|
|
|
|
#if !HASSTRERROR
|
1997-06-27 14:53:01 +00:00
|
|
|
|
static char errstr[64];
|
1999-01-12 12:38:06 +00:00
|
|
|
|
# if !defined(ERRLIST_PREDEFINED)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
extern char *sys_errlist[];
|
|
|
|
|
extern int sys_nerr;
|
1999-01-12 12:38:06 +00:00
|
|
|
|
# endif
|
1997-06-27 14:53:01 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* handle pseudo-errors internal to sendmail */
|
|
|
|
|
switch (err)
|
|
|
|
|
{
|
|
|
|
|
case E_SM_OPENTIMEOUT:
|
|
|
|
|
return "Timeout on file open";
|
|
|
|
|
|
|
|
|
|
case E_SM_NOSLINK:
|
|
|
|
|
return "Symbolic links not allowed";
|
|
|
|
|
|
|
|
|
|
case E_SM_NOHLINK:
|
|
|
|
|
return "Hard links not allowed";
|
|
|
|
|
|
|
|
|
|
case E_SM_REGONLY:
|
|
|
|
|
return "Regular files only";
|
|
|
|
|
|
|
|
|
|
case E_SM_ISEXEC:
|
|
|
|
|
return "Executable files not allowed";
|
|
|
|
|
|
|
|
|
|
case E_SM_WWDIR:
|
|
|
|
|
return "World writable directory";
|
|
|
|
|
|
|
|
|
|
case E_SM_GWDIR:
|
|
|
|
|
return "Group writable directory";
|
|
|
|
|
|
|
|
|
|
case E_SM_FILECHANGE:
|
|
|
|
|
return "File changed after open";
|
|
|
|
|
|
|
|
|
|
case E_SM_WWFILE:
|
|
|
|
|
return "World writable file";
|
|
|
|
|
|
|
|
|
|
case E_SM_GWFILE:
|
|
|
|
|
return "Group writable file";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if HASSTRERROR
|
|
|
|
|
return strerror(err);
|
|
|
|
|
#else
|
1998-08-03 05:56:20 +00:00
|
|
|
|
if (err < 0 || err >= sys_nerr)
|
1997-06-27 14:53:01 +00:00
|
|
|
|
{
|
|
|
|
|
sprintf(errstr, "Error %d", err);
|
|
|
|
|
return errstr;
|
|
|
|
|
}
|
|
|
|
|
return sys_errlist[err];
|
|
|
|
|
#endif
|
|
|
|
|
}
|