xref: /freebsd/contrib/sendmail/editmap/editmap.c (revision 2fb4f839)
140266059SGregory Neil Shapiro /*
25dd76dd0SGregory Neil Shapiro  * Copyright (c) 1998-2002, 2004 Proofpoint, Inc. and its suppliers.
340266059SGregory Neil Shapiro  *	All rights reserved.
440266059SGregory Neil Shapiro  * Copyright (c) 1992 Eric P. Allman.  All rights reserved.
540266059SGregory Neil Shapiro  * Copyright (c) 1992, 1993
640266059SGregory Neil Shapiro  *	The Regents of the University of California.  All rights reserved.
740266059SGregory Neil Shapiro  *
840266059SGregory Neil Shapiro  * By using this file, you agree to the terms and conditions set
940266059SGregory Neil Shapiro  * forth in the LICENSE file which can be found at the top level of
1040266059SGregory Neil Shapiro  * the sendmail distribution.
1140266059SGregory Neil Shapiro  *
1240266059SGregory Neil Shapiro  */
1340266059SGregory Neil Shapiro 
1440266059SGregory Neil Shapiro #include <sm/gen.h>
1540266059SGregory Neil Shapiro #ifndef lint
1640266059SGregory Neil Shapiro SM_UNUSED(static char copyright[]) =
175dd76dd0SGregory Neil Shapiro "@(#) Copyright (c) 1998-2001 Proofpoint, Inc. and its suppliers.\n\
1840266059SGregory Neil Shapiro 	All rights reserved.\n\
1940266059SGregory Neil Shapiro      Copyright (c) 1992 Eric P. Allman.  All rights reserved.\n\
2040266059SGregory Neil Shapiro      Copyright (c) 1992, 1993\n\
2140266059SGregory Neil Shapiro 	The Regents of the University of California.  All rights reserved.\n";
2240266059SGregory Neil Shapiro #endif /* ! lint */
2340266059SGregory Neil Shapiro 
2440266059SGregory Neil Shapiro #ifndef lint
254313cc83SGregory Neil Shapiro SM_UNUSED(static char id[]) = "@(#)$Id: editmap.c,v 1.26 2013-11-22 20:51:26 ca Exp $";
265b0945b5SGregory Neil Shapiro #endif
2740266059SGregory Neil Shapiro 
2840266059SGregory Neil Shapiro 
2940266059SGregory Neil Shapiro #include <sys/types.h>
3040266059SGregory Neil Shapiro #ifndef ISC_UNIX
3140266059SGregory Neil Shapiro # include <sys/file.h>
325b0945b5SGregory Neil Shapiro #endif
3340266059SGregory Neil Shapiro #include <ctype.h>
3440266059SGregory Neil Shapiro #include <stdlib.h>
3540266059SGregory Neil Shapiro #include <unistd.h>
3640266059SGregory Neil Shapiro #ifdef EX_OK
3740266059SGregory Neil Shapiro # undef EX_OK		/* unistd.h may have another use for this */
385b0945b5SGregory Neil Shapiro #endif
3940266059SGregory Neil Shapiro #include <sysexits.h>
4040266059SGregory Neil Shapiro #include <assert.h>
412fb4f839SGregory Neil Shapiro #include <sm/sendmail.h>
4240266059SGregory Neil Shapiro #include <sendmail/sendmail.h>
4340266059SGregory Neil Shapiro #include <sendmail/pathnames.h>
4440266059SGregory Neil Shapiro #include <libsmdb/smdb.h>
4540266059SGregory Neil Shapiro 
4640266059SGregory Neil Shapiro uid_t	RealUid;
4740266059SGregory Neil Shapiro gid_t	RealGid;
4840266059SGregory Neil Shapiro char	*RealUserName;
4940266059SGregory Neil Shapiro uid_t	RunAsUid;
50ffb83623SGregory Neil Shapiro gid_t	RunAsGid;
5140266059SGregory Neil Shapiro char	*RunAsUserName;
5240266059SGregory Neil Shapiro int	Verbose = 2;
5340266059SGregory Neil Shapiro bool	DontInitGroups = false;
5440266059SGregory Neil Shapiro uid_t	TrustedUid = 0;
5540266059SGregory Neil Shapiro BITMAP256 DontBlameSendmail;
5640266059SGregory Neil Shapiro 
5740266059SGregory Neil Shapiro #define BUFSIZE		1024
5840266059SGregory Neil Shapiro #define ISSEP(c) (isascii(c) && isspace(c))
5940266059SGregory Neil Shapiro 
6040266059SGregory Neil Shapiro 
61b6bacd31SGregory Neil Shapiro static void usage __P((char *));
62b6bacd31SGregory Neil Shapiro 
6340266059SGregory Neil Shapiro static void
usage(progname)6440266059SGregory Neil Shapiro usage(progname)
6540266059SGregory Neil Shapiro 	char *progname;
6640266059SGregory Neil Shapiro {
6740266059SGregory Neil Shapiro 	fprintf(stderr,
6840266059SGregory Neil Shapiro 		"Usage: %s [-C cffile] [-N] [-f] [-q|-u|-x] maptype mapname key [ \"value ...\" ]\n",
6940266059SGregory Neil Shapiro 		progname);
7040266059SGregory Neil Shapiro 	exit(EX_USAGE);
7140266059SGregory Neil Shapiro }
7240266059SGregory Neil Shapiro 
7340266059SGregory Neil Shapiro int
main(argc,argv)7440266059SGregory Neil Shapiro main(argc, argv)
7540266059SGregory Neil Shapiro 	int argc;
7640266059SGregory Neil Shapiro 	char **argv;
7740266059SGregory Neil Shapiro {
7840266059SGregory Neil Shapiro 	char *progname;
7940266059SGregory Neil Shapiro 	char *cfile;
8040266059SGregory Neil Shapiro 	bool query = false;
8140266059SGregory Neil Shapiro 	bool update = false;
8240266059SGregory Neil Shapiro 	bool remove = false;
8340266059SGregory Neil Shapiro 	bool inclnull = false;
8440266059SGregory Neil Shapiro 	bool foldcase = true;
8540266059SGregory Neil Shapiro 	unsigned int nops = 0;
8640266059SGregory Neil Shapiro 	int exitstat;
8740266059SGregory Neil Shapiro 	int opt;
8840266059SGregory Neil Shapiro 	char *typename = NULL;
8940266059SGregory Neil Shapiro 	char *mapname = NULL;
9040266059SGregory Neil Shapiro 	char *keyname = NULL;
9140266059SGregory Neil Shapiro 	char *value = NULL;
9240266059SGregory Neil Shapiro 	int mode;
9340266059SGregory Neil Shapiro 	int smode;
9440266059SGregory Neil Shapiro 	int putflags = 0;
9540266059SGregory Neil Shapiro 	long sff = SFF_ROOTOK|SFF_REGONLY;
9640266059SGregory Neil Shapiro 	struct passwd *pw;
9740266059SGregory Neil Shapiro 	SMDB_DATABASE *database;
9840266059SGregory Neil Shapiro 	SMDB_DBENT db_key, db_val;
9940266059SGregory Neil Shapiro 	SMDB_DBPARAMS params;
10040266059SGregory Neil Shapiro 	SMDB_USER_INFO user_info;
10140266059SGregory Neil Shapiro #if HASFCHOWN
10240266059SGregory Neil Shapiro 	FILE *cfp;
10340266059SGregory Neil Shapiro 	char buf[MAXLINE];
1045b0945b5SGregory Neil Shapiro #endif
10540266059SGregory Neil Shapiro 	static char rnamebuf[MAXNAME];	/* holds RealUserName */
10640266059SGregory Neil Shapiro 	extern char *optarg;
10740266059SGregory Neil Shapiro 	extern int optind;
10840266059SGregory Neil Shapiro 
10940266059SGregory Neil Shapiro 	memset(&params, '\0', sizeof params);
11040266059SGregory Neil Shapiro 	params.smdbp_cache_size = 1024 * 1024;
11140266059SGregory Neil Shapiro 
11240266059SGregory Neil Shapiro 	progname = strrchr(argv[0], '/');
11340266059SGregory Neil Shapiro 	if (progname != NULL)
11440266059SGregory Neil Shapiro 		progname++;
11540266059SGregory Neil Shapiro 	else
11640266059SGregory Neil Shapiro 		progname = argv[0];
11740266059SGregory Neil Shapiro 	cfile = _PATH_SENDMAILCF;
11840266059SGregory Neil Shapiro 
11940266059SGregory Neil Shapiro 	clrbitmap(DontBlameSendmail);
12040266059SGregory Neil Shapiro 	RunAsUid = RealUid = getuid();
12140266059SGregory Neil Shapiro 	RunAsGid = RealGid = getgid();
12240266059SGregory Neil Shapiro 	pw = getpwuid(RealUid);
12340266059SGregory Neil Shapiro 	if (pw != NULL)
12440266059SGregory Neil Shapiro 		(void) sm_strlcpy(rnamebuf, pw->pw_name, sizeof rnamebuf);
12540266059SGregory Neil Shapiro 	else
12640266059SGregory Neil Shapiro 		(void) sm_snprintf(rnamebuf, sizeof rnamebuf,
12740266059SGregory Neil Shapiro 				   "Unknown UID %d", (int) RealUid);
12840266059SGregory Neil Shapiro 	RunAsUserName = RealUserName = rnamebuf;
12940266059SGregory Neil Shapiro 	user_info.smdbu_id = RunAsUid;
13040266059SGregory Neil Shapiro 	user_info.smdbu_group_id = RunAsGid;
13140266059SGregory Neil Shapiro 	(void) sm_strlcpy(user_info.smdbu_name, RunAsUserName,
13240266059SGregory Neil Shapiro 			  SMDB_MAX_USER_NAME_LEN);
13340266059SGregory Neil Shapiro 
134da7d7b9cSGregory Neil Shapiro #define OPTIONS		"C:fquxN"
13540266059SGregory Neil Shapiro 	while ((opt = getopt(argc, argv, OPTIONS)) != -1)
13640266059SGregory Neil Shapiro 	{
13740266059SGregory Neil Shapiro 		switch (opt)
13840266059SGregory Neil Shapiro 		{
13940266059SGregory Neil Shapiro 		  case 'C':
14040266059SGregory Neil Shapiro 			cfile = optarg;
14140266059SGregory Neil Shapiro 			break;
14240266059SGregory Neil Shapiro 
14340266059SGregory Neil Shapiro 		  case 'f':
14440266059SGregory Neil Shapiro 			foldcase = false;
14540266059SGregory Neil Shapiro 			break;
14640266059SGregory Neil Shapiro 
14740266059SGregory Neil Shapiro 		  case 'q':
14840266059SGregory Neil Shapiro 			query = true;
14940266059SGregory Neil Shapiro 			nops++;
15040266059SGregory Neil Shapiro 			break;
15140266059SGregory Neil Shapiro 
15240266059SGregory Neil Shapiro 		  case 'u':
15340266059SGregory Neil Shapiro 			update = true;
15440266059SGregory Neil Shapiro 			nops++;
15540266059SGregory Neil Shapiro 			break;
15640266059SGregory Neil Shapiro 
15740266059SGregory Neil Shapiro 		  case 'x':
15840266059SGregory Neil Shapiro 			remove = true;
15940266059SGregory Neil Shapiro 			nops++;
16040266059SGregory Neil Shapiro 			break;
16140266059SGregory Neil Shapiro 
16240266059SGregory Neil Shapiro 		  case 'N':
16340266059SGregory Neil Shapiro 			inclnull = true;
16440266059SGregory Neil Shapiro 			break;
16540266059SGregory Neil Shapiro 
16640266059SGregory Neil Shapiro 		  default:
16740266059SGregory Neil Shapiro 			usage(progname);
16840266059SGregory Neil Shapiro 			assert(0);  /* NOTREACHED */
16940266059SGregory Neil Shapiro 		}
17040266059SGregory Neil Shapiro 	}
17140266059SGregory Neil Shapiro 
17240266059SGregory Neil Shapiro 	if (!bitnset(DBS_WRITEMAPTOSYMLINK, DontBlameSendmail))
17340266059SGregory Neil Shapiro 		sff |= SFF_NOSLINK;
17440266059SGregory Neil Shapiro 	if (!bitnset(DBS_WRITEMAPTOHARDLINK, DontBlameSendmail))
17540266059SGregory Neil Shapiro 		sff |= SFF_NOHLINK;
17640266059SGregory Neil Shapiro 	if (!bitnset(DBS_LINKEDMAPINWRITABLEDIR, DontBlameSendmail))
17740266059SGregory Neil Shapiro 		sff |= SFF_NOWLINK;
17840266059SGregory Neil Shapiro 
17940266059SGregory Neil Shapiro 	argc -= optind;
18040266059SGregory Neil Shapiro 	argv += optind;
18140266059SGregory Neil Shapiro 	if ((nops != 1) ||
18240266059SGregory Neil Shapiro 	    (query && argc != 3) ||
18340266059SGregory Neil Shapiro 	    (remove && argc != 3) ||
18440266059SGregory Neil Shapiro 	    (update && argc <= 3))
18540266059SGregory Neil Shapiro 	{
18640266059SGregory Neil Shapiro 		usage(progname);
18740266059SGregory Neil Shapiro 		assert(0);  /* NOTREACHED */
18840266059SGregory Neil Shapiro 	}
18940266059SGregory Neil Shapiro 
19040266059SGregory Neil Shapiro 	typename = argv[0];
19140266059SGregory Neil Shapiro 	mapname = argv[1];
19240266059SGregory Neil Shapiro 	keyname = argv[2];
19340266059SGregory Neil Shapiro 	if (update)
19440266059SGregory Neil Shapiro 		value = argv[3];
19540266059SGregory Neil Shapiro 
19640266059SGregory Neil Shapiro 	if (foldcase)
19740266059SGregory Neil Shapiro 	{
1982fb4f839SGregory Neil Shapiro 		char *lower;
19940266059SGregory Neil Shapiro 
2002fb4f839SGregory Neil Shapiro 		lower = makelower(keyname);
2012fb4f839SGregory Neil Shapiro 
2022fb4f839SGregory Neil Shapiro 		/* if it is different then it is a static variable */
2032fb4f839SGregory Neil Shapiro 		if (keyname != lower)
2042fb4f839SGregory Neil Shapiro 			keyname = lower;
20540266059SGregory Neil Shapiro 	}
20640266059SGregory Neil Shapiro 
20740266059SGregory Neil Shapiro 
20840266059SGregory Neil Shapiro #if HASFCHOWN
20940266059SGregory Neil Shapiro 	/* Find TrustedUser value in sendmail.cf */
21040266059SGregory Neil Shapiro 	if ((cfp = fopen(cfile, "r")) == NULL)
21140266059SGregory Neil Shapiro 	{
21240266059SGregory Neil Shapiro 		fprintf(stderr, "%s: %s: %s\n", progname,
21340266059SGregory Neil Shapiro 			cfile, sm_errstring(errno));
21440266059SGregory Neil Shapiro 		exit(EX_NOINPUT);
21540266059SGregory Neil Shapiro 	}
21640266059SGregory Neil Shapiro 	while (fgets(buf, sizeof(buf), cfp) != NULL)
21740266059SGregory Neil Shapiro 	{
21840266059SGregory Neil Shapiro 		register char *b;
21940266059SGregory Neil Shapiro 
22040266059SGregory Neil Shapiro 		if ((b = strchr(buf, '\n')) != NULL)
22140266059SGregory Neil Shapiro 			*b = '\0';
22240266059SGregory Neil Shapiro 
22340266059SGregory Neil Shapiro 		b = buf;
22440266059SGregory Neil Shapiro 		switch (*b++)
22540266059SGregory Neil Shapiro 		{
22640266059SGregory Neil Shapiro 		  case 'O':		/* option */
22740266059SGregory Neil Shapiro 			if (strncasecmp(b, " TrustedUser", 12) == 0 &&
22840266059SGregory Neil Shapiro 			    !(isascii(b[12]) && isalnum(b[12])))
22940266059SGregory Neil Shapiro 			{
23040266059SGregory Neil Shapiro 				b = strchr(b, '=');
23140266059SGregory Neil Shapiro 				if (b == NULL)
23240266059SGregory Neil Shapiro 					continue;
23340266059SGregory Neil Shapiro 				while (isascii(*++b) && isspace(*b))
23440266059SGregory Neil Shapiro 					continue;
23540266059SGregory Neil Shapiro 				if (isascii(*b) && isdigit(*b))
23640266059SGregory Neil Shapiro 					TrustedUid = atoi(b);
23740266059SGregory Neil Shapiro 				else
23840266059SGregory Neil Shapiro 				{
23940266059SGregory Neil Shapiro 					TrustedUid = 0;
24040266059SGregory Neil Shapiro 					pw = getpwnam(b);
24140266059SGregory Neil Shapiro 					if (pw == NULL)
24240266059SGregory Neil Shapiro 						fprintf(stderr,
24340266059SGregory Neil Shapiro 							"TrustedUser: unknown user %s\n", b);
24440266059SGregory Neil Shapiro 					else
24540266059SGregory Neil Shapiro 						TrustedUid = pw->pw_uid;
24640266059SGregory Neil Shapiro 				}
24740266059SGregory Neil Shapiro 
24840266059SGregory Neil Shapiro # ifdef UID_MAX
24940266059SGregory Neil Shapiro 				if (TrustedUid > UID_MAX)
25040266059SGregory Neil Shapiro 				{
25140266059SGregory Neil Shapiro 					fprintf(stderr,
25240266059SGregory Neil Shapiro 						"TrustedUser: uid value (%ld) > UID_MAX (%ld)",
25340266059SGregory Neil Shapiro 						(long) TrustedUid,
25440266059SGregory Neil Shapiro 						(long) UID_MAX);
25540266059SGregory Neil Shapiro 					TrustedUid = 0;
25640266059SGregory Neil Shapiro 				}
25740266059SGregory Neil Shapiro # endif /* UID_MAX */
25840266059SGregory Neil Shapiro 				break;
25940266059SGregory Neil Shapiro 			}
26040266059SGregory Neil Shapiro 
26140266059SGregory Neil Shapiro 
26240266059SGregory Neil Shapiro 		  default:
26340266059SGregory Neil Shapiro 			continue;
26440266059SGregory Neil Shapiro 		}
26540266059SGregory Neil Shapiro 	}
26640266059SGregory Neil Shapiro 	(void) fclose(cfp);
26740266059SGregory Neil Shapiro #endif /* HASFCHOWN */
26840266059SGregory Neil Shapiro 
26940266059SGregory Neil Shapiro 	if (query)
27040266059SGregory Neil Shapiro 	{
27140266059SGregory Neil Shapiro 		mode = O_RDONLY;
27240266059SGregory Neil Shapiro 		smode = S_IRUSR;
27340266059SGregory Neil Shapiro 	}
27440266059SGregory Neil Shapiro 	else
27540266059SGregory Neil Shapiro 	{
27640266059SGregory Neil Shapiro 		mode = O_RDWR | O_CREAT;
27740266059SGregory Neil Shapiro 		sff |= SFF_CREAT|SFF_NOTEXCL;
27840266059SGregory Neil Shapiro 		smode = S_IWUSR;
27940266059SGregory Neil Shapiro 	}
28040266059SGregory Neil Shapiro 
28140266059SGregory Neil Shapiro 	params.smdbp_num_elements = 4096;
28240266059SGregory Neil Shapiro 
28340266059SGregory Neil Shapiro 	errno = smdb_open_database(&database, mapname, mode, smode, sff,
28440266059SGregory Neil Shapiro 				   typename, &user_info, &params);
28540266059SGregory Neil Shapiro 	if (errno != SMDBE_OK)
28640266059SGregory Neil Shapiro 	{
28740266059SGregory Neil Shapiro 		char *hint;
28840266059SGregory Neil Shapiro 
28940266059SGregory Neil Shapiro 		if (errno == SMDBE_UNSUPPORTED_DB_TYPE &&
29040266059SGregory Neil Shapiro 		    (hint = smdb_db_definition(typename)) != NULL)
29140266059SGregory Neil Shapiro 			fprintf(stderr,
29240266059SGregory Neil Shapiro 				"%s: Need to recompile with -D%s for %s support\n",
29340266059SGregory Neil Shapiro 				progname, hint, typename);
29440266059SGregory Neil Shapiro 		else
29540266059SGregory Neil Shapiro 			fprintf(stderr,
29640266059SGregory Neil Shapiro 				"%s: error opening type %s map %s: %s\n",
29740266059SGregory Neil Shapiro 				progname, typename, mapname,
29840266059SGregory Neil Shapiro 				sm_errstring(errno));
29940266059SGregory Neil Shapiro 		exit(EX_CANTCREAT);
30040266059SGregory Neil Shapiro 	}
30140266059SGregory Neil Shapiro 
30240266059SGregory Neil Shapiro 	(void) database->smdb_sync(database, 0);
30340266059SGregory Neil Shapiro 
30440266059SGregory Neil Shapiro 	if (geteuid() == 0 && TrustedUid != 0)
30540266059SGregory Neil Shapiro 	{
30640266059SGregory Neil Shapiro 		errno = database->smdb_set_owner(database, TrustedUid, -1);
30740266059SGregory Neil Shapiro 		if (errno != SMDBE_OK)
30840266059SGregory Neil Shapiro 		{
30940266059SGregory Neil Shapiro 			fprintf(stderr,
31040266059SGregory Neil Shapiro 				"WARNING: ownership change on %s failed %s",
31140266059SGregory Neil Shapiro 				mapname, sm_errstring(errno));
31240266059SGregory Neil Shapiro 		}
31340266059SGregory Neil Shapiro 	}
31440266059SGregory Neil Shapiro 
31540266059SGregory Neil Shapiro 	exitstat = EX_OK;
31640266059SGregory Neil Shapiro 	if (query)
31740266059SGregory Neil Shapiro 	{
31840266059SGregory Neil Shapiro 		memset(&db_key, '\0', sizeof db_key);
31940266059SGregory Neil Shapiro 		memset(&db_val, '\0', sizeof db_val);
32040266059SGregory Neil Shapiro 
32140266059SGregory Neil Shapiro 		db_key.data = keyname;
32240266059SGregory Neil Shapiro 		db_key.size = strlen(keyname);
32340266059SGregory Neil Shapiro 		if (inclnull)
32440266059SGregory Neil Shapiro 			db_key.size++;
32540266059SGregory Neil Shapiro 
32640266059SGregory Neil Shapiro 		errno = database->smdb_get(database, &db_key, &db_val, 0);
32740266059SGregory Neil Shapiro 		if (errno != SMDBE_OK)
32840266059SGregory Neil Shapiro 		{
32940266059SGregory Neil Shapiro 			/* XXX - Need to distinguish between not found */
33040266059SGregory Neil Shapiro 			fprintf(stderr,
33140266059SGregory Neil Shapiro 				"%s: couldn't find key %s in map %s\n",
33240266059SGregory Neil Shapiro 				progname, keyname, mapname);
33340266059SGregory Neil Shapiro 			exitstat = EX_UNAVAILABLE;
33440266059SGregory Neil Shapiro 		}
33540266059SGregory Neil Shapiro 		else
33640266059SGregory Neil Shapiro 		{
33740266059SGregory Neil Shapiro 			printf("%.*s\n", (int) db_val.size,
33840266059SGregory Neil Shapiro 			       (char *) db_val.data);
33940266059SGregory Neil Shapiro 		}
34040266059SGregory Neil Shapiro 	}
34140266059SGregory Neil Shapiro 	else if (update)
34240266059SGregory Neil Shapiro 	{
34340266059SGregory Neil Shapiro 		memset(&db_key, '\0', sizeof db_key);
34440266059SGregory Neil Shapiro 		memset(&db_val, '\0', sizeof db_val);
34540266059SGregory Neil Shapiro 
34640266059SGregory Neil Shapiro 		db_key.data = keyname;
34740266059SGregory Neil Shapiro 		db_key.size = strlen(keyname);
34840266059SGregory Neil Shapiro 		if (inclnull)
34940266059SGregory Neil Shapiro 			db_key.size++;
35040266059SGregory Neil Shapiro 		db_val.data = value;
35140266059SGregory Neil Shapiro 		db_val.size = strlen(value);
35240266059SGregory Neil Shapiro 		if (inclnull)
35340266059SGregory Neil Shapiro 			db_val.size++;
35440266059SGregory Neil Shapiro 
35540266059SGregory Neil Shapiro 		errno = database->smdb_put(database, &db_key, &db_val,
35640266059SGregory Neil Shapiro 					   putflags);
35740266059SGregory Neil Shapiro 		if (errno != SMDBE_OK)
35840266059SGregory Neil Shapiro 		{
35940266059SGregory Neil Shapiro 			fprintf(stderr,
36040266059SGregory Neil Shapiro 				"%s: error updating (%s, %s) in map %s: %s\n",
36140266059SGregory Neil Shapiro 				progname, keyname, value, mapname,
36240266059SGregory Neil Shapiro 				sm_errstring(errno));
36340266059SGregory Neil Shapiro 			exitstat = EX_IOERR;
36440266059SGregory Neil Shapiro 		}
36540266059SGregory Neil Shapiro 	}
36640266059SGregory Neil Shapiro 	else if (remove)
36740266059SGregory Neil Shapiro 	{
36840266059SGregory Neil Shapiro 		memset(&db_key, '\0', sizeof db_key);
36940266059SGregory Neil Shapiro 		memset(&db_val, '\0', sizeof db_val);
37040266059SGregory Neil Shapiro 
37140266059SGregory Neil Shapiro 		db_key.data = keyname;
37240266059SGregory Neil Shapiro 		db_key.size = strlen(keyname);
37340266059SGregory Neil Shapiro 		if (inclnull)
37440266059SGregory Neil Shapiro 			db_key.size++;
37540266059SGregory Neil Shapiro 
37640266059SGregory Neil Shapiro 		errno = database->smdb_del(database, &db_key, 0);
37740266059SGregory Neil Shapiro 
37840266059SGregory Neil Shapiro 		switch (errno)
37940266059SGregory Neil Shapiro 		{
38040266059SGregory Neil Shapiro 		case SMDBE_NOT_FOUND:
38140266059SGregory Neil Shapiro 			fprintf(stderr,
38240266059SGregory Neil Shapiro 				"%s: key %s doesn't exist in map %s\n",
38340266059SGregory Neil Shapiro 				progname, keyname, mapname);
38440266059SGregory Neil Shapiro 			/* Don't set exitstat */
38540266059SGregory Neil Shapiro 			break;
38640266059SGregory Neil Shapiro 		case SMDBE_OK:
38740266059SGregory Neil Shapiro 			/* All's well */
38840266059SGregory Neil Shapiro 			break;
38940266059SGregory Neil Shapiro 		default:
39040266059SGregory Neil Shapiro 			fprintf(stderr,
39140266059SGregory Neil Shapiro 				"%s: couldn't remove key %s in map %s (error)\n",
39240266059SGregory Neil Shapiro 				progname, keyname, mapname);
39340266059SGregory Neil Shapiro 			exitstat = EX_IOERR;
39440266059SGregory Neil Shapiro 			break;
39540266059SGregory Neil Shapiro 		}
39640266059SGregory Neil Shapiro 	}
39740266059SGregory Neil Shapiro 	else
39840266059SGregory Neil Shapiro 	{
39940266059SGregory Neil Shapiro 		assert(0);  /* NOT REACHED */
40040266059SGregory Neil Shapiro 	}
40140266059SGregory Neil Shapiro 
40240266059SGregory Neil Shapiro 	/*
40340266059SGregory Neil Shapiro 	**  Now close the database.
40440266059SGregory Neil Shapiro 	*/
40540266059SGregory Neil Shapiro 
40640266059SGregory Neil Shapiro 	errno = database->smdb_close(database);
40740266059SGregory Neil Shapiro 	if (errno != SMDBE_OK)
40840266059SGregory Neil Shapiro 	{
40940266059SGregory Neil Shapiro 		fprintf(stderr, "%s: close(%s): %s\n",
41040266059SGregory Neil Shapiro 			progname, mapname, sm_errstring(errno));
41140266059SGregory Neil Shapiro 		exitstat = EX_IOERR;
41240266059SGregory Neil Shapiro 	}
41340266059SGregory Neil Shapiro 	smdb_free_database(database);
41440266059SGregory Neil Shapiro 
41540266059SGregory Neil Shapiro 	exit(exitstat);
41640266059SGregory Neil Shapiro 	/* NOTREACHED */
41740266059SGregory Neil Shapiro 	return exitstat;
41840266059SGregory Neil Shapiro }
419