mirror of
http://git.haproxy.org/git/haproxy.git/
synced 2024-12-16 16:34:42 +00:00
MEDIUM: Improve signal handling in systemd wrapper.
Move all code out of the signal handlers, since this is potentially dangerous. To make sure the signal handlers behave as expected, use sigaction() instead of signal(). That also obsoletes messing with the signal mask after restart. Signed-off-by: Conrad Hoffmann <conrad@soundcloud.com>
This commit is contained in:
parent
eb2cf45b72
commit
5b5ea9c933
@ -22,6 +22,8 @@
|
|||||||
#define SD_DEBUG "<7>"
|
#define SD_DEBUG "<7>"
|
||||||
#define SD_NOTICE "<5>"
|
#define SD_NOTICE "<5>"
|
||||||
|
|
||||||
|
static volatile sig_atomic_t caught_signal;
|
||||||
|
|
||||||
static char *pid_file = "/run/haproxy.pid";
|
static char *pid_file = "/run/haproxy.pid";
|
||||||
static int wrapper_argc;
|
static int wrapper_argc;
|
||||||
static char **wrapper_argv;
|
static char **wrapper_argv;
|
||||||
@ -103,7 +105,12 @@ static int read_pids(char ***pid_strv)
|
|||||||
return read;
|
return read;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sigusr2_handler(int signum __attribute__((unused)))
|
static void signal_handler(int signum)
|
||||||
|
{
|
||||||
|
caught_signal = signum;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void do_restart(void)
|
||||||
{
|
{
|
||||||
setenv(REEXEC_FLAG, "1", 1);
|
setenv(REEXEC_FLAG, "1", 1);
|
||||||
fprintf(stderr, SD_NOTICE "haproxy-systemd-wrapper: re-executing\n");
|
fprintf(stderr, SD_NOTICE "haproxy-systemd-wrapper: re-executing\n");
|
||||||
@ -111,7 +118,7 @@ static void sigusr2_handler(int signum __attribute__((unused)))
|
|||||||
execv(wrapper_argv[0], wrapper_argv);
|
execv(wrapper_argv[0], wrapper_argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sigint_handler(int signum __attribute__((unused)))
|
static void do_shutdown(void)
|
||||||
{
|
{
|
||||||
int i, pid;
|
int i, pid;
|
||||||
char **pid_strv = NULL;
|
char **pid_strv = NULL;
|
||||||
@ -147,25 +154,21 @@ int main(int argc, char **argv)
|
|||||||
--argc; ++argv;
|
--argc; ++argv;
|
||||||
init(argc, argv);
|
init(argc, argv);
|
||||||
|
|
||||||
signal(SIGINT, &sigint_handler);
|
struct sigaction sa;
|
||||||
signal(SIGUSR2, &sigusr2_handler);
|
memset(&sa, 0, sizeof(struct sigaction));
|
||||||
|
sa.sa_handler = &signal_handler;
|
||||||
|
sigaction(SIGUSR2, &sa, NULL);
|
||||||
|
sigaction(SIGINT, &sa, NULL);
|
||||||
|
|
||||||
if (getenv(REEXEC_FLAG) != NULL) {
|
if (getenv(REEXEC_FLAG) != NULL) {
|
||||||
/* We are being re-executed: restart HAProxy gracefully */
|
/* We are being re-executed: restart HAProxy gracefully */
|
||||||
int i;
|
int i;
|
||||||
char **pid_strv = NULL;
|
char **pid_strv = NULL;
|
||||||
int nb_pid = read_pids(&pid_strv);
|
int nb_pid = read_pids(&pid_strv);
|
||||||
sigset_t sigs;
|
|
||||||
|
|
||||||
unsetenv(REEXEC_FLAG);
|
unsetenv(REEXEC_FLAG);
|
||||||
spawn_haproxy(pid_strv, nb_pid);
|
spawn_haproxy(pid_strv, nb_pid);
|
||||||
|
|
||||||
/* Unblock SIGUSR2 which was blocked by the signal handler
|
|
||||||
* before re-exec */
|
|
||||||
sigprocmask(SIG_BLOCK, NULL, &sigs);
|
|
||||||
sigdelset(&sigs, SIGUSR2);
|
|
||||||
sigprocmask(SIG_SETMASK, &sigs, NULL);
|
|
||||||
|
|
||||||
for (i = 0; i < nb_pid; ++i)
|
for (i = 0; i < nb_pid; ++i)
|
||||||
free(pid_strv[i]);
|
free(pid_strv[i]);
|
||||||
free(pid_strv);
|
free(pid_strv);
|
||||||
@ -176,8 +179,16 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
status = -1;
|
status = -1;
|
||||||
while (-1 != wait(&status) || errno == EINTR)
|
while (-1 != wait(&status) || errno == EINTR) {
|
||||||
;
|
if (caught_signal == SIGUSR2) {
|
||||||
|
caught_signal = 0;
|
||||||
|
do_restart();
|
||||||
|
}
|
||||||
|
else if (caught_signal == SIGINT) {
|
||||||
|
caught_signal = 0;
|
||||||
|
do_shutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fprintf(stderr, SD_NOTICE "haproxy-systemd-wrapper: exit, haproxy RC=%d\n",
|
fprintf(stderr, SD_NOTICE "haproxy-systemd-wrapper: exit, haproxy RC=%d\n",
|
||||||
status);
|
status);
|
||||||
|
Loading…
Reference in New Issue
Block a user