mirror of
https://github.com/clearlinux/uwsgi.git
synced 2026-06-15 18:05:50 +00:00
1130 lines
31 KiB
C
1130 lines
31 KiB
C
#include "uwsgi.h"
|
|
|
|
extern struct uwsgi_server uwsgi;
|
|
|
|
static void master_check_processes() {
|
|
|
|
// run the function, only if required
|
|
if (!uwsgi.die_on_no_workers) return;
|
|
|
|
int alive_processes = 0;
|
|
int dead_processes = 0;
|
|
|
|
int i;
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
if (uwsgi.workers[i].cheaped == 0 && uwsgi.workers[i].pid > 0) {
|
|
alive_processes++;
|
|
}
|
|
else {
|
|
dead_processes++;
|
|
}
|
|
}
|
|
|
|
if (uwsgi.die_on_no_workers) {
|
|
if (!alive_processes) {
|
|
uwsgi_log_verbose("no more processes running, auto-killing ...\n");
|
|
exit(1);
|
|
// never here;
|
|
}
|
|
}
|
|
}
|
|
|
|
void uwsgi_update_load_counters() {
|
|
|
|
int i;
|
|
uint64_t busy_workers = 0;
|
|
uint64_t idle_workers = 0;
|
|
static time_t last_sos = 0;
|
|
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
if (uwsgi.workers[i].cheaped == 0 && uwsgi.workers[i].pid > 0) {
|
|
if (uwsgi_worker_is_busy(i) == 0) {
|
|
idle_workers++;
|
|
}
|
|
else {
|
|
busy_workers++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (busy_workers >= (uint64_t) uwsgi.numproc) {
|
|
ushared->overloaded++;
|
|
|
|
if (uwsgi.vassal_sos) {
|
|
if (uwsgi.current_time - last_sos > uwsgi.vassal_sos) {
|
|
uwsgi_log_verbose("asking Emperor for reinforcements (overload: %llu)...\n", (unsigned long long) ushared->overloaded);
|
|
vassal_sos();
|
|
last_sos = uwsgi.current_time;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
ushared->busy_workers = busy_workers;
|
|
ushared->idle_workers = idle_workers;
|
|
|
|
}
|
|
|
|
void uwsgi_block_signal(int signum) {
|
|
sigset_t smask;
|
|
sigemptyset(&smask);
|
|
sigaddset(&smask, signum);
|
|
if (sigprocmask(SIG_BLOCK, &smask, NULL)) {
|
|
uwsgi_error("sigprocmask()");
|
|
}
|
|
}
|
|
|
|
void uwsgi_unblock_signal(int signum) {
|
|
sigset_t smask;
|
|
sigemptyset(&smask);
|
|
sigaddset(&smask, signum);
|
|
if (sigprocmask(SIG_UNBLOCK, &smask, NULL)) {
|
|
uwsgi_error("sigprocmask()");
|
|
}
|
|
}
|
|
|
|
void uwsgi_master_manage_udp(int udp_fd) {
|
|
char buf[4096];
|
|
struct sockaddr_in udp_client;
|
|
char udp_client_addr[16];
|
|
int i;
|
|
|
|
socklen_t udp_len = sizeof(udp_client);
|
|
ssize_t rlen = recvfrom(udp_fd, buf, 4096, 0, (struct sockaddr *) &udp_client, &udp_len);
|
|
|
|
if (rlen < 0) {
|
|
uwsgi_error("uwsgi_master_manage_udp()/recvfrom()");
|
|
}
|
|
else if (rlen > 0) {
|
|
|
|
memset(udp_client_addr, 0, 16);
|
|
if (inet_ntop(AF_INET, &udp_client.sin_addr.s_addr, udp_client_addr, 16)) {
|
|
if (buf[0] == UWSGI_MODIFIER_MULTICAST_ANNOUNCE) {
|
|
}
|
|
else if (buf[0] == 0x30 && uwsgi.snmp) {
|
|
manage_snmp(udp_fd, (uint8_t *) buf, rlen, &udp_client);
|
|
}
|
|
else {
|
|
|
|
// loop the various udp manager until one returns true
|
|
int udp_managed = 0;
|
|
for (i = 0; i < 256; i++) {
|
|
if (uwsgi.p[i]->manage_udp) {
|
|
if (uwsgi.p[i]->manage_udp(udp_client_addr, udp_client.sin_port, buf, rlen)) {
|
|
udp_managed = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// else a simple udp logger
|
|
if (!udp_managed) {
|
|
uwsgi_log("[udp:%s:%d] %.*s", udp_client_addr, ntohs(udp_client.sin_port), (int) rlen, buf);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
uwsgi_error("uwsgi_master_manage_udp()/inet_ntop()");
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void suspend_resume_them_all(int signum) {
|
|
|
|
int i;
|
|
int suspend = 0;
|
|
|
|
if (uwsgi.workers[0].suspended == 1) {
|
|
uwsgi_log_verbose("*** (SIGTSTP received) resuming workers ***\n");
|
|
uwsgi.workers[0].suspended = 0;
|
|
}
|
|
else {
|
|
uwsgi_log_verbose("*** PAUSE (press start to resume, if you do not have a joypad send SIGTSTP) ***\n");
|
|
suspend = 1;
|
|
uwsgi.workers[0].suspended = 1;
|
|
}
|
|
|
|
// subscribe/unsubscribe if needed
|
|
uwsgi_subscribe_all(suspend, 1);
|
|
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
uwsgi.workers[i].suspended = suspend;
|
|
if (uwsgi.workers[i].pid > 0) {
|
|
if (kill(uwsgi.workers[i].pid, SIGTSTP)) {
|
|
uwsgi_error("kill()");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void uwsgi_master_check_mercy() {
|
|
|
|
int i;
|
|
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
if (uwsgi.workers[i].pid > 0 && uwsgi.workers[i].cursed_at) {
|
|
if (uwsgi_now() > uwsgi.workers[i].no_mercy_at) {
|
|
uwsgi_log_verbose("worker %d (pid: %d) is taking too much time to die...NO MERCY !!!\n", i, uwsgi.workers[i].pid);
|
|
// yes that look strangem but we avoid callign it again if we skip waitpid() call below
|
|
uwsgi_curse(i, SIGKILL);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < uwsgi.mules_cnt; i++) {
|
|
if (uwsgi.mules[i].pid > 0 && uwsgi.mules[i].cursed_at) {
|
|
if (uwsgi_now() > uwsgi.mules[i].no_mercy_at) {
|
|
uwsgi_log_verbose("mule %d (pid: %d) is taking too much time to die...NO MERCY !!!\n", i + 1, uwsgi.mules[i].pid);
|
|
uwsgi_curse_mule(i, SIGKILL);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void expire_rb_timeouts(struct uwsgi_rbtree *tree) {
|
|
|
|
uint64_t current = (uint64_t) uwsgi_now();
|
|
struct uwsgi_rb_timer *urbt;
|
|
struct uwsgi_signal_rb_timer *usrbt;
|
|
|
|
for (;;) {
|
|
|
|
urbt = uwsgi_min_rb_timer(tree, NULL);
|
|
|
|
if (urbt == NULL)
|
|
return;
|
|
|
|
if (urbt->value <= current) {
|
|
// remove the timeout and add another
|
|
usrbt = (struct uwsgi_signal_rb_timer *) urbt->data;
|
|
uwsgi_del_rb_timer(tree, urbt);
|
|
free(urbt);
|
|
usrbt->iterations_done++;
|
|
uwsgi_route_signal(usrbt->sig);
|
|
if (!usrbt->iterations || usrbt->iterations_done < usrbt->iterations) {
|
|
usrbt->uwsgi_rb_timer = uwsgi_add_rb_timer(tree, uwsgi_now() + usrbt->value, usrbt);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void get_tcp_info(struct uwsgi_socket *uwsgi_sock) {
|
|
|
|
#if defined(__linux__) || defined(__FreeBSD__)
|
|
int fd = uwsgi_sock->fd;
|
|
struct tcp_info ti;
|
|
socklen_t tis = sizeof(struct tcp_info);
|
|
|
|
if (!getsockopt(fd, IPPROTO_TCP, TCP_INFO, &ti, &tis)) {
|
|
|
|
// checks for older kernels
|
|
#if defined(__linux__)
|
|
if (!ti.tcpi_sacked) {
|
|
#elif defined(__FreeBSD__)
|
|
if (!ti.__tcpi_sacked) {
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
#if defined(__linux__)
|
|
uwsgi_sock->queue = (uint64_t) ti.tcpi_unacked;
|
|
uwsgi_sock->max_queue = (uint64_t) ti.tcpi_sacked;
|
|
#elif defined(__FreeBSD__)
|
|
uwsgi_sock->queue = (uint64_t) ti.__tcpi_unacked;
|
|
uwsgi_sock->max_queue = (uint64_t) ti.__tcpi_sacked;
|
|
#endif
|
|
}
|
|
|
|
#endif
|
|
}
|
|
|
|
|
|
#ifdef __linux__
|
|
#include <linux/sockios.h>
|
|
|
|
#ifdef UNBIT
|
|
#define SIOBKLGQ 0x8908
|
|
#endif
|
|
|
|
#ifdef SIOBKLGQ
|
|
|
|
static void get_linux_unbit_SIOBKLGQ(struct uwsgi_socket *uwsgi_sock) {
|
|
|
|
int fd = uwsgi_sock->fd;
|
|
int queue = 0;
|
|
if (ioctl(fd, SIOBKLGQ, &queue) >= 0) {
|
|
uwsgi_sock->queue = (uint64_t) queue;
|
|
uwsgi_sock->max_queue = (uint64_t) uwsgi.listen_queue;
|
|
}
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
static void master_check_listen_queue() {
|
|
|
|
uint64_t backlog = 0;
|
|
struct uwsgi_socket *uwsgi_sock = uwsgi.sockets;
|
|
while(uwsgi_sock) {
|
|
if (uwsgi_sock->family == AF_INET) {
|
|
get_tcp_info(uwsgi_sock);
|
|
}
|
|
#ifdef __linux__
|
|
#ifdef SIOBKLGQ
|
|
else if (uwsgi_sock->family == AF_UNIX) {
|
|
get_linux_unbit_SIOBKLGQ(uwsgi_sock);
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
if (uwsgi_sock->queue > backlog) {
|
|
backlog = uwsgi_sock->queue;
|
|
}
|
|
|
|
if (uwsgi_sock->queue > 0 && uwsgi_sock->queue >= uwsgi_sock->max_queue) {
|
|
uwsgi_log_verbose("*** uWSGI listen queue of socket \"%s\" (fd: %d) full !!! (%llu/%llu) ***\n", uwsgi_sock->name, uwsgi_sock->fd, (unsigned long long) uwsgi_sock->queue, (unsigned long long) uwsgi_sock->max_queue);
|
|
|
|
if (uwsgi.alarm_backlog) {
|
|
char buf[1024];
|
|
int ret = snprintf(buf, 1024, "listen queue of socket \"%s\" (fd: %d) full !!! (%llu/%llu)", uwsgi_sock->name, uwsgi_sock->fd, (unsigned long long) uwsgi_sock->queue, (unsigned long long) uwsgi_sock->max_queue);
|
|
if (ret > 0 && ret < 1024) {
|
|
struct uwsgi_string_list *usl = NULL;
|
|
uwsgi_foreach(usl, uwsgi.alarm_backlog) {
|
|
uwsgi_alarm_trigger(usl->value, buf, ret);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
uwsgi_sock = uwsgi_sock->next;
|
|
}
|
|
|
|
// TODO load should be something more advanced based on different values
|
|
uwsgi.shared->load = backlog;
|
|
|
|
uwsgi.shared->backlog = backlog;
|
|
|
|
if (uwsgi.vassal_sos_backlog > 0 && uwsgi.has_emperor) {
|
|
if (uwsgi.shared->backlog >= (uint64_t) uwsgi.vassal_sos_backlog) {
|
|
// ask emperor for help
|
|
uwsgi_log_verbose("asking Emperor for reinforcements (backlog: %llu)...\n", (unsigned long long) uwsgi.shared->backlog);
|
|
vassal_sos();
|
|
}
|
|
}
|
|
}
|
|
|
|
void vassal_sos() {
|
|
if (!uwsgi.has_emperor) {
|
|
uwsgi_log("[broodlord] instance not governed by an Emperor !!!\n");
|
|
return;
|
|
}
|
|
char byte = 30;
|
|
if (write(uwsgi.emperor_fd, &byte, 1) != 1) {
|
|
uwsgi_error("vassal_sos()/write()");
|
|
}
|
|
}
|
|
|
|
int master_loop(char **argv, char **environ) {
|
|
|
|
struct timeval last_respawn;
|
|
int last_respawn_rate = 0;
|
|
|
|
pid_t diedpid;
|
|
int waitpid_status;
|
|
|
|
time_t now = 0;
|
|
|
|
int i = 0;
|
|
int rlen;
|
|
|
|
int check_interval = 1;
|
|
|
|
struct uwsgi_rb_timer *min_timeout;
|
|
struct uwsgi_rbtree *rb_timers = uwsgi_init_rb_timer();
|
|
|
|
if (uwsgi.procname_master) {
|
|
uwsgi_set_processname(uwsgi.procname_master);
|
|
}
|
|
else if (uwsgi.procname) {
|
|
uwsgi_set_processname(uwsgi.procname);
|
|
}
|
|
else if (uwsgi.auto_procname) {
|
|
uwsgi_set_processname("uWSGI master");
|
|
}
|
|
|
|
|
|
uwsgi.current_time = uwsgi_now();
|
|
|
|
uwsgi_unix_signal(SIGTSTP, suspend_resume_them_all);
|
|
uwsgi_unix_signal(SIGHUP, grace_them_all);
|
|
|
|
uwsgi_unix_signal(SIGTERM, kill_them_all);
|
|
uwsgi_unix_signal(SIGQUIT, reap_them_all);
|
|
|
|
uwsgi_unix_signal(SIGINT, kill_them_all);
|
|
uwsgi_unix_signal(SIGUSR1, stats);
|
|
|
|
atexit(uwsgi_master_cleanup_hooks);
|
|
|
|
uwsgi.master_queue = event_queue_init();
|
|
|
|
/* route signals to workers... */
|
|
#ifdef UWSGI_DEBUG
|
|
uwsgi_log("adding %d to signal poll\n", uwsgi.shared->worker_signal_pipe[0]);
|
|
#endif
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.shared->worker_signal_pipe[0]);
|
|
|
|
if (uwsgi.master_fifo) {
|
|
uwsgi.master_fifo_fd = uwsgi_master_fifo();
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.master_fifo_fd);
|
|
}
|
|
|
|
if (uwsgi.notify_socket) {
|
|
uwsgi.notify_socket_fd = bind_to_unix_dgram(uwsgi.notify_socket);
|
|
uwsgi_log("notification socket enabled on %s (fd: %d)\n", uwsgi.notify_socket, uwsgi.notify_socket_fd);
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.notify_socket_fd);
|
|
}
|
|
|
|
if (uwsgi.spoolers) {
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.shared->spooler_signal_pipe[0]);
|
|
}
|
|
|
|
if (uwsgi.mules_cnt > 0) {
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.shared->mule_signal_pipe[0]);
|
|
}
|
|
|
|
if (uwsgi.log_master) {
|
|
uwsgi.log_master_buf = uwsgi_malloc(uwsgi.log_master_bufsize);
|
|
if (!uwsgi.threaded_logger) {
|
|
#ifdef UWSGI_DEBUG
|
|
uwsgi_log("adding %d to master logging\n", uwsgi.shared->worker_log_pipe[0]);
|
|
#endif
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.shared->worker_log_pipe[0]);
|
|
if (uwsgi.req_log_master) {
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.shared->worker_req_log_pipe[0]);
|
|
}
|
|
}
|
|
else {
|
|
uwsgi_threaded_logger_spawn();
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef UWSGI_SSL
|
|
uwsgi_start_legions();
|
|
#endif
|
|
uwsgi_metrics_start_collector();
|
|
|
|
uwsgi_add_reload_fds();
|
|
|
|
uwsgi_cache_start_sweepers();
|
|
uwsgi_cache_start_sync_servers();
|
|
|
|
uwsgi.wsgi_req->buffer = uwsgi.workers[0].cores[0].buffer;
|
|
|
|
if (uwsgi.has_emperor) {
|
|
if (uwsgi.emperor_proxy) {
|
|
uwsgi.emperor_fd_proxy = bind_to_unix(uwsgi.emperor_proxy, uwsgi.listen_queue, 0, 0);
|
|
if (uwsgi.emperor_fd_proxy < 0) exit(1);
|
|
if (chmod(uwsgi.emperor_proxy, S_IRUSR|S_IWUSR)) {
|
|
uwsgi_error("[emperor-proxy] chmod()");
|
|
exit(1);
|
|
}
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.emperor_fd_proxy);
|
|
}
|
|
else {
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.emperor_fd);
|
|
}
|
|
}
|
|
|
|
#ifdef __linux__
|
|
if (uwsgi.setns_socket) {
|
|
uwsgi.setns_socket_fd = bind_to_unix(uwsgi.setns_socket, uwsgi.listen_queue, 0, 0);
|
|
if (uwsgi.setns_socket_fd < 0) exit(1);
|
|
if (chmod(uwsgi.setns_socket, S_IRUSR|S_IWUSR)) {
|
|
uwsgi_error("[setns-socket] chmod()");
|
|
exit(1);
|
|
}
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.setns_socket_fd);
|
|
}
|
|
#endif
|
|
|
|
if (uwsgi.zerg_server) {
|
|
uwsgi.zerg_server_fd = bind_to_unix(uwsgi.zerg_server, uwsgi.listen_queue, 0, 0);
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.zerg_server_fd);
|
|
uwsgi_log("*** Zerg server enabled on %s ***\n", uwsgi.zerg_server);
|
|
}
|
|
|
|
if (uwsgi.stats) {
|
|
char *tcp_port = strrchr(uwsgi.stats, ':');
|
|
if (tcp_port) {
|
|
// disable deferred accept for this socket
|
|
int current_defer_accept = uwsgi.no_defer_accept;
|
|
uwsgi.no_defer_accept = 1;
|
|
uwsgi.stats_fd = bind_to_tcp(uwsgi.stats, uwsgi.listen_queue, tcp_port);
|
|
uwsgi.no_defer_accept = current_defer_accept;
|
|
}
|
|
else {
|
|
uwsgi.stats_fd = bind_to_unix(uwsgi.stats, uwsgi.listen_queue, uwsgi.chmod_socket, uwsgi.abstract_socket);
|
|
}
|
|
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.stats_fd);
|
|
uwsgi_log("*** Stats server enabled on %s fd: %d ***\n", uwsgi.stats, uwsgi.stats_fd);
|
|
}
|
|
|
|
|
|
if (uwsgi.stats_pusher_instances) {
|
|
if (!uwsgi_thread_new(uwsgi_stats_pusher_loop)) {
|
|
uwsgi_log("!!! unable to spawn stats pusher thread !!!\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
if (uwsgi.udp_socket) {
|
|
uwsgi.udp_fd = bind_to_udp(uwsgi.udp_socket, 0, 0);
|
|
if (uwsgi.udp_fd < 0) {
|
|
uwsgi_log("unable to bind to udp socket. SNMP services will be disabled.\n");
|
|
}
|
|
else {
|
|
uwsgi_log("UDP server enabled.\n");
|
|
event_queue_add_fd_read(uwsgi.master_queue, uwsgi.udp_fd);
|
|
}
|
|
}
|
|
|
|
uwsgi.snmp_fd = uwsgi_setup_snmp();
|
|
|
|
if (uwsgi.status.is_cheap) {
|
|
uwsgi_add_sockets_to_queue(uwsgi.master_queue, -1);
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
uwsgi.workers[i].cheaped = 1;
|
|
}
|
|
uwsgi_log("cheap mode enabled: waiting for socket connection...\n");
|
|
}
|
|
|
|
|
|
// spawn mules
|
|
for (i = 0; i < uwsgi.mules_cnt; i++) {
|
|
size_t mule_patch_size = 0;
|
|
uwsgi.mules[i].patch = uwsgi_string_get_list(&uwsgi.mules_patches, i, &mule_patch_size);
|
|
uwsgi_mule(i + 1);
|
|
}
|
|
|
|
// spawn gateways
|
|
for (i = 0; i < ushared->gateways_cnt; i++) {
|
|
if (ushared->gateways[i].pid == 0) {
|
|
gateway_respawn(i);
|
|
}
|
|
}
|
|
|
|
// spawn daemons
|
|
uwsgi_daemons_spawn_all();
|
|
|
|
// first subscription
|
|
uwsgi_subscribe_all(0, 1);
|
|
|
|
// sync the cache store if needed
|
|
uwsgi_cache_sync_all();
|
|
|
|
if (uwsgi.queue_store && uwsgi.queue_filesize) {
|
|
if (msync(uwsgi.queue_header, uwsgi.queue_filesize, MS_ASYNC)) {
|
|
uwsgi_error("msync()");
|
|
}
|
|
}
|
|
|
|
// update touches timestamps
|
|
uwsgi_check_touches(uwsgi.touch_reload);
|
|
uwsgi_check_touches(uwsgi.touch_logrotate);
|
|
uwsgi_check_touches(uwsgi.touch_logreopen);
|
|
uwsgi_check_touches(uwsgi.touch_chain_reload);
|
|
uwsgi_check_touches(uwsgi.touch_workers_reload);
|
|
uwsgi_check_touches(uwsgi.touch_gracefully_stop);
|
|
// update exec touches
|
|
struct uwsgi_string_list *usl = uwsgi.touch_exec;
|
|
while (usl) {
|
|
char *space = strchr(usl->value, ' ');
|
|
if (space) {
|
|
*space = 0;
|
|
usl->len = strlen(usl->value);
|
|
usl->custom_ptr = space + 1;
|
|
}
|
|
usl = usl->next;
|
|
}
|
|
uwsgi_check_touches(uwsgi.touch_exec);
|
|
// update signal touches
|
|
usl = uwsgi.touch_signal;
|
|
while (usl) {
|
|
char *space = strchr(usl->value, ' ');
|
|
if (space) {
|
|
*space = 0;
|
|
usl->len = strlen(usl->value);
|
|
usl->custom_ptr = space + 1;
|
|
}
|
|
usl = usl->next;
|
|
}
|
|
uwsgi_check_touches(uwsgi.touch_signal);
|
|
// daemon touches
|
|
struct uwsgi_daemon *ud = uwsgi.daemons;
|
|
while (ud) {
|
|
if (ud->touch) {
|
|
uwsgi_check_touches(ud->touch);
|
|
}
|
|
ud = ud->next;
|
|
}
|
|
// hook touches
|
|
uwsgi_foreach(usl, uwsgi.hook_touch) {
|
|
char *space = strchr(usl->value, ' ');
|
|
if (space) {
|
|
*space = 0;
|
|
usl->len = strlen(usl->value);
|
|
uwsgi_string_new_list((struct uwsgi_string_list **)&usl->custom_ptr, space+1);
|
|
}
|
|
}
|
|
uwsgi_check_touches(uwsgi.hook_touch);
|
|
|
|
// fsmon
|
|
uwsgi_fsmon_setup();
|
|
|
|
uwsgi_foreach(usl, uwsgi.signal_timers) {
|
|
char *space = strchr(usl->value, ' ');
|
|
if (!space) {
|
|
uwsgi_log("invalid signal timer syntax, must be: <signal> <seconds>\n");
|
|
exit(1);
|
|
}
|
|
*space = 0;
|
|
uwsgi_add_timer(atoi(usl->value), atoi(space+1));
|
|
*space = ' ';
|
|
}
|
|
|
|
uwsgi_foreach(usl, uwsgi.rb_signal_timers) {
|
|
char *space = strchr(usl->value, ' ');
|
|
if (!space) {
|
|
uwsgi_log("invalid redblack signal timer syntax, must be: <signal> <seconds>\n");
|
|
exit(1);
|
|
}
|
|
*space = 0;
|
|
uwsgi_signal_add_rb_timer(atoi(usl->value), atoi(space+1), 0);
|
|
*space = ' ';
|
|
}
|
|
|
|
// setup cheaper algos (can be stacked)
|
|
uwsgi.cheaper_algo = uwsgi_cheaper_algo_spare;
|
|
if (uwsgi.requested_cheaper_algo) {
|
|
uwsgi.cheaper_algo = NULL;
|
|
struct uwsgi_cheaper_algo *uca = uwsgi.cheaper_algos;
|
|
while (uca) {
|
|
if (!strcmp(uca->name, uwsgi.requested_cheaper_algo)) {
|
|
uwsgi.cheaper_algo = uca->func;
|
|
break;
|
|
}
|
|
uca = uca->next;
|
|
}
|
|
|
|
if (!uwsgi.cheaper_algo) {
|
|
uwsgi_log("unable to find requested cheaper algorithm, falling back to spare\n");
|
|
uwsgi.cheaper_algo = uwsgi_cheaper_algo_spare;
|
|
}
|
|
|
|
}
|
|
|
|
// here really starts the master loop
|
|
uwsgi_hooks_run(uwsgi.hook_master_start, "master-start", 1);
|
|
|
|
for (;;) {
|
|
//uwsgi_log("uwsgi.ready_to_reload %d %d\n", uwsgi.ready_to_reload, uwsgi.numproc);
|
|
|
|
// run master_cycle hook for every plugin
|
|
for (i = 0; i < uwsgi.gp_cnt; i++) {
|
|
if (uwsgi.gp[i]->master_cycle) {
|
|
uwsgi.gp[i]->master_cycle();
|
|
}
|
|
}
|
|
for (i = 0; i < 256; i++) {
|
|
if (uwsgi.p[i]->master_cycle) {
|
|
uwsgi.p[i]->master_cycle();
|
|
}
|
|
}
|
|
|
|
// check for death (before reload !!!)
|
|
uwsgi_master_check_death();
|
|
// check for realod
|
|
if (uwsgi_master_check_reload(argv)) {
|
|
return -1;
|
|
}
|
|
|
|
// check chain reload
|
|
uwsgi_master_check_chain();
|
|
|
|
// check if some worker is taking too much to die...
|
|
uwsgi_master_check_mercy();
|
|
|
|
// check for daemons (smart and dumb)
|
|
uwsgi_daemons_smart_check();
|
|
|
|
// cheaper management
|
|
if (uwsgi.cheaper && !uwsgi.status.is_cheap && !uwsgi_instance_is_reloading && !uwsgi_instance_is_dying && !uwsgi.workers[0].suspended) {
|
|
if (!uwsgi_calc_cheaper())
|
|
return 0;
|
|
}
|
|
|
|
// spooler cheap management
|
|
if (uwsgi.spooler_cheap) {
|
|
if ((uwsgi.master_cycles % uwsgi.spooler_frequency) == 0) {
|
|
uwsgi_spooler_cheap_check();
|
|
}
|
|
}
|
|
|
|
|
|
// check if someone is dead
|
|
diedpid = waitpid(WAIT_ANY, &waitpid_status, WNOHANG);
|
|
if (diedpid == -1) {
|
|
if (errno == ECHILD) {
|
|
// something did not work as expected, just assume all has been cleared
|
|
uwsgi_master_commit_status();
|
|
diedpid = 0;
|
|
}
|
|
else {
|
|
uwsgi_error("waitpid()");
|
|
/* here is better to reload all the uWSGI stack */
|
|
uwsgi_log("something horrible happened...\n");
|
|
reap_them_all(0);
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
// no one died just run all of the standard master tasks
|
|
if (diedpid == 0) {
|
|
|
|
/* all processes ok, doing status scan after N seconds */
|
|
check_interval = uwsgi.master_interval;
|
|
if (!check_interval) {
|
|
check_interval = 1;
|
|
uwsgi.master_interval = 1;
|
|
}
|
|
|
|
|
|
// add unregistered file monitors
|
|
// locking is not needed as monitors can only increase
|
|
for (i = 0; i < ushared->files_monitored_cnt; i++) {
|
|
if (!ushared->files_monitored[i].registered) {
|
|
ushared->files_monitored[i].fd = event_queue_add_file_monitor(uwsgi.master_queue, ushared->files_monitored[i].filename, &ushared->files_monitored[i].id);
|
|
ushared->files_monitored[i].registered = 1;
|
|
}
|
|
}
|
|
|
|
|
|
// add unregistered timers
|
|
// locking is not needed as timers can only increase
|
|
for (i = 0; i < ushared->timers_cnt; i++) {
|
|
if (!ushared->timers[i].registered) {
|
|
ushared->timers[i].fd = event_queue_add_timer(uwsgi.master_queue, &ushared->timers[i].id, ushared->timers[i].value);
|
|
ushared->timers[i].registered = 1;
|
|
}
|
|
}
|
|
|
|
// add unregistered rb_timers
|
|
// locking is not needed as rb_timers can only increase
|
|
for (i = 0; i < ushared->rb_timers_cnt; i++) {
|
|
if (!ushared->rb_timers[i].registered) {
|
|
ushared->rb_timers[i].uwsgi_rb_timer = uwsgi_add_rb_timer(rb_timers, uwsgi_now() + ushared->rb_timers[i].value, &ushared->rb_timers[i]);
|
|
ushared->rb_timers[i].registered = 1;
|
|
}
|
|
}
|
|
|
|
int interesting_fd = -1;
|
|
|
|
if (ushared->rb_timers_cnt > 0) {
|
|
min_timeout = uwsgi_min_rb_timer(rb_timers, NULL);
|
|
if (min_timeout) {
|
|
int delta = min_timeout->value - uwsgi_now();
|
|
if (delta <= 0) {
|
|
expire_rb_timeouts(rb_timers);
|
|
}
|
|
// if the timer expires before the check_interval, override it
|
|
else if (delta < check_interval) {
|
|
check_interval = delta;
|
|
}
|
|
}
|
|
}
|
|
|
|
// wait for event
|
|
rlen = event_queue_wait(uwsgi.master_queue, check_interval, &interesting_fd);
|
|
|
|
if (rlen == 0) {
|
|
if (ushared->rb_timers_cnt > 0) {
|
|
expire_rb_timeouts(rb_timers);
|
|
}
|
|
}
|
|
|
|
// update load counter
|
|
uwsgi_update_load_counters();
|
|
|
|
master_check_processes();
|
|
|
|
|
|
// check uwsgi-cron table
|
|
if (ushared->cron_cnt) {
|
|
uwsgi_manage_signal_cron(uwsgi_now());
|
|
}
|
|
|
|
if (uwsgi.crons) {
|
|
uwsgi_manage_command_cron(uwsgi_now());
|
|
}
|
|
|
|
// some event returned
|
|
if (rlen > 0) {
|
|
// if the following function returns -1, a new worker has just spawned
|
|
if (uwsgi_master_manage_events(interesting_fd)) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
now = uwsgi_now();
|
|
if (now - uwsgi.current_time < 1) {
|
|
continue;
|
|
}
|
|
uwsgi.current_time = now;
|
|
|
|
// checking logsize
|
|
if (uwsgi.logfile) {
|
|
uwsgi_check_logrotate();
|
|
}
|
|
|
|
// this will be incremented at (more or less) regular intervals
|
|
uwsgi.master_cycles++;
|
|
|
|
// recalculate requests counter on race conditions risky configurations
|
|
// a bit of inaccuracy is better than locking;)
|
|
uwsgi_master_fix_request_counters();
|
|
|
|
// check for idle
|
|
uwsgi_master_check_idle();
|
|
|
|
check_interval = uwsgi.master_interval;
|
|
if (!check_interval) {
|
|
check_interval = 1;
|
|
uwsgi.master_interval = 1;
|
|
}
|
|
|
|
|
|
// check listen_queue status
|
|
master_check_listen_queue();
|
|
|
|
int someone_killed = 0;
|
|
// check if some worker has to die (harakiri, evil checks...)
|
|
if (uwsgi_master_check_workers_deadline()) someone_killed++;
|
|
if (uwsgi_master_check_gateways_deadline()) someone_killed++;
|
|
if (uwsgi_master_check_mules_deadline()) someone_killed++;
|
|
if (uwsgi_master_check_spoolers_deadline()) someone_killed++;
|
|
if (uwsgi_master_check_crons_deadline()) someone_killed++;
|
|
|
|
// this could trigger a complete exit...
|
|
uwsgi_master_check_mountpoints();
|
|
|
|
#ifdef __linux__
|
|
#ifdef MADV_MERGEABLE
|
|
if (uwsgi.linux_ksm > 0 && (uwsgi.master_cycles % uwsgi.linux_ksm) == 0) {
|
|
uwsgi_linux_ksm_map();
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
// resubscribe every 10 cycles by default
|
|
if (((uwsgi.subscriptions || uwsgi.subscriptions2) && ((uwsgi.master_cycles % uwsgi.subscribe_freq) == 0 || uwsgi.master_cycles == 1)) && !uwsgi_instance_is_reloading && !uwsgi_instance_is_dying && !uwsgi.workers[0].suspended) {
|
|
uwsgi_subscribe_all(0, 0);
|
|
}
|
|
|
|
uwsgi_cache_sync_all();
|
|
|
|
if (uwsgi.queue_store && uwsgi.queue_filesize && uwsgi.queue_store_sync && ((uwsgi.master_cycles % uwsgi.queue_store_sync) == 0)) {
|
|
if (msync(uwsgi.queue_header, uwsgi.queue_filesize, MS_ASYNC)) {
|
|
uwsgi_error("msync()");
|
|
}
|
|
}
|
|
|
|
// check touch_reload
|
|
if (!uwsgi_instance_is_reloading && !uwsgi_instance_is_dying) {
|
|
char *touched = uwsgi_check_touches(uwsgi.touch_reload);
|
|
if (touched) {
|
|
uwsgi_log_verbose("*** %s has been touched... grace them all !!! ***\n", touched);
|
|
uwsgi_block_signal(SIGHUP);
|
|
grace_them_all(0);
|
|
uwsgi_unblock_signal(SIGHUP);
|
|
continue;
|
|
}
|
|
touched = uwsgi_check_touches(uwsgi.touch_workers_reload);
|
|
if (touched) {
|
|
uwsgi_log_verbose("*** %s has been touched... workers reload !!! ***\n", touched);
|
|
uwsgi_reload_workers();
|
|
continue;
|
|
}
|
|
touched = uwsgi_check_touches(uwsgi.touch_chain_reload);
|
|
if (touched) {
|
|
if (uwsgi.status.chain_reloading == 0) {
|
|
uwsgi_log_verbose("*** %s has been touched... chain reload !!! ***\n", touched);
|
|
uwsgi.status.chain_reloading = 1;
|
|
}
|
|
else {
|
|
uwsgi_log_verbose("*** %s has been touched... but chain reload is already running ***\n", touched);
|
|
}
|
|
}
|
|
|
|
// be sure to run it as the last touch check
|
|
touched = uwsgi_check_touches(uwsgi.touch_exec);
|
|
if (touched) {
|
|
if (uwsgi_run_command(touched, NULL, -1) >= 0) {
|
|
uwsgi_log_verbose("[uwsgi-touch-exec] running %s\n", touched);
|
|
}
|
|
}
|
|
touched = uwsgi_check_touches(uwsgi.touch_signal);
|
|
if (touched) {
|
|
uint8_t signum = atoi(touched);
|
|
uwsgi_route_signal(signum);
|
|
uwsgi_log_verbose("[uwsgi-touch-signal] raising %u\n", signum);
|
|
}
|
|
|
|
// daemon touches
|
|
struct uwsgi_daemon *ud = uwsgi.daemons;
|
|
while (ud) {
|
|
if (ud->pid > 0 && ud->touch) {
|
|
touched = uwsgi_check_touches(ud->touch);
|
|
if (touched) {
|
|
uwsgi_log_verbose("*** %s has been touched... reloading daemon \"%s\" (pid: %d) !!! ***\n", touched, ud->command, (int) ud->pid);
|
|
if (kill(-ud->pid, ud->stop_signal)) {
|
|
// killing process group failed, try to kill by process id
|
|
if (kill(ud->pid, ud->stop_signal)) {
|
|
uwsgi_error("[uwsgi-daemon/touch] kill()");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
ud = ud->next;
|
|
}
|
|
|
|
// hook touches
|
|
touched = uwsgi_check_touches(uwsgi.hook_touch);
|
|
if (touched) {
|
|
uwsgi_hooks_run((struct uwsgi_string_list *) touched, "touch", 0);
|
|
}
|
|
|
|
}
|
|
|
|
// allows the KILL signal to be delivered;
|
|
if (someone_killed > 0) sleep(1);
|
|
continue;
|
|
|
|
}
|
|
|
|
// no one died
|
|
if (diedpid <= 0)
|
|
continue;
|
|
|
|
// check for deadlocks first
|
|
uwsgi_deadlock_check(diedpid);
|
|
|
|
// reload gateways and daemons only on normal workflow (+outworld status)
|
|
if (!uwsgi_instance_is_reloading && !uwsgi_instance_is_dying) {
|
|
|
|
if (uwsgi_master_check_emperor_death(diedpid))
|
|
continue;
|
|
if (uwsgi_master_check_spoolers_death(diedpid))
|
|
continue;
|
|
if (uwsgi_master_check_mules_death(diedpid))
|
|
continue;
|
|
if (uwsgi_master_check_gateways_death(diedpid))
|
|
continue;
|
|
if (uwsgi_master_check_daemons_death(diedpid))
|
|
continue;
|
|
if (uwsgi_master_check_cron_death(diedpid))
|
|
continue;
|
|
}
|
|
|
|
|
|
/* What happens here ?
|
|
|
|
case 1) the diedpid is not a worker, report it and continue
|
|
case 2) the diedpid is a worker and we are not in a reload procedure -> reload it
|
|
case 3) the diedpid is a worker and we are in graceful reload -> uwsgi.ready_to_reload++ and continue
|
|
case 3) the diedpid is a worker and we are in brutal reload -> uwsgi.ready_to_die++ and continue
|
|
|
|
|
|
*/
|
|
|
|
int thewid = find_worker_id(diedpid);
|
|
if (thewid <= 0) {
|
|
// check spooler, mules, gateways and daemons
|
|
struct uwsgi_spooler *uspool = uwsgi.spoolers;
|
|
while (uspool) {
|
|
if (uspool->pid > 0 && diedpid == uspool->pid) {
|
|
uwsgi_log("spooler (pid: %d) annihilated\n", (int) diedpid);
|
|
goto next;
|
|
}
|
|
uspool = uspool->next;
|
|
}
|
|
|
|
for (i = 0; i < uwsgi.mules_cnt; i++) {
|
|
if (uwsgi.mules[i].pid == diedpid) {
|
|
uwsgi_log("mule %d (pid: %d) annihilated\n", i + 1, (int) diedpid);
|
|
uwsgi.mules[i].pid = 0;
|
|
goto next;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < ushared->gateways_cnt; i++) {
|
|
if (ushared->gateways[i].pid == diedpid) {
|
|
uwsgi_log("gateway %d (%s, pid: %d) annihilated\n", i + 1, ushared->gateways[i].fullname, (int) diedpid);
|
|
goto next;
|
|
}
|
|
}
|
|
|
|
if (uwsgi_daemon_check_pid_death(diedpid))
|
|
goto next;
|
|
|
|
if (WIFEXITED(waitpid_status)) {
|
|
uwsgi_log("subprocess %d exited with code %d\n", (int) diedpid, WEXITSTATUS(waitpid_status));
|
|
}
|
|
else if (WIFSIGNALED(waitpid_status)) {
|
|
uwsgi_log("subprocess %d exited by signal %d\n", (int) diedpid, WTERMSIG(waitpid_status));
|
|
}
|
|
else if (WIFSTOPPED(waitpid_status)) {
|
|
uwsgi_log("subprocess %d stopped\n", (int) diedpid);
|
|
}
|
|
next:
|
|
continue;
|
|
}
|
|
|
|
|
|
// ok a worker died...
|
|
uwsgi.workers[thewid].pid = 0;
|
|
// only to be safe :P
|
|
for(i=0;i<uwsgi.cores;i++) {
|
|
uwsgi.workers[thewid].cores[i].harakiri = 0;
|
|
}
|
|
|
|
// ok, if we are reloading or dying, just continue the master loop
|
|
// as soon as all of the workers have pid == 0, the action (exit, or reload) is triggered
|
|
if (uwsgi_instance_is_reloading || uwsgi_instance_is_dying) {
|
|
if (!uwsgi.workers[thewid].cursed_at)
|
|
uwsgi.workers[thewid].cursed_at = uwsgi_now();
|
|
uwsgi_log("worker %d buried after %d seconds\n", thewid, (int) (uwsgi_now() - uwsgi.workers[thewid].cursed_at));
|
|
uwsgi.workers[thewid].cursed_at = 0;
|
|
continue;
|
|
}
|
|
|
|
// if we are stopping workers, just end here
|
|
|
|
if (WIFEXITED(waitpid_status) && WEXITSTATUS(waitpid_status) == UWSGI_FAILED_APP_CODE) {
|
|
uwsgi_log("OOPS ! failed loading app in worker %d (pid %d) :( trying again...\n", thewid, (int) diedpid);
|
|
if (uwsgi.lazy_apps && uwsgi.need_app) {
|
|
uwsgi_log_verbose("need-app requested, destroying the instance...\n");
|
|
kill_them_all(0);
|
|
continue;
|
|
}
|
|
}
|
|
else if (WIFEXITED(waitpid_status) && WEXITSTATUS(waitpid_status) == UWSGI_DE_HIJACKED_CODE) {
|
|
uwsgi_log("...restoring worker %d (pid: %d)...\n", thewid, (int) diedpid);
|
|
}
|
|
else if (WIFEXITED(waitpid_status) && WEXITSTATUS(waitpid_status) == UWSGI_EXCEPTION_CODE) {
|
|
uwsgi_log("... monitored exception detected, respawning worker %d (pid: %d)...\n", thewid, (int) diedpid);
|
|
}
|
|
else if (WIFEXITED(waitpid_status) && WEXITSTATUS(waitpid_status) == UWSGI_QUIET_CODE) {
|
|
// noop
|
|
}
|
|
else if (WIFEXITED(waitpid_status) && WEXITSTATUS(waitpid_status) == UWSGI_BRUTAL_RELOAD_CODE) {
|
|
uwsgi_log("!!! inconsistent state reported by worker %d (pid: %d) !!!\n", thewid, (int) diedpid);
|
|
reap_them_all(0);
|
|
continue;
|
|
}
|
|
else if (WIFEXITED(waitpid_status) && WEXITSTATUS(waitpid_status) == UWSGI_GO_CHEAP_CODE) {
|
|
uwsgi_log("worker %d asked for cheap mode (pid: %d)...\n", thewid, (int) diedpid);
|
|
uwsgi.workers[thewid].cheaped = 1;
|
|
}
|
|
else if (uwsgi.workers[thewid].manage_next_request) {
|
|
if (WIFSIGNALED(waitpid_status)) {
|
|
uwsgi_log("DAMN ! worker %d (pid: %d) died, killed by signal %d :( trying respawn ...\n", thewid, (int) diedpid, (int) WTERMSIG(waitpid_status));
|
|
}
|
|
else {
|
|
uwsgi_log("DAMN ! worker %d (pid: %d) died :( trying respawn ...\n", thewid, (int) diedpid);
|
|
}
|
|
}
|
|
else if (uwsgi.workers[thewid].cursed_at > 0) {
|
|
uwsgi_log("worker %d killed successfully (pid: %d)\n", thewid, (int) diedpid);
|
|
}
|
|
// manage_next_request is zero, but killed by signal...
|
|
else if (WIFSIGNALED(waitpid_status)) {
|
|
uwsgi_log("DAMN ! worker %d (pid: %d) MISTERIOUSLY killed by signal %d :( trying respawn ...\n", thewid, (int) diedpid, (int) WTERMSIG(waitpid_status));
|
|
}
|
|
|
|
if (uwsgi.workers[thewid].cheaped == 1) {
|
|
uwsgi_log("uWSGI worker %d cheaped.\n", thewid);
|
|
continue;
|
|
}
|
|
|
|
// avoid fork bombing
|
|
gettimeofday(&last_respawn, NULL);
|
|
if (last_respawn.tv_sec <= uwsgi.respawn_delta + check_interval) {
|
|
last_respawn_rate++;
|
|
if (last_respawn_rate > uwsgi.numproc) {
|
|
if (uwsgi.forkbomb_delay > 0) {
|
|
uwsgi_log("worker respawning too fast !!! i have to sleep a bit (%d seconds)...\n", uwsgi.forkbomb_delay);
|
|
/* use --forkbomb-delay 0 to disable sleeping */
|
|
sleep(uwsgi.forkbomb_delay);
|
|
}
|
|
last_respawn_rate = 0;
|
|
}
|
|
}
|
|
else {
|
|
last_respawn_rate = 0;
|
|
}
|
|
gettimeofday(&last_respawn, NULL);
|
|
uwsgi.respawn_delta = last_respawn.tv_sec;
|
|
|
|
// are we chain reloading it ?
|
|
if (uwsgi.status.chain_reloading == thewid) {
|
|
uwsgi.status.chain_reloading++;
|
|
}
|
|
|
|
// respawn the worker (if needed)
|
|
if (uwsgi_respawn_worker(thewid))
|
|
return 0;
|
|
|
|
// end of the loop
|
|
}
|
|
|
|
// never here
|
|
}
|
|
|
|
void uwsgi_reload_workers() {
|
|
int i;
|
|
uwsgi_block_signal(SIGHUP);
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
if (uwsgi.workers[i].pid > 0) {
|
|
uwsgi_curse(i, SIGHUP);
|
|
}
|
|
}
|
|
uwsgi_unblock_signal(SIGHUP);
|
|
}
|
|
|
|
void uwsgi_chain_reload() {
|
|
if (!uwsgi.status.chain_reloading) {
|
|
uwsgi_log_verbose("chain reload starting...\n");
|
|
uwsgi.status.chain_reloading = 1;
|
|
}
|
|
else {
|
|
uwsgi_log_verbose("chain reload already running...\n");
|
|
}
|
|
}
|
|
|
|
void uwsgi_brutally_reload_workers() {
|
|
int i;
|
|
for (i = 1; i <= uwsgi.numproc; i++) {
|
|
if (uwsgi.workers[i].pid > 0) {
|
|
uwsgi_log_verbose("killing worker %d (pid: %d)\n", i, (int) uwsgi.workers[i].pid);
|
|
uwsgi_curse(i, SIGINT);
|
|
}
|
|
}
|
|
}
|