/* * Copyright 2001-2004 Brandon Long * All Rights Reserved. * * ClearSilver Templating System * * This code is made available under the terms of the ClearSilver License. * http://www.clearsilver.net/license.hdf * */ /* Initial version based on multi-proc based server (like apache 1.x) * * Parts are: * 1) server Init * 2) sub-proc start * 3) sub-proc init * 4) sub-proc process request * 5) sub-proc cleanup * 6) server cleanup * * Parts 1 & 6 aren't part of the framework, and at this point, I don't * think I need to worry about 3 & 5 either, but maybe in the future. */ #include "cs_config.h" #include <unistd.h> #include <sys/types.h> #include <sys/wait.h> #include <limits.h> #include <errno.h> #include <string.h> #include <signal.h> #include "neo_misc.h" #include "neo_err.h" #include "neo_net.h" #include "ulocks.h" #include "neo_server.h" static NEOERR *nserver_child_loop(NSERVER *server, int num) { NEOERR *err = STATUS_OK, *clean_err; int loop = 0; NSOCK *child_sock; if (server->init_cb) { err = server->init_cb(server->data, num); if (err) return nerr_pass(err); } while (loop++ < server->num_requests) { err = fLock(server->accept_lock); if (err) break; err = ne_net_accept(&child_sock, server->server_fd, server->data_timeout); fUnlock(server->accept_lock); if (err) break; err = server->req_cb(server->data, num, child_sock); if (err) { ne_net_close(&child_sock); } else { err = ne_net_close(&child_sock); } nerr_log_error(err); nerr_ignore(&err); } ne_warn("nserver child loop handled %d connections", loop-1); if (server->clean_cb) { clean_err = server->clean_cb(server->data, num); if (clean_err) { nerr_log_error(clean_err); nerr_ignore(&clean_err); } } return nerr_pass(err); } static void ignore_pipe(void) { struct sigaction sa; memset(&sa, 0, sizeof(struct sigaction)); sa.sa_handler = SIG_IGN; sigemptyset(&sa.sa_mask); sa.sa_flags = SA_RESTART; sigaction(SIGPIPE, &sa, NULL); } /* Handle shutdown by accepting a TERM signal and then passing it to our * program group */ static int ShutdownPending = 0; static void sig_term(int sig) { ShutdownPending = 1; ne_net_shutdown(); } static void setup_term(void) { struct sigaction sa; memset(&sa, 0, sizeof(struct sigaction)); sa.sa_handler = sig_term; sigemptyset(&sa.sa_mask); sa.sa_flags = 0; sigaction(SIGTERM, &sa, NULL); } NEOERR *nserver_proc_start(NSERVER *server, BOOL debug) { NEOERR *err; if (server->req_cb == NULL) return nerr_raise(NERR_ASSERT, "nserver requires a request callback"); ignore_pipe(); setup_term(); ShutdownPending = 0; err = fFind(&(server->accept_lock), server->lockfile); if (err && nerr_handle(&err, NERR_NOT_FOUND)) { err = fCreate(&(server->accept_lock), server->lockfile); } if (err) return nerr_pass(err); do { err = ne_net_listen(server->port, &(server->server_fd)); if (err) break; if (debug == TRUE) { err = nserver_child_loop(server, 0); break; } else { /* create children and restart them as necessary */ pid_t child; int count, status; for (count = 0; count < server->num_children; count++) { child = fork(); if (child == -1) { err = nerr_raise_errno(NERR_SYSTEM, "Unable to fork child"); break; } if (!child) { err = nserver_child_loop(server, count); if (err) exit(-1); exit(0); } ne_warn("Starting child pid %d", child); } if (count < server->num_children) break; while (!ShutdownPending) { child = wait3(&status, 0, NULL); if (child == -1) { ne_warn("wait3 failed [%d] %s", errno, strerror(errno)); continue; } if (WIFSTOPPED(status)) { ne_warn("pid %d stopped on signal %d", child, WSTOPSIG(status)); continue; } if (WIFEXITED(status)) { /* at some point, we might do something here with the * particular exit value */ ne_warn("pid %d exited, returned %d", child, WEXITSTATUS(status)); } else if (WIFSIGNALED(status)) { ne_warn("pid %d exited on signal %d", child, WTERMSIG(status)); } count++; child = fork(); if (child == -1) { err = nerr_raise_errno(NERR_SYSTEM, "Unable to fork child"); break; } if (!child) { err = nserver_child_loop(server, count); if (err) exit(-1); exit(0); } ne_warn("Starting child pid %d", child); } /* At some point, we might want to actually maintain information * on our children, and then we can be more specific here in terms * of making sure they all shutdown... for now, fergitaboutit */ if (ShutdownPending) { killpg(0, SIGTERM); } } } while (0); fDestroy(server->accept_lock); return nerr_pass(err); }