blob: 1e2884ab7be0b16114a75d2f2e2fd5a56a8c7baa [file] [log] [blame]
Maxim Dounind26e7c52008-09-12 20:50:35 +04001package Test::Nginx;
Maxim Dounin668f89e2008-09-07 05:00:28 +04002
3# (C) Maxim Dounin
4
5# Generict module for nginx tests.
6
7###############################################################################
8
9use warnings;
10use strict;
11
12use base qw/ Exporter /;
13
Maxim Douninf79ada02008-11-01 16:19:28 +030014our @EXPORT = qw/ log_in log_out http http_get http_head /;
Maxim Dounin668f89e2008-09-07 05:00:28 +040015
16###############################################################################
17
Maxim Dounin668f89e2008-09-07 05:00:28 +040018use File::Temp qw/ tempdir /;
19use IO::Socket;
Maxim Dounin9ad211a2008-09-08 06:08:38 +040020use Socket qw/ CRLF /;
Maxim Dounine0a4a1b2008-10-10 20:09:14 +040021use Test::More qw//;
Maxim Dounin668f89e2008-09-07 05:00:28 +040022
Maxim Dounin668f89e2008-09-07 05:00:28 +040023###############################################################################
24
Maxim Dounin63257d02008-11-18 20:43:47 +030025our $NGINX = defined $ENV{TEST_NGINX_BINARY} ? $ENV{TEST_NGINX_BINARY}
26 : '../nginx/objs/nginx';
Maxim Dounine0a4a1b2008-10-10 20:09:14 +040027
Maxim Dounind26e7c52008-09-12 20:50:35 +040028sub new {
29 my $self = {};
30 bless $self;
Maxim Dounin46bf6af2008-09-13 02:57:01 +040031
32 $self->{_testdir} = tempdir(
33 'nginx-test-XXXXXXXXXX',
34 TMPDIR => 1,
35 CLEANUP => not $ENV{LEAVE}
36 )
37 or die "Can't create temp directory: $!\n";
38
Maxim Dounind26e7c52008-09-12 20:50:35 +040039 return $self;
40}
41
42sub DESTROY {
43 my ($self) = @_;
44 $self->stop();
45}
46
Maxim Dounine0a4a1b2008-10-10 20:09:14 +040047sub has {
Maxim Dounin09263aa2008-10-10 20:23:39 +040048 my ($self, $feature) = @_;
49
50 my %regex = (
51 mail => '--with-mail',
Maxim Dounina01dff62008-10-14 02:18:03 +040052 flv => '--with-http_flv_module',
Maxim Dounin09263aa2008-10-10 20:23:39 +040053 rewrite => '(?s)^(?!.*--without-http_rewrite_module)',
Maxim Dounin09263aa2008-10-10 20:23:39 +040054 );
55
Maxim Dounine0a4a1b2008-10-10 20:09:14 +040056 Test::More::plan(skip_all => "$feature not compiled in")
Maxim Dounin09263aa2008-10-10 20:23:39 +040057 unless `$NGINX -V 2>&1` =~ $regex{$feature};
58
59 return $self;
60}
61
Maxim Dounin323cf242008-10-11 10:58:43 +040062sub has_daemon($) {
63 my ($self, $daemon) = @_;
64
65 Test::More::plan(skip_all => "$daemon not found")
66 unless `which $daemon`;
67
68 return $self;
69}
70
71sub plan($) {
Maxim Dounin09263aa2008-10-10 20:23:39 +040072 my ($self, $plan) = @_;
73
74 Test::More::plan(tests => $plan);
75
76 return $self;
Maxim Dounine0a4a1b2008-10-10 20:09:14 +040077}
78
Maxim Dounin323cf242008-10-11 10:58:43 +040079sub run(;$) {
Maxim Dounind26e7c52008-09-12 20:50:35 +040080 my ($self, $conf) = @_;
Maxim Dounin668f89e2008-09-07 05:00:28 +040081
Maxim Dounin46bf6af2008-09-13 02:57:01 +040082 my $testdir = $self->{_testdir};
Maxim Dounin668f89e2008-09-07 05:00:28 +040083
Maxim Dounin46bf6af2008-09-13 02:57:01 +040084 if (defined $conf) {
85 my $c = `cat $conf`;
86 $self->write_file_expand('nginx.conf', $c);
87 }
Maxim Dounin668f89e2008-09-07 05:00:28 +040088
89 my $pid = fork();
90 die "Unable to fork(): $!\n" unless defined $pid;
91
92 if ($pid == 0) {
Maxim Dounine0a4a1b2008-10-10 20:09:14 +040093 exec($NGINX, '-c', "$testdir/nginx.conf", '-g',
Maxim Dounin668f89e2008-09-07 05:00:28 +040094 "pid $testdir/nginx.pid; "
Maxim Dounin5cd63152008-09-30 21:39:23 +040095 . "error_log $testdir/error.log debug;")
Maxim Dounin668f89e2008-09-07 05:00:28 +040096 or die "Unable to exec(): $!\n";
97 }
98
99 # wait for nginx to start
100
Maxim Douninfd1126f2008-10-24 14:42:21 +0400101 for (1 .. 30) {
102 select undef, undef, undef, 0.05;
103 last if -e "$self->{_testdir}/nginx.pid";
104 }
Maxim Dounind26e7c52008-09-12 20:50:35 +0400105
Maxim Dounin323cf242008-10-11 10:58:43 +0400106 die "Can't start nginx" unless -e "$self->{_testdir}/nginx.pid";
107
Maxim Douninc0ef0162008-09-26 18:24:28 +0400108 $self->{_started} = 1;
Maxim Dounind26e7c52008-09-12 20:50:35 +0400109 return $self;
Maxim Dounin668f89e2008-09-07 05:00:28 +0400110}
111
Maxim Dounin46bf6af2008-09-13 02:57:01 +0400112sub stop() {
Maxim Dounind26e7c52008-09-12 20:50:35 +0400113 my ($self) = @_;
114
Maxim Douninc0ef0162008-09-26 18:24:28 +0400115 while ($self->{_daemons} && scalar @{$self->{_daemons}}) {
116 my $p = shift @{$self->{_daemons}};
Maxim Dounin27a152e2009-03-28 01:17:25 +0300117 kill 'TERM', $p;
Maxim Douninc0ef0162008-09-26 18:24:28 +0400118 wait;
119 }
120
121 return $self unless $self->{_started};
122
123 kill 'TERM', `cat $self->{_testdir}/nginx.pid`;
Maxim Dounin668f89e2008-09-07 05:00:28 +0400124 wait;
Maxim Dounin668f89e2008-09-07 05:00:28 +0400125
Maxim Douninc0ef0162008-09-26 18:24:28 +0400126 $self->{_started} = 0;
127
Maxim Dounind26e7c52008-09-12 20:50:35 +0400128 return $self;
Maxim Dounin668f89e2008-09-07 05:00:28 +0400129}
130
Maxim Dounin46bf6af2008-09-13 02:57:01 +0400131sub write_file($$) {
Maxim Dounind26e7c52008-09-12 20:50:35 +0400132 my ($self, $name, $content) = @_;
Maxim Dounind4ea27b2008-09-11 19:55:04 +0400133
Maxim Dounind26e7c52008-09-12 20:50:35 +0400134 open F, '>' . $self->{_testdir} . '/' . $name
Maxim Dounind4ea27b2008-09-11 19:55:04 +0400135 or die "Can't create $name: $!";
136 print F $content;
137 close F;
Maxim Dounind26e7c52008-09-12 20:50:35 +0400138
139 return $self;
Maxim Dounind4ea27b2008-09-11 19:55:04 +0400140}
141
Maxim Dounin46bf6af2008-09-13 02:57:01 +0400142sub write_file_expand($$) {
143 my ($self, $name, $content) = @_;
144
145 $content =~ s/%%TESTDIR%%/$self->{_testdir}/gms;
146
147 return $self->write_file($name, $content);
148}
149
Maxim Dounin323cf242008-10-11 10:58:43 +0400150sub run_daemon($;@) {
151 my ($self, $code, @args) = @_;
Maxim Douninc0ef0162008-09-26 18:24:28 +0400152
153 my $pid = fork();
154 die "Can't fork daemon: $!\n" unless defined $pid;
155
156 if ($pid == 0) {
Maxim Dounin323cf242008-10-11 10:58:43 +0400157 if (ref($code) eq 'CODE') {
158 $code->(@args);
159 exit 0;
160 } else {
161 exec($code, @args);
162 }
Maxim Douninc0ef0162008-09-26 18:24:28 +0400163 }
164
165 $self->{_daemons} = [] unless defined $self->{_daemons};
166 push @{$self->{_daemons}}, $pid;
167
168 return $self;
169}
170
Maxim Dounin6fbc1352008-11-10 19:04:34 +0300171sub testdir() {
172 my ($self) = @_;
173 return $self->{_testdir};
174}
175
Maxim Dounin668f89e2008-09-07 05:00:28 +0400176###############################################################################
177
178sub log_out {
179 my ($msg) = @_;
180 $msg =~ s/^/# >> /gm;
181 $msg .= "\n" unless $msg =~ /\n\Z/;
182 print $msg;
183}
184
185sub log_in {
186 my ($msg) = @_;
187 $msg =~ s/^/# << /gm;
Maxim Dounin4621cc52008-09-07 19:06:49 +0400188 $msg =~ s/([^\x20-\x7e])/sprintf('\\x%02x', ord($1)) . (($1 eq "\n") ? "\n" : '')/gmxe;
Maxim Dounin668f89e2008-09-07 05:00:28 +0400189 $msg .= "\n" unless $msg =~ /\n\Z/;
190 print $msg;
191}
192
193###############################################################################
194
Maxim Dounin323cf242008-10-11 10:58:43 +0400195sub http_get($) {
196 my ($url) = @_;
197 return http(<<EOF);
198GET $url HTTP/1.0
199Host: localhost
200
201EOF
202}
203
Maxim Douninf79ada02008-11-01 16:19:28 +0300204sub http_head($) {
205 my ($url) = @_;
206 return http(<<EOF);
207HEAD $url HTTP/1.0
208Host: localhost
209
210EOF
211}
212
Maxim Dounin323cf242008-10-11 10:58:43 +0400213sub http($) {
Maxim Dounin668f89e2008-09-07 05:00:28 +0400214 my ($request) = @_;
215 my $reply;
216 eval {
217 local $SIG{ALRM} = sub { die "alarm\n" };
218 alarm(2);
219 my $s = IO::Socket::INET->new(
220 Proto => 'tcp',
Maxim Dounin3c472202008-11-02 15:01:41 +0300221 PeerAddr => '127.0.0.1:8080'
Maxim Dounin668f89e2008-09-07 05:00:28 +0400222 );
223 log_out($request);
224 $s->print($request);
225 local $/;
226 $reply = $s->getline();
227 log_in($reply);
228 alarm(0);
229 };
230 alarm(0);
231 if ($@) {
232 log_in('(timeout)');
233 return undef;
234 }
235 return $reply;
236}
237
238###############################################################################
239
Maxim Dounin668f89e2008-09-07 05:00:28 +04002401;
241
242###############################################################################