line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Protocol::UWSGI; |
2
|
|
|
|
|
|
|
# ABSTRACT: support for the UWSGI protocol |
3
|
1
|
|
|
1
|
|
27856
|
use strict; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
35
|
|
4
|
1
|
|
|
1
|
|
4
|
use warnings; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
28
|
|
5
|
|
|
|
|
|
|
|
6
|
1
|
|
|
1
|
|
885
|
use parent qw(Exporter); |
|
1
|
|
|
|
|
314
|
|
|
1
|
|
|
|
|
6
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
our $VERSION = '1.000'; |
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
=head1 NAME |
11
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
Protocol::UWSGI - handle the UWSGI wire protocol |
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
=head1 VERSION |
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
Version 1.000 |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
=head1 SYNOPSIS |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
use strict; |
21
|
|
|
|
|
|
|
use warnings; |
22
|
|
|
|
|
|
|
use Protocol::UWSGI qw(:all); |
23
|
|
|
|
|
|
|
# Encode... |
24
|
|
|
|
|
|
|
my $req = build_request( |
25
|
|
|
|
|
|
|
uri => 'http://localhost', |
26
|
|
|
|
|
|
|
method => 'GET', |
27
|
|
|
|
|
|
|
remote => '1.2.3.4:1234', |
28
|
|
|
|
|
|
|
); |
29
|
|
|
|
|
|
|
# ... and decode again |
30
|
|
|
|
|
|
|
warn "URI was " . uri_from_env( |
31
|
|
|
|
|
|
|
extract_frame(\$req) |
32
|
|
|
|
|
|
|
); |
33
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
=head1 DESCRIPTION |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
Provides protocol-level support for UWSGI packet generation/decoding, as |
37
|
|
|
|
|
|
|
defined by L. |
38
|
|
|
|
|
|
|
Currently expects to deal with PSGI data (modifier 1 == 5), although this |
39
|
|
|
|
|
|
|
may be extended later if there's any demand for the other packet types. |
40
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
This is unlikely to be useful in an application - it's intended to provide |
42
|
|
|
|
|
|
|
support for dealing with the protocol in an existing framework: it deals |
43
|
|
|
|
|
|
|
with the abstract protocol only, and has no network transport handling at |
44
|
|
|
|
|
|
|
all. Try L for an implementation that actually does |
45
|
|
|
|
|
|
|
something useful. |
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
Typically you'd create a UNIX socket and listen for requests, passing |
48
|
|
|
|
|
|
|
any data to the L function and handling the resulting |
49
|
|
|
|
|
|
|
data if that function returns something other than undef: |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
# Detect read - first packet is usually the UWSGI header, everything |
52
|
|
|
|
|
|
|
# after that would be the HTTP request body if there is one: |
53
|
|
|
|
|
|
|
sub on_read { |
54
|
|
|
|
|
|
|
my ($self, $buffref) = @_; |
55
|
|
|
|
|
|
|
while(my $pkt = extract_frame($buffref)) { |
56
|
|
|
|
|
|
|
$self->handle_uwsgi($pkt); |
57
|
|
|
|
|
|
|
} |
58
|
|
|
|
|
|
|
} |
59
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
# and probably an EOF handler to detect client hangup |
61
|
|
|
|
|
|
|
# sub on_eof { ... } |
62
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
=head1 IMPLEMENTATION - Server |
64
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
A server implementation typically accepts requests from a reverse |
66
|
|
|
|
|
|
|
proxy, such as nginx, and returns HTTP responses. |
67
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
Import the :server tag to get L, L |
69
|
|
|
|
|
|
|
and in future maybe L functions: |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
use Protocol::UWSGI qw(:server); |
72
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
=head1 IMPLEMENTATION - Client |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
A client implementation typically accepts HTTP requests and converts |
76
|
|
|
|
|
|
|
them to UWSGI for passing to a UWSGI-capable application. |
77
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
Import the :client tag to get L: |
79
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
use Protocol::UWSGI qw(:client); |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
=cut |
83
|
|
|
|
|
|
|
|
84
|
1
|
|
|
1
|
|
1005
|
use Encode (); |
|
1
|
|
|
|
|
13151
|
|
|
1
|
|
|
|
|
24
|
|
85
|
1
|
|
|
1
|
|
915
|
use URI; |
|
1
|
|
|
|
|
8053
|
|
|
1
|
|
|
|
|
39
|
|
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
use constant { |
88
|
1
|
|
|
|
|
1166
|
PSGI_MODIFIER1 => 5, |
89
|
|
|
|
|
|
|
PSGI_MODIFIER2 => 0, |
90
|
1
|
|
|
1
|
|
8
|
}; |
|
1
|
|
|
|
|
2
|
|
91
|
|
|
|
|
|
|
|
92
|
|
|
|
|
|
|
our @EXPORT_OK = qw( |
93
|
|
|
|
|
|
|
extract_frame |
94
|
|
|
|
|
|
|
uri_from_env |
95
|
|
|
|
|
|
|
|
96
|
|
|
|
|
|
|
build_request |
97
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
PSGI_MODIFIER1 |
99
|
|
|
|
|
|
|
PSGI_MODIFIER2 |
100
|
|
|
|
|
|
|
); |
101
|
|
|
|
|
|
|
our %EXPORT_TAGS = ( |
102
|
|
|
|
|
|
|
'server' => [qw(extract_frame uri_from_env)], |
103
|
|
|
|
|
|
|
'client' => [qw(build_request)], |
104
|
|
|
|
|
|
|
'all' => \@EXPORT_OK |
105
|
|
|
|
|
|
|
); |
106
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
=head1 FUNCTIONS |
108
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
If you're handling incoming UWSGI requests, you'll need to instantiate |
110
|
|
|
|
|
|
|
via L then decode the request using L. |
111
|
|
|
|
|
|
|
|
112
|
|
|
|
|
|
|
If you're making UWSGI requests against an external UWSGI server, |
113
|
|
|
|
|
|
|
that'll be L. |
114
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
Just want to decode captured traffic? L again. |
116
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
=cut |
118
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
=head2 extract_frame |
120
|
|
|
|
|
|
|
|
121
|
|
|
|
|
|
|
Attempts to extract a single UWSGI packet from the given buffer (which |
122
|
|
|
|
|
|
|
should be passed as a scalar ref, e.g. |
123
|
|
|
|
|
|
|
|
124
|
|
|
|
|
|
|
my $buffref = \"..."; |
125
|
|
|
|
|
|
|
my $req = Protocol::UWSGI->extract_frame($buffref) |
126
|
|
|
|
|
|
|
or die "could not find UWSGI frame"; |
127
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
If we had enough data for a packet, that packet will be removed from |
129
|
|
|
|
|
|
|
the buffer and returned. There may be additional packet data that |
130
|
|
|
|
|
|
|
can be extracted, or non-UWSGI data such as HTTP request body. |
131
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
If this returns undef, there's not enough data to process - in this case, |
133
|
|
|
|
|
|
|
the buffer is guaranteed not to be modified. |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
This may be called as a class method or an instance method. |
136
|
|
|
|
|
|
|
The instance state will remain unchanged after calling this method. |
137
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
Note that there is no constructor provided in this |
139
|
|
|
|
|
|
|
class - if you want to call this as an instance method, |
140
|
|
|
|
|
|
|
you'll need to bless manually or be applying this as |
141
|
|
|
|
|
|
|
a role/mixin. |
142
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
=cut |
144
|
|
|
|
|
|
|
|
145
|
|
|
|
|
|
|
sub extract_frame { |
146
|
4
|
|
|
4
|
1
|
7
|
my ($buffref) = @_; |
147
|
|
|
|
|
|
|
|
148
|
4
|
|
|
|
|
16
|
my ($modifier1, $length, $modifier2) = unpack 'C1v1C1', $$buffref; |
149
|
|
|
|
|
|
|
# no, still too short |
150
|
4
|
50
|
33
|
|
|
23
|
return undef unless $length && length $$buffref >= $length + 4; |
151
|
|
|
|
|
|
|
|
152
|
|
|
|
|
|
|
# then do the modifier-specific handling |
153
|
4
|
50
|
|
|
|
9
|
die "Unsupported modifier1 $modifier1" unless $modifier1 == PSGI_MODIFIER1; |
154
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
# hack bits off the buffer |
156
|
4
|
|
|
|
|
9
|
substr $$buffref, 0, 4, ''; |
157
|
|
|
|
|
|
|
|
158
|
4
|
|
|
|
|
50
|
my %env = unpack '(v1/a*)*', substr $$buffref, 0, $length, ''; |
159
|
4
|
|
|
|
|
32
|
\%env |
160
|
|
|
|
|
|
|
} |
161
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
# For cases where non-PSGI modifiers are wanted. Takes about 2.5x as long. |
163
|
|
|
|
|
|
|
sub extract_frame_universal { |
164
|
0
|
|
|
0
|
0
|
0
|
my $buffref = shift; |
165
|
|
|
|
|
|
|
# too short |
166
|
0
|
0
|
|
|
|
0
|
return undef unless length $$buffref >= 4; |
167
|
|
|
|
|
|
|
|
168
|
0
|
|
|
|
|
0
|
my ($modifier1, $length, $modifier2) = unpack 'C1v1C1', $$buffref; |
169
|
|
|
|
|
|
|
# no, still too short |
170
|
0
|
0
|
|
|
|
0
|
return undef unless length $$buffref >= $length + 4; |
171
|
|
|
|
|
|
|
|
172
|
|
|
|
|
|
|
# hack bits off the buffer |
173
|
0
|
|
|
|
|
0
|
substr $$buffref, 0, 4, ''; |
174
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
# then do the modifier-specific handling |
176
|
0
|
|
|
|
|
0
|
return extract_modifier( |
177
|
|
|
|
|
|
|
modifier1 => $modifier1, |
178
|
|
|
|
|
|
|
modifier2 => $modifier2, |
179
|
|
|
|
|
|
|
length => $length, |
180
|
|
|
|
|
|
|
buffer => $buffref, |
181
|
|
|
|
|
|
|
); |
182
|
|
|
|
|
|
|
} |
183
|
|
|
|
|
|
|
|
184
|
|
|
|
|
|
|
=head2 bytes_required |
185
|
|
|
|
|
|
|
|
186
|
|
|
|
|
|
|
Returns the number of additional bytes we'll need in order to proceed. |
187
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
If zero, this means we should be able to extract a valid frame. |
189
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
=cut |
191
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
sub bytes_required { |
193
|
0
|
|
|
0
|
1
|
0
|
my $buffref = shift; |
194
|
0
|
0
|
|
|
|
0
|
return 4 - length($$buffref) unless length $$buffref >= 4; |
195
|
|
|
|
|
|
|
|
196
|
0
|
|
|
|
|
0
|
(undef, my $length) = unpack 'C1v1', $$buffref; |
197
|
0
|
0
|
|
|
|
0
|
return ($length + 4) - length $$buffref unless length $$buffref >= $length + 4; |
198
|
|
|
|
|
|
|
|
199
|
0
|
|
|
|
|
0
|
return 0; |
200
|
|
|
|
|
|
|
} |
201
|
|
|
|
|
|
|
|
202
|
|
|
|
|
|
|
=head2 build_request |
203
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
Builds an UWSGI request using the given modifier, defaulting |
205
|
|
|
|
|
|
|
to modifier1 == 5 and modifier2 == 0, i.e. PSGI request. |
206
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
Takes the following named parameters: |
208
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
=over 4 |
210
|
|
|
|
|
|
|
|
211
|
|
|
|
|
|
|
=item * modifier1 - the modifier1 value, defaults to 5 if not provided |
212
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
=item * modifier2 - the modifier2 value, defaults to 0 if not provided |
214
|
|
|
|
|
|
|
|
215
|
|
|
|
|
|
|
=item * method - the HTTP request method |
216
|
|
|
|
|
|
|
|
217
|
|
|
|
|
|
|
=item * uri - which L we're requesting, can be passed as a plain string |
218
|
|
|
|
|
|
|
in which case we'll upgrade to a L object internally |
219
|
|
|
|
|
|
|
|
220
|
|
|
|
|
|
|
=item * headers - a hashref of HTTP headers, e.g. { 'Content-Type' => 'text/html' } |
221
|
|
|
|
|
|
|
|
222
|
|
|
|
|
|
|
=back |
223
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
Returns a scalar containing packet data or raises an exception on failure. |
225
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
=cut |
227
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
sub build_request { |
229
|
5
|
|
|
5
|
1
|
4269
|
my %args = @_; |
230
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
# my $type = delete $args{type} or die 'no type provided'; |
232
|
5
|
50
|
|
|
|
19
|
my $uri = delete $args{uri} or die 'no URI provided'; |
233
|
5
|
50
|
|
|
|
29
|
$uri = URI->new($uri) unless ref $uri; |
234
|
|
|
|
|
|
|
|
235
|
5
|
|
|
|
|
8816
|
my %env; |
236
|
5
|
|
|
|
|
20
|
$env{REQUEST_METHOD} = uc delete $args{method}; |
237
|
5
|
|
|
|
|
28
|
$env{UWSGI_SCHEME} = $uri->scheme; |
238
|
5
|
|
|
|
|
381
|
$env{HTTP_HOST} = $uri->host; |
239
|
5
|
|
50
|
|
|
147
|
$env{SERVER_PORT} = $uri->port // 80; |
240
|
5
|
|
|
|
|
120
|
$env{PATH_INFO} = $uri->path; |
241
|
5
|
50
|
|
|
|
61
|
$env{QUERY_STRING} = $uri->query if defined $uri->query; |
242
|
5
|
50
|
|
|
|
68
|
@env{qw(REMOTE_ADDR REMOTE_PORT)} = split ':', delete $args{remote}, 2 if $args{remote}; |
243
|
|
|
|
|
|
|
|
244
|
5
|
|
100
|
|
|
25
|
$args{headers} ||= {}; |
245
|
5
|
|
|
|
|
5
|
foreach my $k (keys %{$args{headers}}) { |
|
5
|
|
|
|
|
18
|
|
246
|
3
|
|
|
|
|
6
|
(my $env_k = uc $k) =~ tr/-/_/; |
247
|
3
|
|
50
|
|
|
14
|
$env{"HTTP_$env_k"} = $args{headers}{$k} // ''; |
248
|
|
|
|
|
|
|
} |
249
|
5
|
|
|
|
|
13
|
delete $args{headers}; |
250
|
|
|
|
|
|
|
|
251
|
5
|
|
|
|
|
13
|
my @modifier = delete @args{qw(modifier1 modifier2)}; |
252
|
5
|
|
|
|
|
9
|
my $data = ''; |
253
|
5
|
|
|
|
|
36
|
%env = (%args, %env); |
254
|
5
|
|
|
|
|
31
|
foreach my $k (sort keys %env) { |
255
|
38
|
50
|
|
|
|
589
|
die "Undef value found for $k" unless defined $env{$k}; |
256
|
38
|
|
|
|
|
49
|
$data .= pack 'v1/av1/a', map { Encode::encode('utf8', $_) } $k, $env{$k}; |
|
76
|
|
|
|
|
726
|
|
257
|
|
|
|
|
|
|
} |
258
|
|
|
|
|
|
|
|
259
|
5
|
|
50
|
|
|
172
|
return pack('C1v1C1', |
|
|
|
50
|
|
|
|
|
260
|
|
|
|
|
|
|
$modifier[0] // PSGI_MODIFIER1, |
261
|
|
|
|
|
|
|
length($data), |
262
|
|
|
|
|
|
|
$modifier[1] // PSGI_MODIFIER2, |
263
|
|
|
|
|
|
|
) . $data; |
264
|
|
|
|
|
|
|
} |
265
|
|
|
|
|
|
|
|
266
|
|
|
|
|
|
|
=head2 extract_modifier |
267
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
Used internally to extract and handle the modifier-specific data. |
269
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
=cut |
271
|
|
|
|
|
|
|
|
272
|
|
|
|
|
|
|
sub extract_modifier { |
273
|
0
|
|
|
0
|
1
|
0
|
my %args = @_; |
274
|
|
|
|
|
|
|
|
275
|
0
|
0
|
|
|
|
0
|
die "Unsupported modifier1 $args{modifier1}" unless $args{modifier1} == PSGI_MODIFIER1; |
276
|
|
|
|
|
|
|
|
277
|
0
|
0
|
|
|
|
0
|
my $len = delete $args{length} or die "no length found"; |
278
|
0
|
0
|
|
|
|
0
|
my $buffer = delete $args{buffer} or die "no buffer found"; |
279
|
0
|
|
|
|
|
0
|
my %env; |
280
|
0
|
|
|
|
|
0
|
while($len) { |
281
|
0
|
|
|
|
|
0
|
my ($k, $v) = unpack 'v1/a*v1/a*', $$buffer; |
282
|
0
|
|
|
|
|
0
|
$env{$k} = $v; |
283
|
0
|
|
|
|
|
0
|
my $sublen = 4 + length($k) + length($v); |
284
|
0
|
|
|
|
|
0
|
substr $$buffer, 0, $sublen, ''; |
285
|
0
|
|
|
|
|
0
|
$len -= $sublen; |
286
|
|
|
|
|
|
|
} |
287
|
0
|
|
|
|
|
0
|
return \%env; |
288
|
|
|
|
|
|
|
} |
289
|
|
|
|
|
|
|
|
290
|
|
|
|
|
|
|
=head2 uri_from_env |
291
|
|
|
|
|
|
|
|
292
|
|
|
|
|
|
|
Returns a L object parsed from a request ("environment"). |
293
|
|
|
|
|
|
|
|
294
|
|
|
|
|
|
|
=cut |
295
|
|
|
|
|
|
|
|
296
|
|
|
|
|
|
|
sub uri_from_env { |
297
|
4
|
|
|
4
|
1
|
7
|
my ($env) = @_; |
298
|
4
|
|
|
|
|
15
|
my $uri = $env->{UWSGI_SCHEME} . '://' . $env->{HTTP_HOST} . ':' . $env->{SERVER_PORT} . $env->{PATH_INFO}; |
299
|
4
|
50
|
50
|
|
|
21
|
$uri .= '?' . $env->{QUERY_STRING} if length($env->{QUERY_STRING} // ''); |
300
|
4
|
|
|
|
|
33
|
return URI->new($uri); |
301
|
|
|
|
|
|
|
} |
302
|
|
|
|
|
|
|
|
303
|
|
|
|
|
|
|
1; |
304
|
|
|
|
|
|
|
|
305
|
|
|
|
|
|
|
__END__ |