1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
|
=encoding utf-8
=head1 Name
ngx_http_lua_upstream - Nginx C module to expose Lua API to ngx_lua for Nginx upstreams
=head1 Status
This module is production ready.
=head1 Synopsis
http {
upstream foo.com {
server 127.0.0.1 fail_timeout=53 weight=4 max_fails=100;
server agentzh.org:81;
}
upstream bar {
server 127.0.0.2;
}
server {
listen 8080;
# sample output for the following /upstream interface:
# upstream foo.com:
# addr = 127.0.0.1:80, weight = 4, fail_timeout = 53, max_fails = 100
# addr = 106.184.1.99:81, weight = 1, fail_timeout = 10, max_fails = 1
# upstream bar:
# addr = 127.0.0.2:80, weight = 1, fail_timeout = 10, max_fails = 1
location = /upstreams {
default_type text/plain;
content_by_lua_block {
local concat = table.concat
local upstream = require "ngx.upstream"
local get_servers = upstream.get_servers
local get_upstreams = upstream.get_upstreams
local us = get_upstreams()
for _, u in ipairs(us) do
ngx.say("upstream ", u, ":")
local srvs, err = get_servers(u)
if not srvs then
ngx.say("failed to get servers in upstream ", u)
else
for _, srv in ipairs(srvs) do
local first = true
for k, v in pairs(srv) do
if first then
first = false
ngx.print(" ")
else
ngx.print(", ")
end
if type(v) == "table" then
ngx.print(k, " = {", concat(v, ", "), "}")
else
ngx.print(k, " = ", v)
end
end
ngx.print("\n")
end
end
end
}
}
}
}
=head1 Functions
=head2 get_upstreams
C<syntax: names = upstream.get_upstreams()>
Get a list of the names for all the named upstream groups (i.e., explicit C<upstream {}> blocks).
Note that implicit upstream groups created by C<proxy_pass> and etc are excluded.
=head2 get_servers
C<syntax: servers = upstream.get_servers(upstream_name)>
Get configurations for all the servers in the specified upstream group. Please one server may take multiple addresses when its server name can be resolved to multiple addresses.
The return value is an array-like Lua table. Each table entry is a hash-like Lua table that takes the following keys:
=over
=item *
addr
socket address(es). can be either a Lua string or an array-like Lua table of Lua strings.
=item *
backup
=item *
fail_timeout
=item *
max_fails
=item *
name
=item *
weight
=back
=head2 get_primary_peers
C<syntax: peers = upstream.get_primary_peers(upstream_name)>
Get configurations for all the primary (non-backup) peers in the specified upstream group.
The return value is an array-like Lua table for all the primary peers. Each table entry is a (nested) hash-like Lua table that takes the following keys:
=over
=item *
current_weight
=item *
effective_weight
=item *
fail_timeout
=item *
fails
=item *
id
Identifier (ID) for the peer. This ID can be used to reference a peer in a group in the peer modifying API.
=item *
max_fails
=item *
name
Socket address for the current peer
=item *
weight
=item *
accessed
Timestamp for the last access (in seconds since the Epoch)
=item *
checked
Timestamp for the last check (in seconds since the Epoch)
=item *
down
Holds true if the peer has been marked as "down", otherwise this key is not present
=item *
conns
Number of active connections to the peer (this requires NGINX 1.9.0 or above).
=back
=head2 get_backup_peers
C<syntax: peers = upstream.get_backup_peers(upstream_name)>
Get configurations for all the backup peers in the specified upstream group.
The return value has the same structure as L<get_primary_peers> function.
=head2 set_peer_down
C<syntax: ok, err = upstream.set_peer_down(upstream_name, is_backup, peer_id, down_value)>
Set the "down" (boolean) attribute of the specified peer.
To uniquely specify a peer, you need to specify the upstream name, whether or not it is a backup peer, and the peer id (starting from 0).
Note that this method only changes the peer settings in the current Nginx worker
process. You need to synchronize the changes across all the Nginx workers yourself if you
want a server-wide change (for example, by means of L<ngx_lua|https://github.com/openresty/lua-nginx-module#ngxshareddict>'s L<ngx.shared.DICT|https://github.com/openresty/lua-nginx-module#ngxshareddict>).
Below is an example. Consider we have a "bar" upstream block in C<nginx.conf>:
upstream bar {
server 127.0.0.2;
server 127.0.0.3 backup;
server 127.0.0.4 fail_timeout=23 weight=7 max_fails=200 backup;
}
then
upstream.set_peer_down("bar", false, 0, true)
will turn down the primary peer corresponding to C<server 127.0.0.2>.
Similarly,
upstream.set_peer_down("bar", true, 1, true)
will turn down the backup peer corresponding to C<server 127.0.0.4 ...>.
You can turn on a peer again by providing a C<false> value as the 4th argument.
=head2 current_upstream_name
C<syntax: name = upstream.current_upstream_name()>
Returns the name of the proxied upstream for the current request.
If there is no upstream for this request (no C<proxy_pass> call), or this
function is called in a phase prior to the content phase, then the return value
will be C<nil>. If a port is explicitly included in the upstream definition or
C<proxy_pass> directive, it will be included in the return value of this function.
Example:
-- upstream my_upstream { ... }
-- proxy_pass http://my_upstream;
upstream.current_upstream_name() --> my_upstream
-- proxy_pass http://example.com:1234;
upstream.current_upstream_name() --> example.com:1234
Note that implicit upstreams created by C<proxy_pass> are included, contrary to
the output of C<upstream.get_upstreams()>.
=head1 TODO
=over
=item *
Add API to add or remove servers to existing upstream groups.
=back
=head1 Compatibility
The following versions of Nginx should work with this module:
=over
=item *
B<1.11.x> (last tested: 1.11.2)
=item *
B<1.10.x>
=item *
B<1.9.x> (last tested: 1.9.15)
=item *
B<1.8.x>
=item *
B<1.7.x> (last tested: 1.7.10)
=item *
B<1.6.x>
=item *
B<1.5.x> (last tested: 1.5.12)
=back
=head1 Installation
This module is bundled and enabled by default in the L<OpenResty|http://openresty.org> bundle. And you are recommended to use OpenResty.
=over
=item 1.
Grab the nginx source code from L<nginx.org|http://nginx.org/>, for example,
the version 1.11.2 (see L<nginx compatibility>),
=item 2.
then grab the source code of the L<ngx_lua|https://github.com/openresty/lua-nginx-module#installation> as well as its dependencies like L<LuaJIT|http://luajit.org/download.html>.
=item 3.
and finally build the source with this module:
=back
wget 'http://nginx.org/download/nginx-1.11.2.tar.gz'
tar -xzvf nginx-1.11.2.tar.gz
cd nginx-1.11.2/
# assuming your luajit is installed to /opt/luajit:
export LUAJIT_LIB=/opt/luajit/lib
# assuming you are using LuaJIT v2.1:
export LUAJIT_INC=/opt/luajit/include/luajit-2.1
# Here we assume you would install you nginx under /opt/nginx/.
./configure --prefix=/opt/nginx \
--with-ld-opt="-Wl,-rpath,$LUAJIT_LIB" \
--add-module=/path/to/lua-nginx-module \
--add-module=/path/to/lua-upstream-nginx-module
make -j2
make install
Starting from NGINX 1.9.11, you can also compile this module as a dynamic module, by using the C<--add-dynamic-module=PATH> option instead of C<--add-module=PATH> on the
C<./configure> command line above. And then you can explicitly load the module in your C<nginx.conf> via the L<load_module|http://nginx.org/en/docs/ngx_core_module.html#load_module>
directive, for example,
load_module /path/to/modules/ngx_http_lua_upstream_module.so;
=head1 Author
Yichun "agentzh" Zhang (章亦春) E<lt>agentzh@gmail.comE<gt>, OpenResty Inc.
=head1 Copyright and License
This module is licensed under the BSD license.
Copyright (C) 2014-2017, by Yichun "agentzh" Zhang, OpenResty Inc.
All rights reserved.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
=over
=item *
Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
=back
=over
=item *
Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
=back
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
=head1 See Also
=over
=item *
the ngx_lua module: http://github.com/openresty/lua-nginx-module#readme
=item *
the L<lua-resty-upstream-healthcheck|https://github.com/openresty/lua-resty-upstream-healthcheck> library which makes use of the Lua API provided by this module.
=back
|