X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~ian/git?p=secnet.git;a=blobdiff_plain;f=secnet.c;h=a21e52b39753c68c3fa562203c429b218088ebc3;hp=b0bb4b3d6a477b99af53376f4f7cfc951bffea9a;hb=4f5e39ecfaa49376b0a5c3a4c384e91a828c1105;hpb=4efd681a66c15bc6f81eefc69396669e165e5e0f diff --git a/secnet.c b/secnet.c index b0bb4b3..a21e52b 100644 --- a/secnet.c +++ b/secnet.c @@ -1,36 +1,39 @@ -/* $Log: secnet.c,v $ - * Revision 1.1 1996/03/13 22:27:41 sde1000 - * Initial revision - * - */ - -static char *version="secnet version " VERSION " $Date: 1996/03/13 22:27:41 $"; +extern char version[]; +#include "secnet.h" #include #include #include -#include #include +#include #include -#include -#include -#include #include -#include #include -#include -#include "secnet.h" #include "util.h" #include "conffile.h" -#include "modules.h" +#include "process.h" -/* Command-line options (possibly config-file options too) */ +/* XXX should be from autoconf */ static char *configfile="/etc/secnet/secnet.conf"; +static char *sites_key="sites"; +bool_t just_check_config=False; static char *userid=NULL; static uid_t uid=0; -static bool_t background=True; +bool_t background=True; static char *pidfile=NULL; +bool_t require_root_privileges=False; +string_t require_root_privileges_explanation=NULL; + +static pid_t secnet_pid; + +/* from log.c */ +extern uint32_t message_level; +extern bool_t secnet_is_daemon; +extern struct log_if *system_log; + +/* from process.c */ +extern void start_signal_handling(void); /* Structures dealing with poll() call */ struct poll_interest { @@ -64,10 +67,12 @@ static void parse_options(int argc, char **argv) {"quiet", 0, 0, 'f'}, {"debug", 1, 0, 'd'}, {"config", 1, 0, 'c'}, + {"just-check-config", 0, 0, 'j'}, + {"sites-key", 1, 0, 's'}, {0,0,0,0} }; - c=getopt_long(argc, argv, "vwdnc:ft:", + c=getopt_long(argc, argv, "vwdnjc:ft:s:", long_options, &option_index); if (c==-1) break; @@ -75,42 +80,51 @@ static void parse_options(int argc, char **argv) switch(c) { case 2: /* Help */ - fprintf(stderr, - "Usage: secnet [OPTION]...\n\n" - " -f, --silent, --quiet suppress error messages\n" - " -w, --nowarnings suppress warnings\n" - " -v, --verbose output extra diagnostics\n" - " -c, --config=filename specify a configuration file\n" - " -n, --nodetach do not run in background\n" - " -d, --debug=item,... set debug options\n" - " --help display this help and exit\n" - " --version output version information and exit\n" + printf("Usage: secnet [OPTION]...\n\n" + " -f, --silent, --quiet suppress error messages\n" + " -w, --nowarnings suppress warnings\n" + " -v, --verbose output extra diagnostics\n" + " -c, --config=filename specify a configuration file\n" + " -j, --just-check-config stop after reading " + "configuration file\n" + " -s, --sites-key=name configuration key that " + "specifies active sites\n" + " -n, --nodetach do not run in background\n" + " -d, --debug=item,... set debug options\n" + " --help display this help and exit\n" + " --version output version information " + "and exit\n" ); exit(0); break; case 1: /* Version */ - fprintf(stderr,"%s\n",version); + printf("%s\n",version); exit(0); break; case 'v': - message_level|=M_INFO|M_WARNING|M_ERROR|M_FATAL; + message_level|=M_INFO|M_NOTICE|M_WARNING|M_ERR|M_SECURITY| + M_FATAL; break; - case 'n': - background=False; + case 'w': + message_level&=(~M_WARNING); break; case 'd': - message_level|=M_DEBUG_CONFIG|M_DEBUG_PHASE; + message_level|=M_DEBUG_CONFIG|M_DEBUG_PHASE|M_DEBUG; break; case 'f': message_level=M_FATAL; break; + case 'n': + background=False; + break; + case 'c': if (optarg) configfile=safe_strdup(optarg,"config_filename"); @@ -118,16 +132,27 @@ static void parse_options(int argc, char **argv) fatal("secnet: no config filename specified"); break; + case 'j': + just_check_config=True; + break; + + case 's': + if (optarg) + sites_key=safe_strdup(optarg,"sites-key"); + else + fatal("secnet: no sites key specified"); + break; + case '?': break; default: - Message(M_WARNING,"secnet: Unknown getopt code %c\n",c); + Message(M_ERR,"secnet: Unknown getopt code %c\n",c); } } if (argc-optind != 0) { - Message(M_WARNING,"secnet: You gave extra command line parameters, " + Message(M_ERR,"secnet: You gave extra command line parameters, " "which were ignored.\n"); } } @@ -137,7 +162,6 @@ static void setup(dict_t *config) list_t *l; item_t *site; dict_t *system; - struct log_if *log; struct passwd *pw; struct cloc loc; int i; @@ -145,7 +169,7 @@ static void setup(dict_t *config) l=dict_lookup(config,"system"); if (!l || list_elem(l,0)->type!=t_dict) { - fatal("configuration does not include a \"system\" dictionary\n"); + fatal("configuration does not include a \"system\" dictionary"); } system=list_elem(l,0)->data.dict; loc=list_elem(l,0)->loc; @@ -153,10 +177,9 @@ static void setup(dict_t *config) /* Arrange systemwide log facility */ l=dict_lookup(system,"log"); if (!l) { - fatal("configuration does not include a system/log facility\n"); + fatal("configuration does not include a system/log facility"); } - log=init_log(l); - log->log(log->st,LOG_DEBUG,"secnet " VERSION ": logging started"); + system_log=init_log(l); /* Who are we supposed to run as? */ userid=dict_read_string(system,"userid",False,"system",loc); @@ -170,29 +193,38 @@ static void setup(dict_t *config) } while(pw); endpwent(); if (uid==0) { - fatal("userid \"%s\" not found\n",userid); + fatal("userid \"%s\" not found",userid); } } /* Pidfile name */ pidfile=dict_read_string(system,"pidfile",False,"system",loc); + /* Check whether we need root privileges */ + if (require_root_privileges && uid!=0) { + fatal("the configured feature \"%s\" requires " + "that secnet retain root privileges while running.", + require_root_privileges_explanation); + } + /* Go along site list, starting sites */ - l=dict_lookup(config,"sites"); + l=dict_lookup(config,sites_key); if (!l) { - fatal("configuration did not define any remote sites\n"); - } - i=0; - while ((site=list_elem(l, i++))) { - struct site_if *s; - if (site->type!=t_closure) { - cfgfatal(site->loc,"system","non-closure in site list"); - } - if (site->data.closure->type!=CL_SITE) { - cfgfatal(site->loc,"system","non-site closure in site list"); + Message(M_WARNING,"secnet: configuration key \"%s\" is missing; no " + "remote sites are defined\n",sites_key); + } else { + i=0; + while ((site=list_elem(l, i++))) { + struct site_if *s; + if (site->type!=t_closure) { + cfgfatal(site->loc,"system","non-closure in site list"); + } + if (site->data.closure->type!=CL_SITE) { + cfgfatal(site->loc,"system","non-site closure in site list"); + } + s=site->data.closure->interface; + s->control(s->st,True); } - s=site->data.closure->interface; - s->control(s->st,True); } } @@ -233,14 +265,17 @@ static void run(void) fds=alloca(sizeof(*fds)*total_nfds); if (!fds) { - fatal("run: couldn't alloca\n"); + fatal("run: couldn't alloca"); } - while (!finished) { + Message(M_NOTICE,"%s [%d]: starting\n",version,secnet_pid); + + do { if (gettimeofday(&tv_now, NULL)!=0) { fatal_perror("main loop: gettimeofday"); } - now=(tv_now.tv_sec*1000)+(tv_now.tv_usec/1000); + now=((uint64_t)tv_now.tv_sec*(uint64_t)1000)+ + ((uint64_t)tv_now.tv_usec/(uint64_t)1000); idx=0; for (i=reg; i; i=i->next) { i->after(i->state, fds+idx, i->nfds, &tv_now, &now); @@ -255,16 +290,17 @@ static void run(void) if (rv!=0) { /* XXX we need to handle this properly: increase the nfds available */ - fatal("run: beforepoll_fn (%s) returns %d\n",i->desc,rv); + fatal("run: beforepoll_fn (%s) returns %d",i->desc,rv); } if (timeout<-1) { - fatal("run: beforepoll_fn (%s) set timeout to %d\n",timeout); + fatal("run: beforepoll_fn (%s) set timeout to %d",timeout); } idx+=nfds; remain-=nfds; i->nfds=nfds; } do { + if (finished) break; rv=poll(fds, idx, timeout); if (rv<0) { if (errno!=EINTR) { @@ -272,7 +308,7 @@ static void run(void) } } } while (rv<0); - } + } while (!finished); } static void droppriv(void) @@ -282,21 +318,26 @@ static void droppriv(void) add_hook(PHASE_SHUTDOWN,system_phase_hook,NULL); - /* Background now, if we're supposed to: we may be unable to write the - pidfile if we don't. */ - if (background) { - printf("goto background\n"); - /* Open the pidfile before forking - that way the parent can tell - whether it succeeds */ - if (pidfile) { - pf=fopen(pidfile,"w"); - if (!pf) { - fatal_perror("cannot open pidfile \"%s\"",pidfile); - } - } else { - Message(M_WARNING,"secnet: no pidfile configured, but " - "backgrounding anyway\n"); + /* Open the pidfile for writing now: we may be unable to do so + once we drop privileges. */ + if (pidfile) { + pf=fopen(pidfile,"w"); + if (!pf) { + fatal_perror("cannot open pidfile \"%s\"",pidfile); + } + } + if (!background && pf) { + fprintf(pf,"%d\n",getpid()); + fclose(pf); + } + + /* Now drop privileges */ + if (uid!=0) { + if (setuid(uid)!=0) { + fatal_perror("can't set uid to \"%s\"",userid); } + } + if (background) { p=fork(); if (p>0) { if (pf) { @@ -308,29 +349,34 @@ static void droppriv(void) } else if (p==0) { /* Child process - all done, just carry on */ if (pf) fclose(pf); - printf("child\n"); + /* Close stdin, stdout and stderr; we don't need them any more */ + /* XXX we must leave stderr pointing to something useful - + a pipe to a log destination, for example, or just leave + it alone. */ + close(0); + close(1); + /* XXX close(2); */ + secnet_is_daemon=True; + setsid(); } else { /* Error */ fatal_perror("cannot fork"); exit(1); } - } else { - if (pidfile) { - pf=fopen(pidfile,"w"); - if (!pf) { - fatal_perror("cannot open pidfile \"%s\"",pidfile); - } - fprintf(pf,"%d\n",getpid()); - fclose(pf); - } } + secnet_pid=getpid(); +} - /* Drop privilege now, if configured to do so */ - if (uid!=0) { - if (setuid(uid)!=0) { - fatal_perror("can't set uid to \"%s\"",userid); - } - } +static signal_notify_fn finish,ignore_hup; +static void finish(void *st, int signum) +{ + finished=True; + Message(M_NOTICE,"%s [%d]: received %s\n",version,secnet_pid,(string_t)st); +} +static void ignore_hup(void *st, int signum) +{ + Message(M_INFO,"%s [%d]: received SIGHUP\n",version,secnet_pid); + return; } int main(int argc, char **argv) @@ -345,15 +391,27 @@ int main(int argc, char **argv) enter_phase(PHASE_SETUP); setup(config); + + if (just_check_config) { + Message(M_INFO,"configuration file check complete\n"); + exit(0); + } + + enter_phase(PHASE_GETRESOURCES); + /* Appropriate phase hooks will have been run */ enter_phase(PHASE_DROPPRIV); droppriv(); + start_signal_handling(); + request_signal_notification(SIGTERM,finish,"SIGTERM"); + if (!background) request_signal_notification(SIGINT,finish,"SIGINT"); + request_signal_notification(SIGHUP,ignore_hup,NULL); enter_phase(PHASE_RUN); run(); enter_phase(PHASE_SHUTDOWN); + Message(M_NOTICE,"%s [%d]: finished\n",version,secnet_pid); return 0; } -