#if !defined(lint) && !defined(LINT)
static const char rcsid[] =
"$Id: do_command.c,v 1.3 1998/08/14 00:32:39 vixie Exp $";
#endif
#include "cron.h"
#if defined(LOGIN_CAP)
# include <login_cap.h>
#endif
#ifdef PAM
# include <security/pam_appl.h>
# include <security/openpam.h>
#endif
static void child_process(entry *, user *);
static WAIT_T wait_on_child(PID_T, const char *);
extern char *environ;
void
do_command(entry *e, user *u)
{
pid_t pid;
Debug(DPROC, ("[%d] do_command(%s, (%s,%d,%d))\n",
getpid(), e->cmd, u->name, e->uid, e->gid))
switch ((pid = fork())) {
case -1:
log_it("CRON", getpid(), "error", "can't fork");
if (e->flags & INTERVAL)
e->lastexit = time(NULL);
break;
case 0:
pidfile_close(pfh);
child_process(e, u);
Debug(DPROC, ("[%d] child process done, exiting\n", getpid()))
_exit(OK_EXIT);
break;
default:
Debug(DPROC, ("[%d] main process forked child #%d, "
"returning to work\n", getpid(), pid))
if (e->flags & INTERVAL) {
e->lastexit = 0;
e->child = pid;
}
break;
}
Debug(DPROC, ("[%d] main process returning to work\n", getpid()))
}
#ifdef PAM
static void
pam_cleanup(pam_handle_t **pamhp, int *session_opened, int *cred_established,
char ***pam_envp, const char *usernm, pid_t pid, int log_errors,
int end_status)
{
int pam_err;
if (*pamhp == NULL)
return;
if (*session_opened) {
pam_err = pam_close_session(*pamhp, PAM_SILENT);
if (log_errors && pam_err != PAM_SUCCESS) {
log_it(usernm, pid, "SESSION-CLOSE",
pam_strerror(*pamhp, pam_err));
}
*session_opened = 0;
}
if (*cred_established) {
pam_err = pam_setcred(*pamhp, PAM_DELETE_CRED);
if (log_errors && pam_err != PAM_SUCCESS) {
log_it(usernm, pid, "CRED-DELETE",
pam_strerror(*pamhp, pam_err));
}
*cred_established = 0;
}
if (*pam_envp != NULL) {
openpam_free_envlist(*pam_envp);
*pam_envp = NULL;
}
pam_end(*pamhp, end_status);
*pamhp = NULL;
}
#endif
static void
child_process(entry *e, user *u)
{
int stdin_pipe[2], stdout_pipe[2];
char *input_data;
const char *usernm, *mailto, *mailfrom, *mailcc, *mailbcc;
PID_T jobpid, stdinjob, mailpid;
FILE *mail;
int bytes = 1;
int status = 0;
const char *homedir = NULL;
#ifdef PAM
pam_handle_t *pamh = NULL;
int pam_err = PAM_SUCCESS;
int pam_session_opened = 0;
int pam_cred_established = 0;
char **pam_envp = NULL;
#endif
# if defined(LOGIN_CAP)
struct passwd *pwd;
login_cap_t *lc;
# endif
Debug(DPROC, ("[%d] child_process('%s')\n", getpid(), e->cmd))
setproctitle("running job");
usernm = env_get("LOGNAME", e->envp);
mailto = env_get("MAILTO", e->envp);
mailcc = env_get("MAILCC", e->envp);
mailbcc = env_get("MAILBCC", e->envp);
mailfrom = env_get("MAILFROM", e->envp);
#ifdef PAM
if (strcmp(u->name, SYS_NAME)) {
struct pam_conv pamc = {
.conv = openpam_nullconv,
.appdata_ptr = NULL
};
Debug(DPROC, ("[%d] checking account with PAM\n", getpid()))
if (strcmp(u->name, usernm)) {
log_it(usernm, getpid(), "username ambiguity", u->name);
exit(ERROR_EXIT);
}
pam_err = pam_start("cron", usernm, &pamc, &pamh);
if (pam_err != PAM_SUCCESS) {
log_it("CRON", getpid(), "error", "can't start PAM");
exit(ERROR_EXIT);
}
pam_err = pam_set_item(pamh, PAM_TTY, "cron");
if (pam_err != PAM_SUCCESS) {
log_it("CRON", getpid(), "error", "can't set PAM_TTY");
pam_cleanup(&pamh, &pam_session_opened,
&pam_cred_established, &pam_envp, usernm,
getpid(), 0, pam_err);
exit(ERROR_EXIT);
}
pam_err = pam_acct_mgmt(pamh, PAM_SILENT);
if (pam_err != PAM_SUCCESS && pam_err != PAM_NEW_AUTHTOK_REQD) {
log_it(usernm, getpid(), "USER", "account unavailable");
pam_cleanup(&pamh, &pam_session_opened,
&pam_cred_established, &pam_envp, usernm,
getpid(), 0, pam_err);
exit(ERROR_EXIT);
}
pam_err = pam_setcred(pamh, PAM_ESTABLISH_CRED);
if (pam_err != PAM_SUCCESS) {
log_it(usernm, getpid(), "CRED",
pam_strerror(pamh, pam_err));
pam_cleanup(&pamh, &pam_session_opened,
&pam_cred_established, &pam_envp, usernm,
getpid(), 0, pam_err);
exit(ERROR_EXIT);
}
pam_cred_established = 1;
pam_err = pam_open_session(pamh, PAM_SILENT);
if (pam_err != PAM_SUCCESS) {
log_it(usernm, getpid(), "SESSION",
pam_strerror(pamh, pam_err));
pam_cleanup(&pamh, &pam_session_opened,
&pam_cred_established, &pam_envp, usernm,
getpid(), 0, pam_err);
exit(ERROR_EXIT);
}
pam_session_opened = 1;
pam_envp = pam_getenvlist(pamh);
}
#endif
(void) signal(SIGCHLD, SIG_DFL);
if (pipe(stdin_pipe) != 0 || pipe(stdout_pipe) != 0) {
log_it("CRON", getpid(), "error", "can't pipe");
#ifdef PAM
if (pamh != NULL && strcmp(u->name, SYS_NAME)) {
pam_cleanup(&pamh, &pam_session_opened,
&pam_cred_established, &pam_envp, usernm,
getpid(), 1, pam_err);
}
#endif
exit(ERROR_EXIT);
}
{
int escaped = FALSE;
int ch;
char *p;
for (input_data = p = e->cmd;
(ch = *input_data) != '\0';
input_data++, p++) {
if (p != input_data)
*p = ch;
if (escaped) {
if (ch == '%' || ch == '\\')
*--p = ch;
escaped = FALSE;
continue;
}
if (ch == '\\') {
escaped = TRUE;
continue;
}
if (ch == '%') {
*input_data++ = '\0';
break;
}
}
*p = '\0';
}
switch (jobpid = fork()) {
case -1:
log_it("CRON", getpid(), "error", "can't fork");
#ifdef PAM
if (pamh != NULL && strcmp(u->name, SYS_NAME)) {
pam_cleanup(&pamh, &pam_session_opened,
&pam_cred_established, &pam_envp, usernm,
getpid(), 1, pam_err);
}
#endif
exit(ERROR_EXIT);
case 0:
Debug(DPROC, ("[%d] grandchild process fork()'ed\n",
getpid()))
#ifdef PAM
pamh = NULL;
#endif
if (e->uid == ROOT_UID)
Jitter = RootJitter;
if (Jitter != 0) {
srandom(getpid());
sleep(random() % Jitter);
}
if ((e->flags & DONT_LOG) == 0) {
char *x = mkprints((u_char *)e->cmd, strlen(e->cmd));
log_it(usernm, getpid(), "CMD", x);
free(x);
}
#ifdef SYSLOG
closelog();
#endif
(void) setsid();
close(stdin_pipe[WRITE_PIPE]);
close(stdout_pipe[READ_PIPE]);
close(STDIN); dup2(stdin_pipe[READ_PIPE], STDIN);
close(STDOUT); dup2(stdout_pipe[WRITE_PIPE], STDOUT);
close(STDERR); dup2(STDOUT, STDERR);
close(stdin_pipe[READ_PIPE]);
close(stdout_pipe[WRITE_PIPE]);
environ = NULL;
# if defined(LOGIN_CAP)
if ((pwd = getpwnam(usernm)) == NULL)
pwd = getpwuid(e->uid);
lc = NULL;
if (pwd != NULL) {
if (pwd->pw_dir != NULL
&& pwd->pw_dir[0] != '\0') {
homedir = strdup(pwd->pw_dir);
if (homedir == NULL) {
warn("strdup");
_exit(ERROR_EXIT);
}
}
pwd->pw_gid = e->gid;
if (e->class != NULL)
lc = login_getclass(e->class);
}
if (pwd &&
setusercontext(lc, pwd, e->uid,
LOGIN_SETALL) == 0)
(void) endpwent();
else {
(void) endpwent();
# endif
if (setgid(e->gid) != 0) {
log_it(usernm, getpid(),
"error", "setgid failed");
_exit(ERROR_EXIT);
}
if (initgroups(usernm, e->gid) != 0) {
log_it(usernm, getpid(),
"error", "initgroups failed");
_exit(ERROR_EXIT);
}
if (setlogin(usernm) != 0) {
log_it(usernm, getpid(),
"error", "setlogin failed");
_exit(ERROR_EXIT);
}
if (setuid(e->uid) != 0) {
log_it(usernm, getpid(),
"error", "setuid failed");
_exit(ERROR_EXIT);
}
#if defined(LOGIN_CAP)
}
if (lc != NULL)
login_close(lc);
#endif
{
char *new_home = env_get("HOME", e->envp);
if (new_home != NULL && new_home[0] != '\0')
chdir(new_home);
else if (homedir != NULL)
chdir(homedir);
else
chdir("/");
}
{
char *shell = env_get("SHELL", e->envp);
char **p;
#ifdef PAM
if (pam_envp != NULL) {
char **pp;
for (pp = pam_envp; *pp != NULL; pp++) {
if (putenv(*pp) != 0) {
warn("putenv");
_exit(ERROR_EXIT);
}
}
free(pam_envp);
pam_envp = NULL;
}
#endif
for (p = e->envp; *p; ++p) {
if (putenv(*p) != 0) {
warn("putenv");
_exit(ERROR_EXIT);
}
}
if (homedir != NULL
&& setenv("HOME", homedir, 0) < 0) {
warn("setenv(HOME)");
_exit(ERROR_EXIT);
}
if (setenv("PATH", _PATH_DEFPATH, 0) < 0) {
warn("setenv(PATH)");
_exit(ERROR_EXIT);
}
# if DEBUGGING
if (DebugFlags & DTEST) {
fprintf(stderr,
"debug DTEST is on, not exec'ing command.\n");
fprintf(stderr,
"\tcmd='%s' shell='%s'\n", e->cmd, shell);
_exit(OK_EXIT);
}
# endif
execl(shell, shell, "-c", e->cmd, (char *)NULL);
warn("execl: couldn't exec `%s'", shell);
_exit(ERROR_EXIT);
}
break;
default:
break;
}
#ifdef PAM
if (jobpid > 0 && pam_envp != NULL) {
openpam_free_envlist(pam_envp);
pam_envp = NULL;
}
#endif
Debug(DPROC, ("[%d] child continues, closing pipes\n", getpid()))
close(stdin_pipe[READ_PIPE]);
close(stdout_pipe[WRITE_PIPE]);
if (*input_data && (stdinjob = fork()) == 0) {
FILE *out = fdopen(stdin_pipe[WRITE_PIPE], "w");
int need_newline = FALSE;
int escaped = FALSE;
int ch;
if (out == NULL) {
warn("fdopen failed in child2");
_exit(ERROR_EXIT);
}
Debug(DPROC, ("[%d] child2 sending data to grandchild\n", getpid()))
close(stdout_pipe[READ_PIPE]);
while ((ch = *input_data++) != '\0') {
if (escaped) {
if (ch != '%')
putc('\\', out);
} else {
if (ch == '%')
ch = '\n';
}
if (!(escaped = (ch == '\\'))) {
putc(ch, out);
need_newline = (ch != '\n');
}
}
if (escaped)
putc('\\', out);
if (need_newline)
putc('\n', out);
fclose(out);
Debug(DPROC, ("[%d] child2 done sending to grandchild\n", getpid()))
exit(0);
}
close(stdin_pipe[WRITE_PIPE]);
Debug(DPROC, ("[%d] child reading output from grandchild\n", getpid()))
{
FILE *in = fdopen(stdout_pipe[READ_PIPE], "r");
int ch;
if (in == NULL) {
warn("fdopen failed in child");
_exit(ERROR_EXIT);
}
mail = NULL;
ch = getc(in);
if (ch != EOF) {
Debug(DPROC|DEXT,
("[%d] got data (%x:%c) from grandchild\n",
getpid(), ch, ch))
if (mailto == NULL) {
if (defmailto)
mailto = defmailto;
else
mailto = usernm;
}
if (mailto && *mailto == '\0')
mailto = NULL;
if (mailto) {
char **env;
char mailcmd[MAX_COMMAND];
char hostname[MAXHOSTNAMELEN];
if (gethostname(hostname, MAXHOSTNAMELEN) == -1)
hostname[0] = '\0';
hostname[sizeof(hostname) - 1] = '\0';
if (snprintf(mailcmd, sizeof(mailcmd), MAILFMT,
MAILARG) >= sizeof(mailcmd)) {
warnx("mail command too long");
(void) _exit(ERROR_EXIT);
}
if (!(mail = cron_popen(mailcmd, "w", e, &mailpid))) {
warn("%s", mailcmd);
(void) _exit(ERROR_EXIT);
}
if (mailfrom == NULL || *mailfrom == '\0')
fprintf(mail, "From: Cron Daemon <%s@%s>\n",
usernm, hostname);
else
fprintf(mail, "From: Cron Daemon <%s>\n",
mailfrom);
fprintf(mail, "To: %s\n", mailto);
fprintf(mail, "CC: %s\n", mailcc);
fprintf(mail, "BCC: %s\n", mailbcc);
fprintf(mail, "Subject: Cron <%s@%s> %s\n",
usernm, first_word(hostname, "."),
e->cmd);
#ifdef MAIL_DATE
fprintf(mail, "Date: %s\n",
arpadate(&TargetTime));
#endif
for (env = e->envp; *env; env++)
fprintf(mail, "X-Cron-Env: <%s>\n",
*env);
fprintf(mail, "\n");
putc(ch, mail);
}
while (EOF != (ch = getc(in))) {
bytes++;
if (mail)
putc(ch, mail);
}
}
Debug(DPROC, ("[%d] got EOF from grandchild\n", getpid()))
fclose(in);
}
if (jobpid > 0) {
WAIT_T waiter;
waiter = wait_on_child(jobpid, "grandchild command job");
if (WIFEXITED(waiter) && WEXITSTATUS(waiter) == 0
&& (e->flags & MAIL_WHEN_ERR) == MAIL_WHEN_ERR
&& mail) {
Debug(DPROC, ("[%d] %s executed successfully, mail suppressed\n",
getpid(), "grandchild command job"))
kill(mailpid, SIGKILL);
(void)fclose(mail);
mail = NULL;
}
if (mail) {
Debug(DPROC, ("[%d] closing pipe to mail\n",
getpid()))
status = cron_pclose(mail);
if (status) {
char buf[MAX_TEMPSTR];
snprintf(buf, sizeof(buf),
"mailed %d byte%s of output but got status 0x%04x\n",
bytes, (bytes==1)?"":"s",
status);
log_it(usernm, getpid(), "MAIL", buf);
}
}
}
if (*input_data && stdinjob > 0)
wait_on_child(stdinjob, "grandchild stdinjob");
#ifdef PAM
if (pamh != NULL && strcmp(u->name, SYS_NAME)) {
pam_cleanup(&pamh, &pam_session_opened, &pam_cred_established,
&pam_envp, usernm, getpid(), 1, PAM_SUCCESS);
}
#endif
}
static WAIT_T
wait_on_child(PID_T childpid, const char *name)
{
WAIT_T waiter;
PID_T pid;
Debug(DPROC, ("[%d] waiting for %s (%d) to finish\n",
getpid(), name, childpid))
#ifdef POSIX
while ((pid = waitpid(childpid, &waiter, 0)) < 0 && errno == EINTR)
#else
while ((pid = wait4(childpid, &waiter, 0, NULL)) < 0 && errno == EINTR)
#endif
;
if (pid < OK)
return waiter;
Debug(DPROC, ("[%d] %s (%d) finished, status=%04x",
getpid(), name, pid, WEXITSTATUS(waiter)))
if (WIFSIGNALED(waiter) && WCOREDUMP(waiter))
Debug(DPROC, (", dumped core"))
Debug(DPROC, ("\n"))
return waiter;
}