diff options
-rw-r--r-- | src/cfg_file.l | 46 | ||||
-rw-r--r-- | src/controller.c | 7 | ||||
-rw-r--r-- | src/lmtp.c | 27 | ||||
-rw-r--r-- | src/lmtp.h | 2 | ||||
-rw-r--r-- | src/main.c | 162 | ||||
-rw-r--r-- | src/util.c | 15 | ||||
-rw-r--r-- | src/util.h | 1 | ||||
-rw-r--r-- | src/worker.c | 1 |
8 files changed, 152 insertions, 109 deletions
diff --git a/src/cfg_file.l b/src/cfg_file.l index ac2624155..4e8e4bb8d 100644 --- a/src/cfg_file.l +++ b/src/cfg_file.l @@ -10,6 +10,7 @@ #define MAX_INCLUDE_DEPTH 10 YY_BUFFER_STATE include_stack[MAX_INCLUDE_DEPTH]; +int line_stack[MAX_INCLUDE_DEPTH]; int include_stack_ptr = 0; extern struct config_file *cfg; @@ -101,36 +102,41 @@ yes|YES|no|NO|[yY]|[nN] yylval.flag=parse_flag(yytext); return FLAG; [a-zA-Z0-9].[a-zA-Z0-9\/.-]+ yylval.string=strdup(yytext); return DOMAINNAME; <incl>[ \t]* /* eat the whitespace */ <incl>[^ \t\n]+ { /* got the include file name */ - if (include_stack_ptr >= MAX_INCLUDE_DEPTH) { - yyerror ("yylex: includes nested too deeply"); - return -1; - } + /* got the include file name */ + if ( include_stack_ptr >= MAX_INCLUDE_DEPTH ) { + yyerror ("yylex: includes nested too deeply" ); + return -1; + } - include_stack[include_stack_ptr++] = - YY_CURRENT_BUFFER; + line_stack[include_stack_ptr] = yylineno; + include_stack[include_stack_ptr++] = YY_CURRENT_BUFFER; - yyin = fopen (yytext, "r"); + yylineno = 1; + yyin = fopen (yytext, "r"); - if (! yyin) { - yyerror ("yylex: cannot open include file"); + if (!yyin) { + yyerror ("yylex: cannot open include file"); return -1; } - yy_switch_to_buffer (yy_create_buffer (yyin, YY_BUF_SIZE)); + yy_switch_to_buffer (yy_create_buffer (yyin, YY_BUF_SIZE)); - BEGIN(INITIAL); - } + BEGIN(INITIAL); +} <<EOF>> { - if ( --include_stack_ptr < 0 ) { + if ( --include_stack_ptr < 0 ) { + include_stack_ptr = 0; + yylineno = 1; post_load_config (cfg); - yyterminate (); - } - else { - yy_delete_buffer (YY_CURRENT_BUFFER); - yy_switch_to_buffer (include_stack[include_stack_ptr]); - } - } + yyterminate (); + } + else { + yy_delete_buffer (YY_CURRENT_BUFFER); + yy_switch_to_buffer (include_stack[include_stack_ptr] ); + yylineno = line_stack[include_stack_ptr]; + } +} <module>\n /* ignore EOL */; <module>[ \t]+ /* ignore whitespace */; diff --git a/src/controller.c b/src/controller.c index 26c773cff..76caab0df 100644 --- a/src/controller.c +++ b/src/controller.c @@ -90,11 +90,11 @@ sigusr_handler (int fd, short what, void *arg) struct rspamd_worker *worker = (struct rspamd_worker *)arg; /* Do not accept new connections, preparing to end worker's process */ struct timeval tv; - tv.tv_sec = SOFT_SHUTDOWN_TIME; + tv.tv_sec = 2; tv.tv_usec = 0; event_del (&worker->sig_ev); event_del (&worker->bind_ev); - msg_info ("controller's shutdown is pending in %d sec", SOFT_SHUTDOWN_TIME); + msg_info ("controller's shutdown is pending in %d sec", 2); event_loopexit (&tv); return; } @@ -568,6 +568,9 @@ start_controller (struct rspamd_worker *worker) io_tv.tv_usec = 0; event_loop (0); + close (listen_sock); + + exit (EXIT_SUCCESS); } diff --git a/src/lmtp.c b/src/lmtp.c index 276aad0cb..a8f10ee2c 100644 --- a/src/lmtp.c +++ b/src/lmtp.c @@ -62,7 +62,7 @@ sigusr_handler (int fd, short what, void *arg) event_del (&worker->sig_ev); event_del (&worker->bind_ev); do_reopen_log = 1; - msg_info ("worker's shutdown is pending in %d sec", SOFT_SHUTDOWN_TIME); + msg_info ("lmtp worker's shutdown is pending in %d sec", SOFT_SHUTDOWN_TIME); event_loopexit (&tv); return; } @@ -244,11 +244,10 @@ accept_socket (int fd, short what, void *arg) * Start lmtp worker process */ void -start_lmtp_worker (struct rspamd_worker *worker) +start_lmtp_worker (struct rspamd_worker *worker, int listen_sock) { struct sigaction signals; - int listen_sock, i; - struct sockaddr_un *un_addr; + int i; char *hostbuf; long int hostmax; @@ -264,25 +263,6 @@ start_lmtp_worker (struct rspamd_worker *worker) signal_set (&worker->sig_ev, SIGUSR2, sigusr_handler, (void *) worker); signal_add (&worker->sig_ev, NULL); - /* Create listen socket */ - if (worker->srv->cfg->lmtp_family == AF_INET) { - if ((listen_sock = make_tcp_socket (&worker->srv->cfg->lmtp_addr, worker->srv->cfg->lmtp_port, TRUE)) == -1) { - msg_err ("start_lmtp: cannot create tcp listen socket. %s", strerror (errno)); - exit(-errno); - } - } - else { - un_addr = (struct sockaddr_un *) alloca (sizeof (struct sockaddr_un)); - if (!un_addr || (listen_sock = make_unix_socket (worker->srv->cfg->lmtp_host, un_addr, TRUE)) == -1) { - msg_err ("start_lmtp: cannot create unix listen socket. %s", strerror (errno)); - exit(-errno); - } - } - - if (listen (listen_sock, -1) == -1) { - msg_err ("start_lmtp: cannot listen on socket. %s", strerror (errno)); - exit(-errno); - } /* Accept event */ event_set(&worker->bind_ev, listen_sock, EV_READ | EV_PERSIST, accept_socket, (void *)worker); event_add(&worker->bind_ev, NULL); @@ -306,6 +286,7 @@ start_lmtp_worker (struct rspamd_worker *worker) io_tv.tv_usec = 0; event_loop (0); + exit (EXIT_SUCCESS); } /* diff --git a/src/lmtp.h b/src/lmtp.h index d7c13c497..b784eed3b 100644 --- a/src/lmtp.h +++ b/src/lmtp.h @@ -15,6 +15,6 @@ #define LMTP_NO_RCPT 554 #define LMTP_TEMP_FAIL 421 -void start_lmtp_worker (struct rspamd_worker *worker); +void start_lmtp_worker (struct rspamd_worker *worker, int listen_sock); #endif diff --git a/src/main.c b/src/main.c index b3a36da98..16af301dc 100644 --- a/src/main.c +++ b/src/main.c @@ -35,7 +35,7 @@ struct config_file *cfg; static void sig_handler (int ); -static struct rspamd_worker * fork_worker (struct rspamd_main *, int, int, enum process_type); +static struct rspamd_worker * fork_worker (struct rspamd_main *, int, enum process_type); sig_atomic_t do_restart; sig_atomic_t do_terminate; @@ -203,45 +203,52 @@ config_logger (struct rspamd_main *rspamd, gboolean is_fatal) } } -static struct rspamd_worker * -fork_worker (struct rspamd_main *rspamd, int listen_sock, int reconfig, enum process_type type) +static void +reread_config (struct rspamd_main *rspamd) { - struct rspamd_worker *cur; + struct config_file *tmp_cfg; char *cfg_file; FILE *f; - struct config_file *tmp_cfg; - /* Starting worker process */ - cur = (struct rspamd_worker *)g_malloc (sizeof (struct rspamd_worker)); - if (cur) { - /* Reconfig needed */ - if (reconfig) { - tmp_cfg = (struct config_file *) g_malloc (sizeof (struct config_file)); - if (tmp_cfg) { - bzero (tmp_cfg, sizeof (struct config_file)); - tmp_cfg->cfg_pool = memory_pool_new (32768); - cfg_file = memory_pool_strdup (tmp_cfg->cfg_pool, rspamd->cfg->cfg_name); - f = fopen (rspamd->cfg->cfg_name , "r"); - if (f == NULL) { - msg_warn ("fork_worker: cannot open file: %s", rspamd->cfg->cfg_name ); - } - else { - yyin = f; - yyrestart (yyin); - if (yyparse() != 0 || yynerrs > 0) { - msg_warn ("fork_worker: yyparse: cannot parse config file, %d errors", yynerrs); - fclose (f); - } - else { - free_config (rspamd->cfg); - g_free (rspamd->cfg); - rspamd->cfg = tmp_cfg; - rspamd->cfg->cfg_name = cfg_file; - config_logger (rspamd, FALSE); - } - } + tmp_cfg = (struct config_file *) g_malloc (sizeof (struct config_file)); + if (tmp_cfg) { + bzero (tmp_cfg, sizeof (struct config_file)); + tmp_cfg->cfg_pool = memory_pool_new (memory_pool_get_size ()); + init_defaults (tmp_cfg); + cfg_file = memory_pool_strdup (tmp_cfg->cfg_pool, rspamd->cfg->cfg_name); + f = fopen (rspamd->cfg->cfg_name , "r"); + if (f == NULL) { + msg_warn ("reread_config: cannot open file: %s", rspamd->cfg->cfg_name ); + } + else { + yyin = f; + yyrestart (yyin); + + if (yyparse() != 0 || yynerrs > 0) { + msg_warn ("reread_config: yyparse: cannot parse config file, %d errors", yynerrs); + fclose (f); + } + else { + msg_debug ("reread_config: replacing config"); + free_config (rspamd->cfg); + close_log (rspamd->cfg); + g_free (rspamd->cfg); + rspamd->cfg = tmp_cfg; + rspamd->cfg->cfg_name = cfg_file; + config_logger (rspamd, FALSE); + msg_info ("reread_config: config rereaded successfully"); } } + } +} + +static struct rspamd_worker * +fork_worker (struct rspamd_main *rspamd, int listen_sock, enum process_type type) +{ + struct rspamd_worker *cur; + /* Starting worker process */ + cur = (struct rspamd_worker *)g_malloc (sizeof (struct rspamd_worker)); + if (cur) { bzero (cur, sizeof (struct rspamd_worker)); TAILQ_INSERT_HEAD (&rspamd->workers, cur, next); cur->srv = rspamd; @@ -261,7 +268,7 @@ fork_worker (struct rspamd_main *rspamd, int listen_sock, int reconfig, enum pro setproctitle ("lmtp process"); pidfile_close (rspamd->pfh); msg_info ("fork_worker: starting lmtp process %d", getpid ()); - start_lmtp_worker (cur); + start_lmtp_worker (cur, listen_sock); case TYPE_WORKER: default: setproctitle ("worker process"); @@ -297,20 +304,46 @@ fork_delayed (struct rspamd_main *rspamd, int listen_sock) while (workers_pending != NULL) { cur = workers_pending; workers_pending = g_list_remove_link (workers_pending, cur); - fork_worker (rspamd, listen_sock, 0, GPOINTER_TO_INT (cur->data)); + fork_worker (rspamd, listen_sock, GPOINTER_TO_INT (cur->data)); g_list_free_1 (cur); } } +static int +create_listen_socket (struct in_addr *addr, int port, int family, char *path) +{ + int listen_sock = -1; + struct sockaddr_un *un_addr; + /* Create listen socket */ + if (family == AF_INET) { + if ((listen_sock = make_tcp_socket (addr, port, TRUE)) == -1) { + msg_err ("create_listen_socket: cannot create tcp listen socket. %s", strerror (errno)); + } + } + else { + un_addr = (struct sockaddr_un *) alloca (sizeof (struct sockaddr_un)); + if (!un_addr || (listen_sock = make_unix_socket (path, un_addr, TRUE)) == -1) { + msg_err ("create_listen_socket: cannot create unix listen socket. %s", strerror (errno)); + } + } + + if (listen_sock != -1) { + if (listen (listen_sock, -1) == -1) { + msg_err ("start_lmtp: cannot listen on socket. %s", strerror (errno)); + } + } + + return listen_sock; +} + int main (int argc, char **argv, char **env) { struct rspamd_main *rspamd; struct module_ctx *cur_module = NULL; - int res = 0, i, listen_sock; + int res = 0, i, listen_sock, lmtp_listen_sock; struct sigaction signals; struct rspamd_worker *cur, *cur_tmp, *active_worker; - struct sockaddr_un *un_addr; FILE *f; pid_t wrk; char *args[] = { "", "-e", "0", NULL }; @@ -379,31 +412,26 @@ main (int argc, char **argv, char **env) } /* Create listen socket */ - if (rspamd->cfg->bind_family == AF_INET) { - if ((listen_sock = make_tcp_socket (&rspamd->cfg->bind_addr, rspamd->cfg->bind_port, TRUE)) == -1) { - msg_err ("main: cannot create tcp listen socket. %s", strerror (errno)); - exit(-errno); - } + listen_sock = create_listen_socket (&rspamd->cfg->bind_addr, rspamd->cfg->bind_port, + rspamd->cfg->bind_family, rspamd->cfg->bind_host); + if (listen_sock == -1) { + exit(-errno); } - else { - un_addr = (struct sockaddr_un *) g_malloc (sizeof (struct sockaddr_un)); - if (!un_addr || (listen_sock = make_unix_socket (rspamd->cfg->bind_host, un_addr, TRUE)) == -1) { - msg_err ("main: cannot create unix listen socket. %s", strerror (errno)); + + if (cfg->lmtp_enable) { + lmtp_listen_sock = create_listen_socket (&rspamd->cfg->lmtp_addr, rspamd->cfg->lmtp_port, + rspamd->cfg->lmtp_family, rspamd->cfg->lmtp_host); + if (listen_sock == -1) { exit(-errno); } } - if (listen (listen_sock, -1) == -1) { - msg_err ("main: cannot listen on socket. %s", strerror (errno)); - exit(-errno); - } - /* Drop privilleges */ drop_priv (cfg); config_logger (rspamd, TRUE); - msg_info ("main: starting..."); + msg_info ("main: rspamd "RVERSION " is starting"); rspamd->cfg->cfg_name = memory_pool_strdup (rspamd->cfg->cfg_pool, rspamd->cfg->cfg_name ); /* Strictly set temp dir */ @@ -464,17 +492,17 @@ main (int argc, char **argv, char **env) rspamd->statfile_pool = statfile_pool_new (cfg->max_statfile_size); for (i = 0; i < cfg->workers_number; i++) { - fork_worker (rspamd, listen_sock, 0, TYPE_WORKER); + fork_worker (rspamd, listen_sock, TYPE_WORKER); } /* Start controller if enabled */ if (cfg->controller_enabled) { - fork_worker (rspamd, listen_sock, 0, TYPE_CONTROLLER); + fork_worker (rspamd, listen_sock, TYPE_CONTROLLER); } /* Start lmtp if enabled */ if (cfg->lmtp_enable) { for (i = 0; i < cfg->lmtp_workers_number; i++) { - fork_worker (rspamd, listen_sock, 0, TYPE_LMTP); + fork_worker (rspamd, lmtp_listen_sock, TYPE_LMTP); } } @@ -504,6 +532,10 @@ main (int argc, char **argv, char **env) /* Normal worker termination, do not fork one more */ msg_info ("main: %s process %d terminated normally", (cur->type != TYPE_WORKER) ? "controller" : "worker", cur->pid); + /* But respawn controller */ + if (cur->type == TYPE_CONTROLLER) { + fork_worker (rspamd, listen_sock, TYPE_CONTROLLER); + } } else { if (WIFSIGNALED (res)) { @@ -524,10 +556,22 @@ main (int argc, char **argv, char **env) } if (do_restart) { do_restart = 0; + do_reopen_log = 1; + msg_info ("main: rspamd "RVERSION " is restarting"); if (active_worker == NULL) { - /* Start new worker that would reread configuration*/ - active_worker = fork_worker (rspamd, listen_sock, 1, TYPE_WORKER); + /* reread_config (rspamd); */ + TAILQ_FOREACH_SAFE (cur, &rspamd->workers, next, cur_tmp) { + if (cur->type == TYPE_WORKER || cur->type == TYPE_LMTP) { + /* Start new workers that would reread configuration */ + active_worker = fork_worker (rspamd, listen_sock, cur->type); + } + /* Immideately send termination request to conroller and wait for SIGCHLD */ + if (cur->type == TYPE_CONTROLLER) { + kill (cur->pid, SIGUSR2); + cur->is_dying = 1; + } + } } /* Do not start new workers until active worker is not ready for accept */ } @@ -537,7 +581,7 @@ main (int argc, char **argv, char **env) if (active_worker != NULL) { msg_info ("main: worker process %d has been successfully started", active_worker->pid); TAILQ_FOREACH_SAFE (cur, &rspamd->workers, next, cur_tmp) { - if (cur != active_worker && !cur->is_dying && cur->type == TYPE_WORKER) { + if (cur != active_worker && !cur->is_dying && cur->type != TYPE_CONTROLLER) { /* Send to old workers SIGUSR2 */ kill (cur->pid, SIGUSR2); cur->is_dying = 1; diff --git a/src/util.c b/src/util.c index a60bafea6..62d656140 100644 --- a/src/util.c +++ b/src/util.c @@ -825,14 +825,13 @@ open_log (struct config_file *cfg) } } -int -reopen_log (struct config_file *cfg) +void +close_log (struct config_file *cfg) { - do_reopen_log = 0; switch (cfg->log_type) { case RSPAMD_LOG_CONSOLE: /* Do nothing with console */ - return 0; + break; case RSPAMD_LOG_SYSLOG: closelog (); break; @@ -840,6 +839,14 @@ reopen_log (struct config_file *cfg) close (cfg->log_fd); break; } + +} + +int +reopen_log (struct config_file *cfg) +{ + do_reopen_log = 0; + close_log (cfg); return open_log (cfg); } diff --git a/src/util.h b/src/util.h index 7267a719b..fa47f21e1 100644 --- a/src/util.h +++ b/src/util.h @@ -48,6 +48,7 @@ int pidfile_remove(struct pidfh *pfh); #endif int open_log (struct config_file *cfg); +void close_log (struct config_file *cfg); int reopen_log (struct config_file *cfg); void syslog_log_function (const gchar *log_domain, GLogLevelFlags log_level, const gchar *message, gpointer arg); void file_log_function (const gchar *log_domain, GLogLevelFlags log_level, const gchar *message, gpointer arg); diff --git a/src/worker.c b/src/worker.c index ee10d9a84..457395a97 100644 --- a/src/worker.c +++ b/src/worker.c @@ -297,6 +297,7 @@ start_worker (struct rspamd_worker *worker, int listen_sock) kill (getppid (), SIGUSR2); event_loop (0); + exit (EXIT_SUCCESS); } /* |