summaryrefslogtreecommitdiffhomepage
path: root/src/nxt_router.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/nxt_router.c297
1 files changed, 214 insertions, 83 deletions
diff --git a/src/nxt_router.c b/src/nxt_router.c
index 1f0f0b0f..e586f63a 100644
--- a/src/nxt_router.c
+++ b/src/nxt_router.c
@@ -13,6 +13,7 @@ typedef struct {
nxt_str_t type;
uint32_t workers;
nxt_msec_t timeout;
+ nxt_msec_t res_timeout;
uint32_t requests;
nxt_conf_value_t *limits_value;
} nxt_router_app_conf_t;
@@ -55,8 +56,11 @@ struct nxt_req_app_link_s {
nxt_msg_info_t msg_info;
nxt_req_conn_link_t *rc;
+ nxt_nsec_t res_time;
+
nxt_queue_link_t link_app_requests; /* for nxt_app_t.requests */
nxt_queue_link_t link_port_pending; /* for nxt_port_t.pending_requests */
+ nxt_queue_link_t link_app_pending; /* for nxt_app_t.pending */
nxt_mp_t *mem_pool;
nxt_work_t work;
@@ -72,6 +76,26 @@ typedef struct {
} nxt_socket_rpc_t;
+struct nxt_port_select_state_s {
+ nxt_app_t *app;
+ nxt_req_app_link_t *ra;
+
+ nxt_port_t *failed_port;
+ int failed_port_use_delta;
+
+ nxt_bool_t can_start_worker;
+ nxt_req_app_link_t *shared_ra;
+ nxt_port_t *port;
+};
+
+typedef struct nxt_port_select_state_s nxt_port_select_state_t;
+
+static void nxt_router_port_select(nxt_task_t *task,
+ nxt_port_select_state_t *state);
+
+static nxt_int_t nxt_router_port_post_select(nxt_task_t *task,
+ nxt_port_select_state_t *state);
+
static nxt_int_t nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app);
nxt_inline void
@@ -172,7 +196,8 @@ static void nxt_router_app_port_error(nxt_task_t *task,
static nxt_port_t * nxt_router_app_get_idle_port(nxt_app_t *app);
static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
uint32_t request_failed, uint32_t got_response);
-static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra);
+static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_app_t *app,
+ nxt_req_app_link_t *ra);
static void nxt_router_conn_init(nxt_task_t *task, void *obj, void *data);
static void nxt_router_conn_http_header_parse(nxt_task_t *task, void *obj,
@@ -575,6 +600,21 @@ nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char* str)
}
+nxt_inline void
+nxt_router_ra_pending(nxt_task_t *task, nxt_app_t *app, nxt_req_app_link_t *ra)
+{
+ nxt_queue_insert_tail(&ra->app_port->pending_requests,
+ &ra->link_port_pending);
+ nxt_queue_insert_tail(&app->pending, &ra->link_app_pending);
+
+ nxt_router_ra_inc_use(ra);
+
+ ra->res_time = nxt_thread_monotonic_time(task->thread) + app->res_timeout;
+
+ nxt_debug(task, "ra stream #%uD enqueue to pending_requests", ra->stream);
+}
+
+
nxt_inline nxt_bool_t
nxt_queue_chk_remove(nxt_queue_link_t *lnk)
{
@@ -615,13 +655,15 @@ nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
nxt_thread_mutex_lock(&rc->app->mutex);
if (ra->link_app_requests.next == NULL
- && ra->link_port_pending.next == NULL)
+ && ra->link_port_pending.next == NULL
+ && ra->link_app_pending.next == NULL)
{
ra = NULL;
} else {
ra_use_delta -= nxt_queue_chk_remove(&ra->link_app_requests);
ra_use_delta -= nxt_queue_chk_remove(&ra->link_port_pending);
+ nxt_queue_chk_remove(&ra->link_app_pending);
}
nxt_thread_mutex_unlock(&rc->app->mutex);
@@ -974,6 +1016,12 @@ static nxt_conf_map_t nxt_router_app_limits_conf[] = {
},
{
+ nxt_string("reschedule_timeout"),
+ NXT_CONF_MAP_MSEC,
+ offsetof(nxt_router_app_conf_t, res_timeout),
+ },
+
+ {
nxt_string("requests"),
NXT_CONF_MAP_INT32,
offsetof(nxt_router_app_conf_t, requests),
@@ -1127,6 +1175,7 @@ nxt_router_conf_create(nxt_task_t *task, nxt_router_temp_conf_t *tmcf,
apcf.workers = 1;
apcf.timeout = 0;
+ apcf.res_timeout = 1000;
apcf.requests = 0;
apcf.limits_value = NULL;
@@ -1156,7 +1205,8 @@ nxt_router_conf_create(nxt_task_t *task, nxt_router_temp_conf_t *tmcf,
nxt_debug(task, "application type: %V", &apcf.type);
nxt_debug(task, "application workers: %D", apcf.workers);
- nxt_debug(task, "application timeout: %D", apcf.timeout);
+ nxt_debug(task, "application request timeout: %D", apcf.timeout);
+ nxt_debug(task, "application reschedule timeout: %D", apcf.res_timeout);
nxt_debug(task, "application requests: %D", apcf.requests);
lang = nxt_app_lang_module(task->thread->runtime, &apcf.type);
@@ -1176,6 +1226,7 @@ nxt_router_conf_create(nxt_task_t *task, nxt_router_temp_conf_t *tmcf,
nxt_queue_init(&app->ports);
nxt_queue_init(&app->requests);
+ nxt_queue_init(&app->pending);
app->name.length = name.length;
nxt_memcpy(app->name.start, name.start, name.length);
@@ -1183,6 +1234,7 @@ nxt_router_conf_create(nxt_task_t *task, nxt_router_temp_conf_t *tmcf,
app->type = lang->type;
app->max_workers = apcf.workers;
app->timeout = apcf.timeout;
+ app->res_timeout = apcf.res_timeout * 1000000;
app->live = 1;
app->max_pending_responses = 2;
app->prepare_msg = nxt_app_prepare_msg[lang->type];
@@ -2481,7 +2533,7 @@ nxt_router_response_error_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg,
if (cancelled) {
nxt_router_ra_inc_use(ra);
- res = nxt_router_app_port(task, ra);
+ res = nxt_router_app_port(task, rc->app, ra);
if (res == NXT_OK) {
port = ra->app_port;
@@ -2702,7 +2754,7 @@ nxt_router_app_first_port_busy(nxt_app_t *app)
nxt_inline nxt_port_t *
-nxt_router_app_get_port_unsafe(nxt_app_t *app)
+nxt_router_pop_first_port(nxt_app_t *app)
{
nxt_port_t *port;
nxt_queue_link_t *lnk;
@@ -2783,14 +2835,17 @@ static void
nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
uint32_t request_failed, uint32_t got_response)
{
- nxt_app_t *app;
- nxt_bool_t send_quit;
- nxt_queue_link_t *lnk;
- nxt_req_app_link_t *ra, *next_ra;
+ nxt_app_t *app;
+ nxt_bool_t send_quit, cancelled;
+ nxt_queue_link_t *lnk;
+ nxt_req_app_link_t *ra, *pending_ra, *re_ra;
+ nxt_port_select_state_t state;
nxt_assert(port != NULL);
nxt_assert(port->app != NULL);
+ ra = NULL;
+
app = port->app;
nxt_thread_mutex_lock(&app->mutex);
@@ -2798,8 +2853,11 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
port->app_pending_responses -= request_failed + got_response;
port->app_responses += got_response;
- if (app->live != 0
- && port->pair[1] != -1
+ if (nxt_slow_path(app->live == 0)) {
+ goto app_dead;
+ }
+
+ if (port->pair[1] != -1
&& (app->max_pending_responses == 0
|| port->app_pending_responses < app->max_pending_responses))
{
@@ -2823,8 +2881,7 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
}
}
- if (app->live != 0
- && !nxt_queue_is_empty(&app->ports)
+ if (!nxt_queue_is_empty(&app->ports)
&& !nxt_queue_is_empty(&app->requests))
{
lnk = nxt_queue_first(&app->requests);
@@ -2833,19 +2890,16 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, link_app_requests);
- ra->app_port = nxt_router_app_get_port_unsafe(app);
+ ra->app_port = nxt_router_pop_first_port(app);
if (ra->app_port->app_pending_responses > 1) {
- nxt_queue_insert_tail(&ra->app_port->pending_requests,
- &ra->link_port_pending);
-
- nxt_router_ra_inc_use(ra);
+ nxt_router_ra_pending(task, app, ra);
}
-
- } else {
- ra = NULL;
}
+app_dead:
+
+ /* Pop first pending request for this port. */
if ((request_failed > 0 || got_response > 0)
&& !nxt_queue_is_empty(&port->pending_requests))
{
@@ -2853,19 +2907,63 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
nxt_queue_remove(lnk);
lnk->next = NULL;
- next_ra = nxt_queue_link_data(lnk, nxt_req_app_link_t,
- link_port_pending);
+ pending_ra = nxt_queue_link_data(lnk, nxt_req_app_link_t,
+ link_port_pending);
+
+ nxt_assert(pending_ra->link_app_pending.next != NULL);
+
+ nxt_queue_remove(&pending_ra->link_app_pending);
+ pending_ra->link_app_pending.next = NULL;
} else {
- next_ra = NULL;
+ pending_ra = NULL;
}
+ /* Try to cancel and re-schedule first stalled request for this app. */
+ if (got_response > 0 && !nxt_queue_is_empty(&app->pending)) {
+ lnk = nxt_queue_first(&app->pending);
+
+ re_ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, link_app_pending);
+
+ if (re_ra->res_time <= nxt_thread_monotonic_time(task->thread)) {
+
+ nxt_debug(task, "app '%V' stalled request #%uD detected",
+ &app->name, re_ra->stream);
+
+ cancelled = nxt_router_msg_cancel(task, &re_ra->msg_info,
+ re_ra->stream);
+
+ if (cancelled) {
+ nxt_router_ra_inc_use(re_ra);
+
+ state.ra = re_ra;
+ state.app = app;
+
+ nxt_router_port_select(task, &state);
+
+ goto re_ra_cancelled;
+ }
+ }
+ }
+
+ re_ra = NULL;
+
+re_ra_cancelled:
+
send_quit = app->live == 0 && port->app_pending_responses > 0;
nxt_thread_mutex_unlock(&app->mutex);
- if (next_ra != NULL) {
- nxt_router_ra_use(task, next_ra, -1);
+ if (pending_ra != NULL) {
+ nxt_router_ra_use(task, pending_ra, -1);
+ }
+
+ if (re_ra != NULL) {
+ if (nxt_router_port_post_select(task, &state) == NXT_OK) {
+ nxt_work_queue_add(&task->thread->engine->fast_work_queue,
+ nxt_router_app_process_request,
+ &task->thread->engine->task, app, re_ra);
+ }
}
if (ra != NULL) {
@@ -2950,22 +3048,16 @@ nxt_router_app_port_close(nxt_task_t *task, nxt_port_t *port)
}
-static nxt_int_t
-nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra)
+static void
+nxt_router_port_select(nxt_task_t *task, nxt_port_select_state_t *state)
{
- int failed_port_use_delta;
- nxt_int_t res;
- nxt_app_t *app;
- nxt_bool_t can_start_worker;
- nxt_conn_t *c;
- nxt_port_t *port, *failed_port;
-
- c = ra->rc->conn;
- app = ra->rc->app;
+ nxt_app_t *app;
+ nxt_req_app_link_t *ra;
- failed_port_use_delta = 0;
+ ra = state->ra;
+ app = state->app;
- nxt_thread_mutex_lock(&app->mutex);
+ state->failed_port_use_delta = 0;
if (nxt_queue_chk_remove(&ra->link_app_requests))
{
@@ -2974,93 +3066,113 @@ nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra)
if (nxt_queue_chk_remove(&ra->link_port_pending))
{
+ nxt_assert(ra->link_app_pending.next != NULL);
+
+ nxt_queue_remove(&ra->link_app_pending);
+ ra->link_app_pending.next = NULL;
+
nxt_router_ra_dec_use(ra);
}
- if (ra->app_port != NULL) {
- failed_port = ra->app_port;
- failed_port_use_delta--;
+ state->failed_port = ra->app_port;
- failed_port->app_pending_responses--;
+ if (ra->app_port != NULL) {
+ state->failed_port_use_delta--;
- if (failed_port->app_link.next != NULL) {
- nxt_queue_remove(&failed_port->app_link);
- failed_port->app_link.next = NULL;
+ state->failed_port->app_pending_responses--;
- failed_port_use_delta--;
+ if (nxt_queue_chk_remove(&state->failed_port->app_link)) {
+ state->failed_port_use_delta--;
}
- } else {
- failed_port = NULL;
+ ra->app_port = NULL;
}
- can_start_worker = (app->workers + app->pending_workers)
- < app->max_workers;
+ state->can_start_worker = (app->workers + app->pending_workers)
+ < app->max_workers;
+ state->port = NULL;
if (nxt_queue_is_empty(&app->ports)
- || (can_start_worker && nxt_router_app_first_port_busy(app)) )
+ || (state->can_start_worker && nxt_router_app_first_port_busy(app)) )
{
ra = nxt_router_ra_create(task, ra);
- if (nxt_fast_path(ra != NULL)) {
+ if (nxt_slow_path(ra == NULL)) {
+ goto fail;
+ }
+
+ if (nxt_slow_path(state->failed_port != NULL)) {
+ nxt_queue_insert_head(&app->requests, &ra->link_app_requests);
+
+ } else {
nxt_queue_insert_tail(&app->requests, &ra->link_app_requests);
+ }
- nxt_router_ra_inc_use(ra);
+ nxt_router_ra_inc_use(ra);
- nxt_debug(task, "ra stream #%uD enqueue to app->requests",
- ra->stream);
+ nxt_debug(task, "ra stream #%uD enqueue to app->requests", ra->stream);
- if (can_start_worker) {
- app->pending_workers++;
- }
+ if (state->can_start_worker) {
+ app->pending_workers++;
}
- port = NULL;
-
} else {
- port = nxt_router_app_get_port_unsafe(app);
+ state->port = nxt_router_pop_first_port(app);
- if (port->app_pending_responses > 1) {
+ if (state->port->app_pending_responses > 1) {
ra = nxt_router_ra_create(task, ra);
- if (nxt_fast_path(ra != NULL)) {
- nxt_queue_insert_tail(&port->pending_requests,
- &ra->link_port_pending);
+ if (nxt_slow_path(ra == NULL)) {
+ goto fail;
+ }
- nxt_router_ra_inc_use(ra);
+ ra->app_port = state->port;
- nxt_debug(task, "ra stream #%uD enqueue to "
- "port->pending_requests", ra->stream);
- }
+ nxt_router_ra_pending(task, app, ra);
}
}
- nxt_thread_mutex_unlock(&app->mutex);
+fail:
+
+ state->shared_ra = ra;
+}
+
+
+static nxt_int_t
+nxt_router_port_post_select(nxt_task_t *task, nxt_port_select_state_t *state)
+{
+ nxt_int_t res;
+ nxt_app_t *app;
+ nxt_req_app_link_t *ra;
- if (failed_port_use_delta != 0) {
- nxt_port_use(task, failed_port, failed_port_use_delta);
+ ra = state->shared_ra;
+ app = state->app;
+
+ if (state->failed_port_use_delta != 0) {
+ nxt_port_use(task, state->failed_port, state->failed_port_use_delta);
}
if (nxt_slow_path(ra == NULL)) {
- nxt_router_gen_error(task, c, 500, "Failed to allocate "
- "req<->app link");
-
- if (port != NULL) {
- nxt_port_use(task, port, -1);
+ if (state->port != NULL) {
+ nxt_port_use(task, state->port, -1);
}
+ nxt_router_ra_error(state->ra, 500,
+ "Failed to allocate shared req<->app link");
+ nxt_router_ra_use(task, state->ra, -1);
+
return NXT_ERROR;
}
- if (port != NULL) {
+ if (state->port != NULL) {
nxt_debug(task, "already have port for app '%V' %p ", &app->name, app);
- ra->app_port = port;
+ ra->app_port = state->port;
return NXT_OK;
}
- if (!can_start_worker) {
+ if (!state->can_start_worker) {
nxt_debug(task, "app '%V' %p too many running or pending workers",
&app->name, app);
@@ -3070,7 +3182,8 @@ nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra)
res = nxt_router_start_worker(task, app);
if (nxt_slow_path(res != NXT_OK)) {
- nxt_router_gen_error(task, c, 500, "Failed to start worker");
+ nxt_router_ra_error(ra, 500, "Failed to start worker");
+ nxt_router_ra_use(task, ra, -1);
return NXT_ERROR;
}
@@ -3079,6 +3192,24 @@ nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra)
}
+static nxt_int_t
+nxt_router_app_port(nxt_task_t *task, nxt_app_t *app, nxt_req_app_link_t *ra)
+{
+ nxt_port_select_state_t state;
+
+ state.ra = ra;
+ state.app = app;
+
+ nxt_thread_mutex_lock(&app->mutex);
+
+ nxt_router_port_select(task, &state);
+
+ nxt_thread_mutex_unlock(&app->mutex);
+
+ return nxt_router_port_post_select(task, &state);
+}
+
+
static void
nxt_router_conn_http_header_parse(nxt_task_t *task, void *obj, void *data)
{
@@ -3392,7 +3523,7 @@ nxt_router_process_http_request(nxt_task_t *task, nxt_conn_t *c,
ra = &ra_local;
nxt_router_ra_init(task, ra, rc);
- res = nxt_router_app_port(task, ra);
+ res = nxt_router_app_port(task, app, ra);
if (res != NXT_OK) {
return;