forked from openresty/test-nginx
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathREADME
294 lines (207 loc) · 10.2 KB
/
README
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
NAME
Test::Nginx - Data-driven test scaffold for Nginx C module and OpenResty
Lua library development
DESCRIPTION
This distribution provides two testing modules for Nginx C module
development:
* Test::Nginx::Socket (This is recommended.)
* Test::Nginx::LWP (This is obsolete.)
All of them are based on Test::Base.
Usually, Test::Nginx::Socket is preferred because it works on a much
lower level and not that fault tolerant like Test::Nginx::LWP.
Also, a lot of connection hang issues (like wrong "r->main->count" value
in nginx 0.8.x) can only be captured by Test::Nginx::Socket because
Perl's LWP::UserAgent client will close the connection itself which will
conceal such issues from the testers.
Test::Nginx automatically starts an nginx instance (from the "PATH" env)
rooted at t/servroot/ and the default config template makes this nginx
instance listen on the port 1984 by default. One can specify a different
port number by setting his port number to the "TEST_NGINX_PORT"
environment, as in
export TEST_NGINX_PORT=1989
etcproxy integration
The default settings in etcproxy
(https://github.com/chaoslawful/etcproxy) makes this small TCP proxy
split the TCP packets into bytes and introduce 1 ms latency among them.
There's usually various TCP chains that we can put etcproxy into, for
example
Test::Nginx <=> nginx
$ ./etcproxy 1234 1984
Here we tell etcproxy to listen on port 1234 and to delegate all the TCP
traffic to the port 1984, the default port that Test::Nginx makes nginx
listen to.
And then we tell Test::Nginx to test against the port 1234, where
etcproxy listens on, rather than the port 1984 that nginx directly
listens on:
$ TEST_NGINX_CLIENT_PORT=1234 prove -r t/
Then the TCP chain now looks like this:
Test::Nginx <=> etcproxy (1234) <=> nginx (1984)
So etcproxy can effectively emulate extreme network conditions and
exercise "unusual" code paths in your nginx server by your tests.
In practice, *tons* of weird bugs can be captured by this setting. Even
ourselves didn't expect that this simple approach is so effective.
nginx <=> memcached
We first start the memcached server daemon on port 11211:
memcached -p 11211 -vv
and then we another etcproxy instance to listen on port 11984 like this
$ ./etcproxy 11984 11211
Then we tell our t/foo.t test script to connect to 11984 rather than
11211:
# foo.t
use Test::Nginx::Socket;
repeat_each(1);
plan tests => 2 * repeat_each() * blocks();
$ENV{TEST_NGINX_MEMCACHED_PORT} ||= 11211; # make this env take a default value
run_tests();
__DATA__
=== TEST 1: sanity
--- config
location /foo {
set $memc_cmd set;
set $memc_key foo;
set $memc_value bar;
memc_pass 127.0.0.1:$TEST_NGINX_MEMCACHED_PORT;
}
--- request
GET /foo
--- response_body_like: STORED
--- error_code: 201
The Test::Nginx library will automatically expand the special macro
$TEST_NGINX_MEMCACHED_PORT to the environment with the same name. You
can define your own $TEST_NGINX_BLAH_BLAH_PORT macros as long as its
prefix is "TEST_NGINX_" and all in upper case letters.
And now we can run your test script against the etcproxy port 11984:
TEST_NGINX_MEMCACHED_PORT=11984 prove t/foo.t
Then the TCP chains look like this:
Test::Nginx <=> nginx (1984) <=> etcproxy (11984) <=> memcached (11211)
If "TEST_NGINX_MEMCACHED_PORT" is not set, then it will take the default
value 11211, which is what we want when there's no etcproxy configured:
Test::Nginx <=> nginx (1984) <=> memcached (11211)
This approach also works for proxied mysql and postgres traffic. Please
see the live test suite of ngx_drizzle and ngx_postgres for more
details.
Usually we set both "TEST_NGINX_CLIENT_PORT" and
"TEST_NGINX_MEMCACHED_PORT" (and etc) at the same time, effectively
yielding the following chain:
Test::Nginx <=> etcproxy (1234) <=> nginx (1984) <=> etcproxy (11984) <=> memcached (11211)
as long as you run two separate etcproxy instances in two separate
terminals.
It's easy to verify if the traffic actually goes through your etcproxy
server. Just check if the terminal running etcproxy emits outputs. By
default, etcproxy always dump out the incoming and outgoing data to
stdout/stderr.
valgrind integration
Test::Nginx has integrated support for valgrind (<http://valgrind.org>)
even though by default it does not bother running it with the tests
because valgrind will significantly slow down the test suite.
First ensure that your valgrind executable visible in your PATH env. And
then run your test suite with the "TEST_NGINX_USE_VALGRIND" env set to
true:
TEST_NGINX_USE_VALGRIND=1 prove -r t
If you see false alarms, you do have a chance to skip them by defining a
./valgrind.suppress file at the root of your module source tree, as in
<https://github.com/chaoslawful/drizzle-nginx-module/blob/master/valgrin
d.suppress>
This is the suppression file for ngx_drizzle. Test::Nginx will
automatically use it to start nginx with valgrind memcheck if this file
does exist at the expected location.
If you do see a lot of "Connection refused" errors while running the
tests this way, then you probably have a slow machine (or a very busy
one) that the default waiting time is not sufficient for valgrind to
start. You can define the sleep time to a larger value by setting the
"TEST_NGINX_SLEEP" env:
TEST_NGINX_SLEEP=1 prove -r t
The time unit used here is "second". The default sleep setting just fits
my ThinkPad ("Core2Duo T9600").
Applying the no-pool patch to your nginx core is recommended while
running nginx with valgrind:
<https://github.com/shrimp/no-pool-nginx>
The nginx memory pool can prevent valgrind from spotting lots of invalid
memory reads/writes as well as certain double-free errors. We did find a
lot more memory issues in many of our modules when we first introduced
the no-pool patch in practice ;)
There's also more advanced features in Test::Nginx that have never
documented. I'd like to write more about them in the near future ;)
Nginx C modules that use Test::Nginx to drive their test suites
ngx_echo
<http://github.com/agentzh/echo-nginx-module>
ngx_headers_more
<http://github.com/agentzh/headers-more-nginx-module>
ngx_chunkin
<http://wiki.nginx.org/NginxHttpChunkinModule>
ngx_memc
<http://wiki.nginx.org/NginxHttpMemcModule>
ngx_drizzle
<http://github.com/chaoslawful/drizzle-nginx-module>
ngx_rds_json
<http://github.com/agentzh/rds-json-nginx-module>
ngx_rds_csv
<http://github.com/agentzh/rds-csv-nginx-module>
ngx_xss
<http://github.com/agentzh/xss-nginx-module>
ngx_srcache
<http://github.com/agentzh/srcache-nginx-module>
ngx_lua
<http://github.com/chaoslawful/lua-nginx-module>
ngx_set_misc
<http://github.com/agentzh/set-misc-nginx-module>
ngx_array_var
<http://github.com/agentzh/array-var-nginx-module>
ngx_form_input
<http://github.com/calio/form-input-nginx-module>
ngx_iconv
<http://github.com/calio/iconv-nginx-module>
ngx_set_cconv
<http://github.com/liseen/set-cconv-nginx-module>
ngx_postgres
<http://github.com/FRiCKLE/ngx_postgres>
ngx_coolkit
<http://github.com/FRiCKLE/ngx_coolkit>
Naxsi
<http://code.google.com/p/naxsi/>
SOURCE REPOSITORY
This module has a Git repository on Github, which has access for all.
http://github.com/agentzh/test-nginx
If you want a commit bit, feel free to drop me a line.
DEBIAN PACKAGES
António P. P. Almeida is maintaining a Debian package for this module in
his Debian repository: <http://debian.perusio.net>
Community
English Mailing List
The "openresty-en" mailing list is for English speakers:
<https://groups.google.com/group/openresty-en>
Chinese Mailing List
The "openresty" mailing list is for Chinese speakers:
<https://groups.google.com/group/openresty>
AUTHORS
agentzh (章亦春) "<[email protected]>"
Antoine BONAVITA "<[email protected]>"
COPYRIGHT & LICENSE
Copyright (c) 2009-2014, agentzh "<[email protected]>".
Copyright (c) 2011-2012, Antoine Bonavita
"<[email protected]>".
This module is licensed under the terms of the BSD license.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
* Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
* 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.
* Neither the name of the authors nor the names of its contributors
may be used to endorse or promote products derived from this
software without specific prior written permission.
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.
SEE ALSO
Test::Nginx::LWP, Test::Nginx::Socket, Test::Base.