#include <sys/types.h>
#include <bitstring.h>
#include <ctype.h>
#include <errno.h>
#include <pwd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
#include <time.h>
#include "macros.h"
#include "structs.h"
#include "funcs.h"
#include "globals.h"
void
free_user(user *u)
{
entry *e;
while ((e = SLIST_FIRST(&u->crontab))) {
SLIST_REMOVE_HEAD(&u->crontab, entries);
job_remove(e, u);
free_entry(e);
}
free(u->name);
free(u);
}
static int ParseErrorCount;
static const char *CrontabFilename;
static void
parse_error(const char *msg)
{
ParseErrorCount++;
syslog(LOG_ERR, "(CRON) %s:%d (%s)", CrontabFilename, LineNumber, msg);
}
user *
load_user(FILE *file, struct passwd *pw, const char *name)
{
char envstr[MAX_ENVSTR];
user *u;
entry *e;
int status, save_errno;
char **envp = NULL, **tenvp;
CrontabFilename = name;
LineNumber = 0;
if ((u = malloc(sizeof(user))) == NULL)
goto done;
if ((u->name = strdup(name)) == NULL) {
save_errno = errno;
free(u);
u = NULL;
errno = save_errno;
goto done;
}
SLIST_INIT(&u->crontab);
if ((envp = env_init()) == NULL) {
save_errno = errno;
free_user(u);
u = NULL;
errno = save_errno;
goto done;
}
ParseErrorCount = 0;
while ((status = load_env(envstr, file)) >= 0) {
switch (status) {
case FALSE:
e = load_entry(file, parse_error, pw, envp);
if (e == NULL) {
if (pw != NULL) {
save_errno = errno;
free_user(u);
u = NULL;
errno = save_errno;
goto done;
}
} else {
SLIST_INSERT_HEAD(&u->crontab, e, entries);
}
break;
case TRUE:
if ((tenvp = env_set(envp, envstr)) == NULL) {
save_errno = errno;
free_user(u);
u = NULL;
errno = save_errno;
goto done;
}
envp = tenvp;
break;
}
}
done:
if (envp != NULL)
env_free(envp);
return (u);
}