xref: /dflybsd-src/usr.sbin/cron/lib/env.c (revision 86d7f5d305c6adaa56ff4582ece9859d73106103)
186d7f5d3SJohn Marino /* Copyright 1988,1990,1993,1994 by Paul Vixie
286d7f5d3SJohn Marino  * All rights reserved
386d7f5d3SJohn Marino  *
486d7f5d3SJohn Marino  * Distribute freely, except: don't remove my name from the source or
586d7f5d3SJohn Marino  * documentation (don't take credit for my work), mark your changes (don't
686d7f5d3SJohn Marino  * get me blamed for your possible bugs), don't alter or remove this
786d7f5d3SJohn Marino  * notice.  May be sold if buildable source is provided to buyer.  No
886d7f5d3SJohn Marino  * warrantee of any kind, express or implied, is included with this
986d7f5d3SJohn Marino  * software; use at your own risk, responsibility for damages (if any) to
1086d7f5d3SJohn Marino  * anyone resulting from the use of this software rests entirely with the
1186d7f5d3SJohn Marino  * user.
1286d7f5d3SJohn Marino  *
1386d7f5d3SJohn Marino  * Send bug reports, bug fixes, enhancements, requests, flames, etc., and
1486d7f5d3SJohn Marino  * I'll try to keep a version up to date.  I can be reached as follows:
1586d7f5d3SJohn Marino  * Paul Vixie          <paul@vix.com>          uunet!decwrl!vixie!paul
1686d7f5d3SJohn Marino  *
1786d7f5d3SJohn Marino  * $FreeBSD: src/usr.sbin/cron/lib/env.c,v 1.7.2.1 2000/07/01 10:35:07 ps Exp $
1886d7f5d3SJohn Marino  * $DragonFly: src/usr.sbin/cron/lib/env.c,v 1.5 2004/12/18 22:48:03 swildner Exp $
1986d7f5d3SJohn Marino  */
2086d7f5d3SJohn Marino 
2186d7f5d3SJohn Marino #include "cron.h"
2286d7f5d3SJohn Marino 
2386d7f5d3SJohn Marino 
2486d7f5d3SJohn Marino char **
env_init(void)2586d7f5d3SJohn Marino env_init(void)
2686d7f5d3SJohn Marino {
2786d7f5d3SJohn Marino 	char **p;
2886d7f5d3SJohn Marino 
2986d7f5d3SJohn Marino 	p = malloc(sizeof(char *));
3086d7f5d3SJohn Marino 	if (p)
3186d7f5d3SJohn Marino 		p[0] = NULL;
3286d7f5d3SJohn Marino 	return (p);
3386d7f5d3SJohn Marino }
3486d7f5d3SJohn Marino 
3586d7f5d3SJohn Marino 
3686d7f5d3SJohn Marino void
env_free(char ** envp)3786d7f5d3SJohn Marino env_free(char **envp)
3886d7f5d3SJohn Marino {
3986d7f5d3SJohn Marino 	char	**p;
4086d7f5d3SJohn Marino 
4186d7f5d3SJohn Marino 	for (p = envp;  *p;  p++)
4286d7f5d3SJohn Marino 		free(*p);
4386d7f5d3SJohn Marino 	free(envp);
4486d7f5d3SJohn Marino }
4586d7f5d3SJohn Marino 
4686d7f5d3SJohn Marino 
4786d7f5d3SJohn Marino char **
env_copy(char ** envp)4886d7f5d3SJohn Marino env_copy(char **envp)
4986d7f5d3SJohn Marino {
5086d7f5d3SJohn Marino 	int count, i;
5186d7f5d3SJohn Marino 	char **p;
5286d7f5d3SJohn Marino 
5386d7f5d3SJohn Marino 	for (count = 0;  envp[count] != NULL;  count++)
5486d7f5d3SJohn Marino 		;
5586d7f5d3SJohn Marino 	p = (char **) malloc((count+1) * sizeof(char *)); /* 1 for the NULL */
5686d7f5d3SJohn Marino 	if (p == NULL) {
5786d7f5d3SJohn Marino 		errno = ENOMEM;
5886d7f5d3SJohn Marino 		return NULL;
5986d7f5d3SJohn Marino 	}
6086d7f5d3SJohn Marino 	for (i = 0;  i < count;  i++)
6186d7f5d3SJohn Marino 		if ((p[i] = strdup(envp[i])) == NULL) {
6286d7f5d3SJohn Marino 			while (--i >= 0)
6386d7f5d3SJohn Marino 				free(p[i]);
6486d7f5d3SJohn Marino 			free(p);
6586d7f5d3SJohn Marino 			errno = ENOMEM;
6686d7f5d3SJohn Marino 			return NULL;
6786d7f5d3SJohn Marino 		}
6886d7f5d3SJohn Marino 	p[count] = NULL;
6986d7f5d3SJohn Marino 	return (p);
7086d7f5d3SJohn Marino }
7186d7f5d3SJohn Marino 
7286d7f5d3SJohn Marino 
7386d7f5d3SJohn Marino char **
env_set(char ** envp,char * envstr)7486d7f5d3SJohn Marino env_set(char **envp, char *envstr)
7586d7f5d3SJohn Marino {
7686d7f5d3SJohn Marino 	int count, found;
7786d7f5d3SJohn Marino 	char **p;
7886d7f5d3SJohn Marino 	char *q;
7986d7f5d3SJohn Marino 
8086d7f5d3SJohn Marino 	/*
8186d7f5d3SJohn Marino 	 * count the number of elements, including the null pointer;
8286d7f5d3SJohn Marino 	 * also set 'found' to -1 or index of entry if already in here.
8386d7f5d3SJohn Marino 	 */
8486d7f5d3SJohn Marino 	found = -1;
8586d7f5d3SJohn Marino 	for (count = 0;  envp[count] != NULL;  count++) {
8686d7f5d3SJohn Marino 		if (!strcmp_until(envp[count], envstr, '='))
8786d7f5d3SJohn Marino 			found = count;
8886d7f5d3SJohn Marino 	}
8986d7f5d3SJohn Marino 	count++;	/* for the NULL */
9086d7f5d3SJohn Marino 
9186d7f5d3SJohn Marino 	if (found != -1) {
9286d7f5d3SJohn Marino 		/*
9386d7f5d3SJohn Marino 		 * it exists already, so just free the existing setting,
9486d7f5d3SJohn Marino 		 * save our new one there, and return the existing array.
9586d7f5d3SJohn Marino 		 */
9686d7f5d3SJohn Marino 		q = envp[found];
9786d7f5d3SJohn Marino 		if ((envp[found] = strdup(envstr)) == NULL) {
9886d7f5d3SJohn Marino 			envp[found] = q;
9986d7f5d3SJohn Marino 			/* XXX env_free(envp); */
10086d7f5d3SJohn Marino 			errno = ENOMEM;
10186d7f5d3SJohn Marino 			return NULL;
10286d7f5d3SJohn Marino 		}
10386d7f5d3SJohn Marino 		free(q);
10486d7f5d3SJohn Marino 		return (envp);
10586d7f5d3SJohn Marino 	}
10686d7f5d3SJohn Marino 
10786d7f5d3SJohn Marino 	/*
10886d7f5d3SJohn Marino 	 * it doesn't exist yet, so resize the array, move null pointer over
10986d7f5d3SJohn Marino 	 * one, save our string over the old null pointer, and return resized
11086d7f5d3SJohn Marino 	 * array.
11186d7f5d3SJohn Marino 	 */
11286d7f5d3SJohn Marino 	p = (char **) realloc((void *) envp,
11386d7f5d3SJohn Marino 			      (unsigned) ((count+1) * sizeof(char *)));
11486d7f5d3SJohn Marino 	if (p == NULL) 	{
11586d7f5d3SJohn Marino 		/* XXX env_free(envp); */
11686d7f5d3SJohn Marino 		errno = ENOMEM;
11786d7f5d3SJohn Marino 		return NULL;
11886d7f5d3SJohn Marino 	}
11986d7f5d3SJohn Marino 	p[count] = p[count-1];
12086d7f5d3SJohn Marino 	if ((p[count-1] = strdup(envstr)) == NULL) {
12186d7f5d3SJohn Marino 		env_free(p);
12286d7f5d3SJohn Marino 		errno = ENOMEM;
12386d7f5d3SJohn Marino 		return NULL;
12486d7f5d3SJohn Marino 	}
12586d7f5d3SJohn Marino 	return (p);
12686d7f5d3SJohn Marino }
12786d7f5d3SJohn Marino 
12886d7f5d3SJohn Marino 
12986d7f5d3SJohn Marino /* return	ERR = end of file
13086d7f5d3SJohn Marino  *		FALSE = not an env setting (file was repositioned)
13186d7f5d3SJohn Marino  *		TRUE = was an env setting
13286d7f5d3SJohn Marino  */
13386d7f5d3SJohn Marino int
load_env(char * envstr,FILE * f)13486d7f5d3SJohn Marino load_env(char *envstr, FILE *f)
13586d7f5d3SJohn Marino {
13686d7f5d3SJohn Marino 	long	filepos;
13786d7f5d3SJohn Marino 	int	fileline;
13886d7f5d3SJohn Marino 	char	name[MAX_ENVSTR], val[MAX_ENVSTR];
13986d7f5d3SJohn Marino 	int	fields;
14086d7f5d3SJohn Marino 
14186d7f5d3SJohn Marino 	filepos = ftell(f);
14286d7f5d3SJohn Marino 	fileline = LineNumber;
14386d7f5d3SJohn Marino 	skip_comments(f);
14486d7f5d3SJohn Marino 	if (EOF == get_string(envstr, MAX_ENVSTR, f, "\n"))
14586d7f5d3SJohn Marino 		return (ERR);
14686d7f5d3SJohn Marino 
14786d7f5d3SJohn Marino 	Debug(DPARS, ("load_env, read <%s>\n", envstr))
14886d7f5d3SJohn Marino 
14986d7f5d3SJohn Marino 	name[0] = val[0] = '\0';
15086d7f5d3SJohn Marino 	fields = sscanf(envstr, "%[^ =] = %[^\n#]", name, val);
15186d7f5d3SJohn Marino 	if (fields != 2) {
15286d7f5d3SJohn Marino 		Debug(DPARS, ("load_env, not 2 fields (%d)\n", fields))
15386d7f5d3SJohn Marino 		fseek(f, filepos, 0);
15486d7f5d3SJohn Marino 		Set_LineNum(fileline);
15586d7f5d3SJohn Marino 		return (FALSE);
15686d7f5d3SJohn Marino 	}
15786d7f5d3SJohn Marino 
15886d7f5d3SJohn Marino 	/* 2 fields from scanf; looks like an env setting
15986d7f5d3SJohn Marino 	 */
16086d7f5d3SJohn Marino 
16186d7f5d3SJohn Marino 	/*
16286d7f5d3SJohn Marino 	 * process value string
16386d7f5d3SJohn Marino 	 */
16486d7f5d3SJohn Marino 	/*local*/{
16586d7f5d3SJohn Marino 		int	len = strdtb(val);
16686d7f5d3SJohn Marino 
16786d7f5d3SJohn Marino 		if (len >= 2) {
16886d7f5d3SJohn Marino 			if (val[0] == '\'' || val[0] == '"') {
16986d7f5d3SJohn Marino 				if (val[len-1] == val[0]) {
17086d7f5d3SJohn Marino 					val[len-1] = '\0';
17186d7f5d3SJohn Marino 					strcpy(val, val+1);
17286d7f5d3SJohn Marino 				}
17386d7f5d3SJohn Marino 			}
17486d7f5d3SJohn Marino 		}
17586d7f5d3SJohn Marino 	}
17686d7f5d3SJohn Marino 
17786d7f5d3SJohn Marino 	if (strlen(name) + 1 + strlen(val) >= MAX_ENVSTR-1)
17886d7f5d3SJohn Marino 		return (FALSE);
17986d7f5d3SJohn Marino 	sprintf(envstr, "%s=%s", name, val);
18086d7f5d3SJohn Marino 	Debug(DPARS, ("load_env, <%s> <%s> -> <%s>\n", name, val, envstr))
18186d7f5d3SJohn Marino 	return (TRUE);
18286d7f5d3SJohn Marino }
18386d7f5d3SJohn Marino 
18486d7f5d3SJohn Marino 
18586d7f5d3SJohn Marino char *
env_get(char * name,char ** envp)18686d7f5d3SJohn Marino env_get(char *name, char **envp)
18786d7f5d3SJohn Marino {
18886d7f5d3SJohn Marino 	int len;
18986d7f5d3SJohn Marino 	char *p, *q;
19086d7f5d3SJohn Marino 
19186d7f5d3SJohn Marino 	len = strlen(name);
19286d7f5d3SJohn Marino 	while ((p = *envp++)) {
19386d7f5d3SJohn Marino 		if (!(q = strchr(p, '=')))
19486d7f5d3SJohn Marino 			continue;
19586d7f5d3SJohn Marino 		if ((q - p) == len && !strncmp(p, name, len))
19686d7f5d3SJohn Marino 			return (q+1);
19786d7f5d3SJohn Marino 	}
19886d7f5d3SJohn Marino 	return (NULL);
19986d7f5d3SJohn Marino }
200