1e18a033bSKonstantin Ananyev
2e18a033bSKonstantin Ananyev/*
3e18a033bSKonstantin Ananyev * Copyright (C) Igor Sysoev
4e18a033bSKonstantin Ananyev * Copyright (C) Nginx, Inc.
5e18a033bSKonstantin Ananyev */
6e18a033bSKonstantin Ananyev
7e18a033bSKonstantin Ananyev
8e18a033bSKonstantin Ananyev#include <ngx_config.h>
9e18a033bSKonstantin Ananyev#include <ngx_core.h>
10e18a033bSKonstantin Ananyev#include <ngx_stream.h>
11e18a033bSKonstantin Ananyev
12e18a033bSKonstantin Ananyev
13e18a033bSKonstantin Ananyevstatic ngx_int_t ngx_stream_upstream_add_variables(ngx_conf_t *cf);
14e18a033bSKonstantin Ananyevstatic ngx_int_t ngx_stream_upstream_addr_variable(ngx_stream_session_t *s,
15e18a033bSKonstantin Ananyev    ngx_stream_variable_value_t *v, uintptr_t data);
16e18a033bSKonstantin Ananyevstatic ngx_int_t ngx_stream_upstream_response_time_variable(
17e18a033bSKonstantin Ananyev    ngx_stream_session_t *s, ngx_stream_variable_value_t *v, uintptr_t data);
18e18a033bSKonstantin Ananyevstatic ngx_int_t ngx_stream_upstream_bytes_variable(ngx_stream_session_t *s,
19e18a033bSKonstantin Ananyev    ngx_stream_variable_value_t *v, uintptr_t data);
20e18a033bSKonstantin Ananyev
21e18a033bSKonstantin Ananyevstatic char *ngx_stream_upstream(ngx_conf_t *cf, ngx_command_t *cmd,
22e18a033bSKonstantin Ananyev    void *dummy);
23e18a033bSKonstantin Ananyevstatic char *ngx_stream_upstream_server(ngx_conf_t *cf, ngx_command_t *cmd,
24e18a033bSKonstantin Ananyev    void *conf);
25e18a033bSKonstantin Ananyevstatic void *ngx_stream_upstream_create_main_conf(ngx_conf_t *cf);
26e18a033bSKonstantin Ananyevstatic char *ngx_stream_upstream_init_main_conf(ngx_conf_t *cf, void *conf);
27e18a033bSKonstantin Ananyev
28e18a033bSKonstantin Ananyev
29e18a033bSKonstantin Ananyevstatic ngx_command_t  ngx_stream_upstream_commands[] = {
30e18a033bSKonstantin Ananyev
31e18a033bSKonstantin Ananyev    { ngx_string("upstream"),
32e18a033bSKonstantin Ananyev      NGX_STREAM_MAIN_CONF|NGX_CONF_BLOCK|NGX_CONF_TAKE1,
33e18a033bSKonstantin Ananyev      ngx_stream_upstream,
34e18a033bSKonstantin Ananyev      0,
35e18a033bSKonstantin Ananyev      0,
36e18a033bSKonstantin Ananyev      NULL },
37e18a033bSKonstantin Ananyev
38e18a033bSKonstantin Ananyev    { ngx_string("server"),
39e18a033bSKonstantin Ananyev      NGX_STREAM_UPS_CONF|NGX_CONF_1MORE,
40e18a033bSKonstantin Ananyev      ngx_stream_upstream_server,
41e18a033bSKonstantin Ananyev      NGX_STREAM_SRV_CONF_OFFSET,
42e18a033bSKonstantin Ananyev      0,
43e18a033bSKonstantin Ananyev      NULL },
44e18a033bSKonstantin Ananyev
45e18a033bSKonstantin Ananyev      ngx_null_command
46e18a033bSKonstantin Ananyev};
47e18a033bSKonstantin Ananyev
48e18a033bSKonstantin Ananyev
49e18a033bSKonstantin Ananyevstatic ngx_stream_module_t  ngx_stream_upstream_module_ctx = {
50e18a033bSKonstantin Ananyev    ngx_stream_upstream_add_variables,     /* preconfiguration */
51e18a033bSKonstantin Ananyev    NULL,                                  /* postconfiguration */
52e18a033bSKonstantin Ananyev
53e18a033bSKonstantin Ananyev    ngx_stream_upstream_create_main_conf,  /* create main configuration */
54e18a033bSKonstantin Ananyev    ngx_stream_upstream_init_main_conf,    /* init main configuration */
55e18a033bSKonstantin Ananyev
56e18a033bSKonstantin Ananyev    NULL,                                  /* create server configuration */
57e18a033bSKonstantin Ananyev    NULL                                   /* merge server configuration */
58e18a033bSKonstantin Ananyev};
59e18a033bSKonstantin Ananyev
60e18a033bSKonstantin Ananyev
61e18a033bSKonstantin Ananyevngx_module_t  ngx_stream_upstream_module = {
62e18a033bSKonstantin Ananyev    NGX_MODULE_V1,
63e18a033bSKonstantin Ananyev    &ngx_stream_upstream_module_ctx,       /* module context */
64e18a033bSKonstantin Ananyev    ngx_stream_upstream_commands,          /* module directives */
65e18a033bSKonstantin Ananyev    NGX_STREAM_MODULE,                     /* module type */
66e18a033bSKonstantin Ananyev    NULL,                                  /* init master */
67e18a033bSKonstantin Ananyev    NULL,                                  /* init module */
68e18a033bSKonstantin Ananyev    NULL,                                  /* init process */
69e18a033bSKonstantin Ananyev    NULL,                                  /* init thread */
70e18a033bSKonstantin Ananyev    NULL,                                  /* exit thread */
71e18a033bSKonstantin Ananyev    NULL,                                  /* exit process */
72e18a033bSKonstantin Ananyev    NULL,                                  /* exit master */
73e18a033bSKonstantin Ananyev    NGX_MODULE_V1_PADDING
74e18a033bSKonstantin Ananyev};
75e18a033bSKonstantin Ananyev
76e18a033bSKonstantin Ananyev
77e18a033bSKonstantin Ananyevstatic ngx_stream_variable_t  ngx_stream_upstream_vars[] = {
78e18a033bSKonstantin Ananyev
79e18a033bSKonstantin Ananyev    { ngx_string("upstream_addr"), NULL,
80e18a033bSKonstantin Ananyev      ngx_stream_upstream_addr_variable, 0,
81e18a033bSKonstantin Ananyev      NGX_STREAM_VAR_NOCACHEABLE, 0 },
82e18a033bSKonstantin Ananyev
83e18a033bSKonstantin Ananyev    { ngx_string("upstream_bytes_sent"), NULL,
84e18a033bSKonstantin Ananyev      ngx_stream_upstream_bytes_variable, 0,
85e18a033bSKonstantin Ananyev      NGX_STREAM_VAR_NOCACHEABLE, 0 },
86e18a033bSKonstantin Ananyev
87e18a033bSKonstantin Ananyev    { ngx_string("upstream_connect_time"), NULL,
88e18a033bSKonstantin Ananyev      ngx_stream_upstream_response_time_variable, 2,
89e18a033bSKonstantin Ananyev      NGX_STREAM_VAR_NOCACHEABLE, 0 },
90e18a033bSKonstantin Ananyev
91e18a033bSKonstantin Ananyev    { ngx_string("upstream_first_byte_time"), NULL,
92e18a033bSKonstantin Ananyev      ngx_stream_upstream_response_time_variable, 1,
93e18a033bSKonstantin Ananyev      NGX_STREAM_VAR_NOCACHEABLE, 0 },
94e18a033bSKonstantin Ananyev
95e18a033bSKonstantin Ananyev    { ngx_string("upstream_session_time"), NULL,
96e18a033bSKonstantin Ananyev      ngx_stream_upstream_response_time_variable, 0,
97e18a033bSKonstantin Ananyev      NGX_STREAM_VAR_NOCACHEABLE, 0 },
98e18a033bSKonstantin Ananyev
99e18a033bSKonstantin Ananyev    { ngx_string("upstream_bytes_received"), NULL,
100e18a033bSKonstantin Ananyev      ngx_stream_upstream_bytes_variable, 1,
101e18a033bSKonstantin Ananyev      NGX_STREAM_VAR_NOCACHEABLE, 0 },
102e18a033bSKonstantin Ananyev
103e18a033bSKonstantin Ananyev    { ngx_null_string, NULL, NULL, 0, 0, 0 }
104e18a033bSKonstantin Ananyev};
105e18a033bSKonstantin Ananyev
106e18a033bSKonstantin Ananyev
107e18a033bSKonstantin Ananyevstatic ngx_int_t
108e18a033bSKonstantin Ananyevngx_stream_upstream_add_variables(ngx_conf_t *cf)
109e18a033bSKonstantin Ananyev{
110e18a033bSKonstantin Ananyev    ngx_stream_variable_t  *var, *v;
111e18a033bSKonstantin Ananyev
112e18a033bSKonstantin Ananyev    for (v = ngx_stream_upstream_vars; v->name.len; v++) {
113e18a033bSKonstantin Ananyev        var = ngx_stream_add_variable(cf, &v->name, v->flags);
114e18a033bSKonstantin Ananyev        if (var == NULL) {
115e18a033bSKonstantin Ananyev            return NGX_ERROR;
116e18a033bSKonstantin Ananyev        }
117e18a033bSKonstantin Ananyev
118e18a033bSKonstantin Ananyev        var->get_handler = v->get_handler;
119e18a033bSKonstantin Ananyev        var->data = v->data;
120e18a033bSKonstantin Ananyev    }
121e18a033bSKonstantin Ananyev
122e18a033bSKonstantin Ananyev    return NGX_OK;
123e18a033bSKonstantin Ananyev}
124e18a033bSKonstantin Ananyev
125e18a033bSKonstantin Ananyev
126e18a033bSKonstantin Ananyevstatic ngx_int_t
127e18a033bSKonstantin Ananyevngx_stream_upstream_addr_variable(ngx_stream_session_t *s,
128e18a033bSKonstantin Ananyev    ngx_stream_variable_value_t *v, uintptr_t data)
129e18a033bSKonstantin Ananyev{
130e18a033bSKonstantin Ananyev    u_char                       *p;
131e18a033bSKonstantin Ananyev    size_t                        len;
132e18a033bSKonstantin Ananyev    ngx_uint_t                    i;
133e18a033bSKonstantin Ananyev    ngx_stream_upstream_state_t  *state;
134e18a033bSKonstantin Ananyev
135e18a033bSKonstantin Ananyev    v->valid = 1;
136e18a033bSKonstantin Ananyev    v->no_cacheable = 0;
137e18a033bSKonstantin Ananyev    v->not_found = 0;
138e18a033bSKonstantin Ananyev
139e18a033bSKonstantin Ananyev    if (s->upstream_states == NULL || s->upstream_states->nelts == 0) {
140e18a033bSKonstantin Ananyev        v->not_found = 1;
141e18a033bSKonstantin Ananyev        return NGX_OK;
142e18a033bSKonstantin Ananyev    }
143e18a033bSKonstantin Ananyev
144e18a033bSKonstantin Ananyev    len = 0;
145e18a033bSKonstantin Ananyev    state = s->upstream_states->elts;
146e18a033bSKonstantin Ananyev
147e18a033bSKonstantin Ananyev    for (i = 0; i < s->upstream_states->nelts; i++) {
148e18a033bSKonstantin Ananyev        if (state[i].peer) {
149e18a033bSKonstantin Ananyev            len += state[i].peer->len;
150e18a033bSKonstantin Ananyev        }
151e18a033bSKonstantin Ananyev
152e18a033bSKonstantin Ananyev        len += 2;
153e18a033bSKonstantin Ananyev    }
154e18a033bSKonstantin Ananyev
155e18a033bSKonstantin Ananyev    p = ngx_pnalloc(s->connection->pool, len);
156e18a033bSKonstantin Ananyev    if (p == NULL) {
157e18a033bSKonstantin Ananyev        return NGX_ERROR;
158e18a033bSKonstantin Ananyev    }
159e18a033bSKonstantin Ananyev
160e18a033bSKonstantin Ananyev    v->data = p;
161e18a033bSKonstantin Ananyev
162e18a033bSKonstantin Ananyev    i = 0;
163e18a033bSKonstantin Ananyev
164e18a033bSKonstantin Ananyev    for ( ;; ) {
165e18a033bSKonstantin Ananyev        if (state[i].peer) {
166e18a033bSKonstantin Ananyev            p = ngx_cpymem(p, state[i].peer->data, state[i].peer->len);
167e18a033bSKonstantin Ananyev        }
168e18a033bSKonstantin Ananyev
169e18a033bSKonstantin Ananyev        if (++i == s->upstream_states->nelts) {
170e18a033bSKonstantin Ananyev            break;
171e18a033bSKonstantin Ananyev        }
172e18a033bSKonstantin Ananyev
173e18a033bSKonstantin Ananyev        *p++ = ',';
174e18a033bSKonstantin Ananyev        *p++ = ' ';
175e18a033bSKonstantin Ananyev    }
176e18a033bSKonstantin Ananyev
177e18a033bSKonstantin Ananyev    v->len = p - v->data;
178e18a033bSKonstantin Ananyev
179e18a033bSKonstantin Ananyev    return NGX_OK;
180e18a033bSKonstantin Ananyev}
181e18a033bSKonstantin Ananyev
182e18a033bSKonstantin Ananyev
183e18a033bSKonstantin Ananyevstatic ngx_int_t
184e18a033bSKonstantin Ananyevngx_stream_upstream_bytes_variable(ngx_stream_session_t *s,
185e18a033bSKonstantin Ananyev    ngx_stream_variable_value_t *v, uintptr_t data)
186e18a033bSKonstantin Ananyev{
187e18a033bSKonstantin Ananyev    u_char                       *p;
188e18a033bSKonstantin Ananyev    size_t                        len;
189e18a033bSKonstantin Ananyev    ngx_uint_t                    i;
190e18a033bSKonstantin Ananyev    ngx_stream_upstream_state_t  *state;
191e18a033bSKonstantin Ananyev
192e18a033bSKonstantin Ananyev    v->valid = 1;
193e18a033bSKonstantin Ananyev    v->no_cacheable = 0;
194e18a033bSKonstantin Ananyev    v->not_found = 0;
195e18a033bSKonstantin Ananyev
196e18a033bSKonstantin Ananyev    if (s->upstream_states == NULL || s->upstream_states->nelts == 0) {
197e18a033bSKonstantin Ananyev        v->not_found = 1;
198e18a033bSKonstantin Ananyev        return NGX_OK;
199e18a033bSKonstantin Ananyev    }
200e18a033bSKonstantin Ananyev
201e18a033bSKonstantin Ananyev    len = s->upstream_states->nelts * (NGX_OFF_T_LEN + 2);
202e18a033bSKonstantin Ananyev
203e18a033bSKonstantin Ananyev    p = ngx_pnalloc(s->connection->pool, len);
204e18a033bSKonstantin Ananyev    if (p == NULL) {
205e18a033bSKonstantin Ananyev        return NGX_ERROR;
206e18a033bSKonstantin Ananyev    }
207e18a033bSKonstantin Ananyev
208e18a033bSKonstantin Ananyev    v->data = p;
209e18a033bSKonstantin Ananyev
210e18a033bSKonstantin Ananyev    i = 0;
211e18a033bSKonstantin Ananyev    state = s->upstream_states->elts;
212e18a033bSKonstantin Ananyev
213e18a033bSKonstantin Ananyev    for ( ;; ) {
214e18a033bSKonstantin Ananyev
215e18a033bSKonstantin Ananyev        if (data == 1) {
216e18a033bSKonstantin Ananyev            p = ngx_sprintf(p, "%O", state[i].bytes_received);
217e18a033bSKonstantin Ananyev
218e18a033bSKonstantin Ananyev        } else {
219e18a033bSKonstantin Ananyev            p = ngx_sprintf(p, "%O", state[i].bytes_sent);
220e18a033bSKonstantin Ananyev        }
221e18a033bSKonstantin Ananyev
222e18a033bSKonstantin Ananyev        if (++i == s->upstream_states->nelts) {
223e18a033bSKonstantin Ananyev            break;
224e18a033bSKonstantin Ananyev        }
225e18a033bSKonstantin Ananyev
226e18a033bSKonstantin Ananyev        *p++ = ',';
227e18a033bSKonstantin Ananyev        *p++ = ' ';
228e18a033bSKonstantin Ananyev    }
229e18a033bSKonstantin Ananyev
230e18a033bSKonstantin Ananyev    v->len = p - v->data;
231e18a033bSKonstantin Ananyev
232e18a033bSKonstantin Ananyev    return NGX_OK;
233e18a033bSKonstantin Ananyev}
234e18a033bSKonstantin Ananyev
235e18a033bSKonstantin Ananyev
236e18a033bSKonstantin Ananyevstatic ngx_int_t
237e18a033bSKonstantin Ananyevngx_stream_upstream_response_time_variable(ngx_stream_session_t *s,
238e18a033bSKonstantin Ananyev    ngx_stream_variable_value_t *v, uintptr_t data)
239e18a033bSKonstantin Ananyev{
240e18a033bSKonstantin Ananyev    u_char                       *p;
241e18a033bSKonstantin Ananyev    size_t                        len;
242e18a033bSKonstantin Ananyev    ngx_uint_t                    i;
243e18a033bSKonstantin Ananyev    ngx_msec_int_t                ms;
244e18a033bSKonstantin Ananyev    ngx_stream_upstream_state_t  *state;
245e18a033bSKonstantin Ananyev
246e18a033bSKonstantin Ananyev    v->valid = 1;
247e18a033bSKonstantin Ananyev    v->no_cacheable = 0;
248e18a033bSKonstantin Ananyev    v->not_found = 0;
249e18a033bSKonstantin Ananyev
250e18a033bSKonstantin Ananyev    if (s->upstream_states == NULL || s->upstream_states->nelts == 0) {
251e18a033bSKonstantin Ananyev        v->not_found = 1;
252e18a033bSKonstantin Ananyev        return NGX_OK;
253e18a033bSKonstantin Ananyev    }
254e18a033bSKonstantin Ananyev
255e18a033bSKonstantin Ananyev    len = s->upstream_states->nelts * (NGX_TIME_T_LEN + 4 + 2);
256e18a033bSKonstantin Ananyev
257e18a033bSKonstantin Ananyev    p = ngx_pnalloc(s->connection->pool, len);
258e18a033bSKonstantin Ananyev    if (p == NULL) {
259e18a033bSKonstantin Ananyev        return NGX_ERROR;
260e18a033bSKonstantin Ananyev    }
261e18a033bSKonstantin Ananyev
262e18a033bSKonstantin Ananyev    v->data = p;
263e18a033bSKonstantin Ananyev
264e18a033bSKonstantin Ananyev    i = 0;
265e18a033bSKonstantin Ananyev    state = s->upstream_states->elts;
266e18a033bSKonstantin Ananyev
267e18a033bSKonstantin Ananyev    for ( ;; ) {
268e18a033bSKonstantin Ananyev
269e18a033bSKonstantin Ananyev        if (data == 1) {
270e18a033bSKonstantin Ananyev            if (state[i].first_byte_time == (ngx_msec_t) -1) {
271e18a033bSKonstantin Ananyev                *p++ = '-';
272e18a033bSKonstantin Ananyev                goto next;
273e18a033bSKonstantin Ananyev            }
274e18a033bSKonstantin Ananyev
275e18a033bSKonstantin Ananyev            ms = state[i].first_byte_time;
276e18a033bSKonstantin Ananyev
277e18a033bSKonstantin Ananyev        } else if (data == 2 && state[i].connect_time != (ngx_msec_t) -1) {
278e18a033bSKonstantin Ananyev            ms = state[i].connect_time;
279e18a033bSKonstantin Ananyev
280e18a033bSKonstantin Ananyev        } else {
281e18a033bSKonstantin Ananyev            ms = state[i].response_time;
282e18a033bSKonstantin Ananyev        }
283e18a033bSKonstantin Ananyev
284e18a033bSKonstantin Ananyev        ms = ngx_max(ms, 0);
285e18a033bSKonstantin Ananyev        p = ngx_sprintf(p, "%T.%03M", (time_t) ms / 1000, ms % 1000);
286e18a033bSKonstantin Ananyev
287e18a033bSKonstantin Ananyev    next:
288e18a033bSKonstantin Ananyev
289e18a033bSKonstantin Ananyev        if (++i == s->upstream_states->nelts) {
290e18a033bSKonstantin Ananyev            break;
291e18a033bSKonstantin Ananyev        }
292e18a033bSKonstantin Ananyev
293e18a033bSKonstantin Ananyev        *p++ = ',';
294e18a033bSKonstantin Ananyev        *p++ = ' ';
295e18a033bSKonstantin Ananyev    }
296e18a033bSKonstantin Ananyev
297e18a033bSKonstantin Ananyev    v->len = p - v->data;
298e18a033bSKonstantin Ananyev
299e18a033bSKonstantin Ananyev    return NGX_OK;
300e18a033bSKonstantin Ananyev}
301e18a033bSKonstantin Ananyev
302e18a033bSKonstantin Ananyev
303e18a033bSKonstantin Ananyevstatic char *
304e18a033bSKonstantin Ananyevngx_stream_upstream(ngx_conf_t *cf, ngx_command_t *cmd, void *dummy)
305e18a033bSKonstantin Ananyev{
306e18a033bSKonstantin Ananyev    char                            *rv;
307e18a033bSKonstantin Ananyev    void                            *mconf;
308e18a033bSKonstantin Ananyev    ngx_str_t                       *value;
309e18a033bSKonstantin Ananyev    ngx_url_t                        u;
310e18a033bSKonstantin Ananyev    ngx_uint_t                       m;
311e18a033bSKonstantin Ananyev    ngx_conf_t                       pcf;
312e18a033bSKonstantin Ananyev    ngx_stream_module_t             *module;
313e18a033bSKonstantin Ananyev    ngx_stream_conf_ctx_t           *ctx, *stream_ctx;
314e18a033bSKonstantin Ananyev    ngx_stream_upstream_srv_conf_t  *uscf;
315e18a033bSKonstantin Ananyev
316e18a033bSKonstantin Ananyev    ngx_memzero(&u, sizeof(ngx_url_t));
317e18a033bSKonstantin Ananyev
318e18a033bSKonstantin Ananyev    value = cf->args->elts;
319e18a033bSKonstantin Ananyev    u.host = value[1];
320e18a033bSKonstantin Ananyev    u.no_resolve = 1;
321e18a033bSKonstantin Ananyev    u.no_port = 1;
322e18a033bSKonstantin Ananyev
323e18a033bSKonstantin Ananyev    uscf = ngx_stream_upstream_add(cf, &u, NGX_STREAM_UPSTREAM_CREATE
324e18a033bSKonstantin Ananyev                                           |NGX_STREAM_UPSTREAM_WEIGHT
325e18a033bSKonstantin Ananyev                                           |NGX_STREAM_UPSTREAM_MAX_CONNS
326e18a033bSKonstantin Ananyev                                           |NGX_STREAM_UPSTREAM_MAX_FAILS
327e18a033bSKonstantin Ananyev                                           |NGX_STREAM_UPSTREAM_FAIL_TIMEOUT
328e18a033bSKonstantin Ananyev                                           |NGX_STREAM_UPSTREAM_DOWN
329e18a033bSKonstantin Ananyev                                           |NGX_STREAM_UPSTREAM_BACKUP);
330e18a033bSKonstantin Ananyev    if (uscf == NULL) {
331e18a033bSKonstantin Ananyev        return NGX_CONF_ERROR;
332e18a033bSKonstantin Ananyev    }
333e18a033bSKonstantin Ananyev
334e18a033bSKonstantin Ananyev
335e18a033bSKonstantin Ananyev    ctx = ngx_pcalloc(cf->pool, sizeof(ngx_stream_conf_ctx_t));
336e18a033bSKonstantin Ananyev    if (ctx == NULL) {
337e18a033bSKonstantin Ananyev        return NGX_CONF_ERROR;
338e18a033bSKonstantin Ananyev    }
339e18a033bSKonstantin Ananyev
340e18a033bSKonstantin Ananyev    stream_ctx = cf->ctx;
341e18a033bSKonstantin Ananyev    ctx->main_conf = stream_ctx->main_conf;
342e18a033bSKonstantin Ananyev
343e18a033bSKonstantin Ananyev    /* the upstream{}'s srv_conf */
344e18a033bSKonstantin Ananyev
345e18a033bSKonstantin Ananyev    ctx->srv_conf = ngx_pcalloc(cf->pool,
346e18a033bSKonstantin Ananyev                                sizeof(void *) * ngx_stream_max_module);
347e18a033bSKonstantin Ananyev    if (ctx->srv_conf == NULL) {
348e18a033bSKonstantin Ananyev        return NGX_CONF_ERROR;
349e18a033bSKonstantin Ananyev    }
350e18a033bSKonstantin Ananyev
351e18a033bSKonstantin Ananyev    ctx->srv_conf[ngx_stream_upstream_module.ctx_index] = uscf;
352e18a033bSKonstantin Ananyev
353e18a033bSKonstantin Ananyev    uscf->srv_conf = ctx->srv_conf;
354e18a033bSKonstantin Ananyev
355e18a033bSKonstantin Ananyev    for (m = 0; cf->cycle->modules[m]; m++) {
356e18a033bSKonstantin Ananyev        if (cf->cycle->modules[m]->type != NGX_STREAM_MODULE) {
357e18a033bSKonstantin Ananyev            continue;
358e18a033bSKonstantin Ananyev        }
359e18a033bSKonstantin Ananyev
360e18a033bSKonstantin Ananyev        module = cf->cycle->modules[m]->ctx;
361e18a033bSKonstantin Ananyev
362e18a033bSKonstantin Ananyev        if (module->create_srv_conf) {
363e18a033bSKonstantin Ananyev            mconf = module->create_srv_conf(cf);
364e18a033bSKonstantin Ananyev            if (mconf == NULL) {
365e18a033bSKonstantin Ananyev                return NGX_CONF_ERROR;
366e18a033bSKonstantin Ananyev            }
367e18a033bSKonstantin Ananyev
368e18a033bSKonstantin Ananyev            ctx->srv_conf[cf->cycle->modules[m]->ctx_index] = mconf;
369e18a033bSKonstantin Ananyev        }
370e18a033bSKonstantin Ananyev    }
371e18a033bSKonstantin Ananyev
372e18a033bSKonstantin Ananyev    uscf->servers = ngx_array_create(cf->pool, 4,
373e18a033bSKonstantin Ananyev                                     sizeof(ngx_stream_upstream_server_t));
374e18a033bSKonstantin Ananyev    if (uscf->servers == NULL) {
375e18a033bSKonstantin Ananyev        return NGX_CONF_ERROR;
376e18a033bSKonstantin Ananyev    }
377e18a033bSKonstantin Ananyev
378e18a033bSKonstantin Ananyev
379e18a033bSKonstantin Ananyev    /* parse inside upstream{} */
380e18a033bSKonstantin Ananyev
381e18a033bSKonstantin Ananyev    pcf = *cf;
382e18a033bSKonstantin Ananyev    cf->ctx = ctx;
383e18a033bSKonstantin Ananyev    cf->cmd_type = NGX_STREAM_UPS_CONF;
384e18a033bSKonstantin Ananyev
385e18a033bSKonstantin Ananyev    rv = ngx_conf_parse(cf, NULL);
386e18a033bSKonstantin Ananyev
387e18a033bSKonstantin Ananyev    *cf = pcf;
388e18a033bSKonstantin Ananyev
389e18a033bSKonstantin Ananyev    if (rv != NGX_CONF_OK) {
390e18a033bSKonstantin Ananyev        return rv;
391e18a033bSKonstantin Ananyev    }
392e18a033bSKonstantin Ananyev
393e18a033bSKonstantin Ananyev    if (uscf->servers->nelts == 0) {
394e18a033bSKonstantin Ananyev        ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
395e18a033bSKonstantin Ananyev                           "no servers are inside upstream");
396e18a033bSKonstantin Ananyev        return NGX_CONF_ERROR;
397e18a033bSKonstantin Ananyev    }
398e18a033bSKonstantin Ananyev
399e18a033bSKonstantin Ananyev    return rv;
400e18a033bSKonstantin Ananyev}
401e18a033bSKonstantin Ananyev
402e18a033bSKonstantin Ananyev
403e18a033bSKonstantin Ananyevstatic char *
404e18a033bSKonstantin Ananyevngx_stream_upstream_server(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
405e18a033bSKonstantin Ananyev{
406e18a033bSKonstantin Ananyev    ngx_stream_upstream_srv_conf_t  *uscf = conf;
407e18a033bSKonstantin Ananyev
408e18a033bSKonstantin Ananyev    time_t                         fail_timeout;
409e18a033bSKonstantin Ananyev    ngx_str_t                     *value, s;
410e18a033bSKonstantin Ananyev    ngx_url_t                      u;
411e18a033bSKonstantin Ananyev    ngx_int_t                      weight, max_conns, max_fails;
412e18a033bSKonstantin Ananyev    ngx_uint_t                     i;
413e18a033bSKonstantin Ananyev    ngx_stream_upstream_server_t  *us;
414e18a033bSKonstantin Ananyev
415e18a033bSKonstantin Ananyev    us = ngx_array_push(uscf->servers);
416e18a033bSKonstantin Ananyev    if (us == NULL) {
417e18a033bSKonstantin Ananyev        return NGX_CONF_ERROR;
418e18a033bSKonstantin Ananyev    }
419