| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
################################################################################ |
|
2
|
|
|
|
|
|
|
# MogileFS::HTTPFile object |
|
3
|
|
|
|
|
|
|
# NOTE: This is meant to be used within IO::WrapTie... |
|
4
|
|
|
|
|
|
|
# |
|
5
|
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
package MogileFS::NewHTTPFile; |
|
7
|
|
|
|
|
|
|
|
|
8
|
4
|
|
|
4
|
|
23
|
use strict; |
|
|
4
|
|
|
|
|
9
|
|
|
|
4
|
|
|
|
|
162
|
|
|
9
|
4
|
|
|
4
|
|
21
|
no strict 'refs'; |
|
|
4
|
|
|
|
|
8
|
|
|
|
4
|
|
|
|
|
96
|
|
|
10
|
|
|
|
|
|
|
|
|
11
|
4
|
|
|
4
|
|
19
|
use Carp; |
|
|
4
|
|
|
|
|
9
|
|
|
|
4
|
|
|
|
|
277
|
|
|
12
|
4
|
|
|
4
|
|
22
|
use POSIX qw( EAGAIN ); |
|
|
4
|
|
|
|
|
7
|
|
|
|
4
|
|
|
|
|
66
|
|
|
13
|
4
|
|
|
4
|
|
5361
|
use Socket qw( PF_INET SOCK_STREAM ); |
|
|
4
|
|
|
|
|
9
|
|
|
|
4
|
|
|
|
|
220
|
|
|
14
|
4
|
|
|
4
|
|
23
|
use Errno qw( EINPROGRESS EISCONN ); |
|
|
4
|
|
|
|
|
7
|
|
|
|
4
|
|
|
|
|
221
|
|
|
15
|
|
|
|
|
|
|
|
|
16
|
4
|
|
|
4
|
|
21
|
use vars qw($PROTO_TCP); |
|
|
4
|
|
|
|
|
9
|
|
|
|
4
|
|
|
|
|
314
|
|
|
17
|
|
|
|
|
|
|
|
|
18
|
4
|
|
|
|
|
40
|
use fields ('host', |
|
19
|
|
|
|
|
|
|
'sock', # IO::Socket; created only when we need it |
|
20
|
|
|
|
|
|
|
'uri', |
|
21
|
|
|
|
|
|
|
'data', # buffered data we have |
|
22
|
|
|
|
|
|
|
'pos', # simulated file position |
|
23
|
|
|
|
|
|
|
'length', # length of data field |
|
24
|
|
|
|
|
|
|
'content_length', # declared length of data we will be receiving (not required) |
|
25
|
|
|
|
|
|
|
'mg', |
|
26
|
|
|
|
|
|
|
'fid', |
|
27
|
|
|
|
|
|
|
'devid', |
|
28
|
|
|
|
|
|
|
'class', |
|
29
|
|
|
|
|
|
|
'key', |
|
30
|
|
|
|
|
|
|
'path', # full URL to save data to |
|
31
|
|
|
|
|
|
|
'backup_dests', |
|
32
|
|
|
|
|
|
|
'bytes_out', # count of how many bytes we've written to the socket |
|
33
|
|
|
|
|
|
|
'data_in', # storage for data we've read from the socket |
|
34
|
|
|
|
|
|
|
'create_close_args', # Extra arguments hashref for the do_request of create_close during CLOSE |
|
35
|
4
|
|
|
4
|
|
20
|
); |
|
|
4
|
|
|
|
|
6
|
|
|
36
|
|
|
|
|
|
|
|
|
37
|
0
|
|
|
0
|
0
|
|
sub path { _getset(shift, 'path'); } |
|
38
|
0
|
|
|
0
|
0
|
|
sub class { _getset(shift, 'class', @_); } |
|
39
|
0
|
|
|
0
|
0
|
|
sub key { _getset(shift, 'key', @_); } |
|
40
|
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
sub _parse_url { |
|
42
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
43
|
0
|
|
|
|
|
|
my $url = shift; |
|
44
|
0
|
0
|
|
|
|
|
return 0 unless $url =~ m!http://(.+?)(/.+)$!; |
|
45
|
0
|
|
|
|
|
|
$self->{host} = $1; |
|
46
|
0
|
|
|
|
|
|
$self->{uri} = $2; |
|
47
|
0
|
|
|
|
|
|
$self->{path} = $url; |
|
48
|
0
|
|
|
|
|
|
return 1; |
|
49
|
|
|
|
|
|
|
} |
|
50
|
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
sub TIEHANDLE { |
|
52
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
53
|
0
|
0
|
|
|
|
|
$self = fields::new($self) unless ref $self; |
|
54
|
|
|
|
|
|
|
|
|
55
|
0
|
|
|
|
|
|
my %args = @_; |
|
56
|
0
|
0
|
|
|
|
|
return undef unless $self->_parse_url($args{path}); |
|
57
|
|
|
|
|
|
|
|
|
58
|
0
|
|
|
|
|
|
$self->{data} = ''; |
|
59
|
0
|
|
|
|
|
|
$self->{length} = 0; |
|
60
|
0
|
|
0
|
|
|
|
$self->{backup_dests} = $args{backup_dests} || []; |
|
61
|
0
|
|
|
|
|
|
$self->{content_length} = $args{content_length} + 0; |
|
62
|
0
|
|
|
|
|
|
$self->{pos} = 0; |
|
63
|
0
|
|
|
|
|
|
$self->{$_} = $args{$_} foreach qw(mg fid devid class key); |
|
64
|
0
|
|
|
|
|
|
$self->{bytes_out} = 0; |
|
65
|
0
|
|
|
|
|
|
$self->{data_in} = ''; |
|
66
|
0
|
|
0
|
|
|
|
$self->{create_close_args} = $args{create_close_args} || {}; |
|
67
|
|
|
|
|
|
|
|
|
68
|
0
|
|
|
|
|
|
return $self; |
|
69
|
|
|
|
|
|
|
} |
|
70
|
|
|
|
|
|
|
*new = *TIEHANDLE; |
|
71
|
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
sub _sock_to_host { # (host) |
|
73
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
74
|
0
|
|
|
|
|
|
my $host = shift; |
|
75
|
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
# setup |
|
77
|
0
|
|
|
|
|
|
my ($ip, $port) = $host =~ /^(.*):(\d+)$/; |
|
78
|
0
|
|
|
|
|
|
my $sock = "Sock_$host"; |
|
79
|
0
|
|
0
|
|
|
|
my $proto = $PROTO_TCP ||= getprotobyname('tcp'); |
|
80
|
0
|
|
|
|
|
|
my $sin; |
|
81
|
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
# create the socket |
|
83
|
0
|
|
|
|
|
|
socket($sock, PF_INET, SOCK_STREAM, $proto); |
|
84
|
0
|
|
|
|
|
|
$sin = Socket::sockaddr_in($port, Socket::inet_aton($ip)); |
|
85
|
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
# unblock the socket |
|
87
|
0
|
|
|
|
|
|
IO::Handle::blocking($sock, 0); |
|
88
|
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
# attempt a connection |
|
90
|
0
|
|
|
|
|
|
my $ret = connect($sock, $sin); |
|
91
|
0
|
0
|
0
|
|
|
|
if (!$ret && $! == EINPROGRESS) { |
|
92
|
0
|
|
|
|
|
|
my $win = ''; |
|
93
|
0
|
|
|
|
|
|
vec($win, fileno($sock), 1) = 1; |
|
94
|
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
# watch for writeability |
|
96
|
0
|
0
|
|
|
|
|
if (select(undef, $win, undef, 3) > 0) { |
|
97
|
0
|
|
|
|
|
|
$ret = connect($sock, $sin); |
|
98
|
|
|
|
|
|
|
|
|
99
|
|
|
|
|
|
|
# EISCONN means connected & won't re-connect, so success |
|
100
|
0
|
0
|
0
|
|
|
|
$ret = 1 if !$ret && $! == EISCONN; |
|
101
|
|
|
|
|
|
|
} |
|
102
|
|
|
|
|
|
|
} |
|
103
|
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
# just throw back the socket we have |
|
105
|
0
|
0
|
|
|
|
|
return $sock if $ret; |
|
106
|
0
|
|
|
|
|
|
return undef; |
|
107
|
|
|
|
|
|
|
} |
|
108
|
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
sub _connect_sock { |
|
110
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
111
|
0
|
0
|
|
|
|
|
return 1 if $self->{sock}; |
|
112
|
|
|
|
|
|
|
|
|
113
|
0
|
|
|
|
|
|
my @down_hosts; |
|
114
|
|
|
|
|
|
|
|
|
115
|
0
|
|
0
|
|
|
|
while (!$self->{sock} && $self->{host}) { |
|
116
|
|
|
|
|
|
|
# attempt to connect |
|
117
|
0
|
0
|
|
|
|
|
return 1 if |
|
118
|
|
|
|
|
|
|
$self->{sock} = $self->_sock_to_host($self->{host}); |
|
119
|
|
|
|
|
|
|
|
|
120
|
0
|
|
|
|
|
|
push @down_hosts, $self->{host}; |
|
121
|
0
|
0
|
|
|
|
|
if (my $dest = shift @{$self->{backup_dests}}) { |
|
|
0
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
# dest is [$devid,$path] |
|
123
|
0
|
|
|
|
|
|
_debug("connecting to $self->{host} (dev $self->{devid}) failed; now trying $dest->[1] (dev $dest->[0])"); |
|
124
|
0
|
0
|
|
|
|
|
$self->_parse_url($dest->[1]) or _fail("bogus URL"); |
|
125
|
0
|
|
|
|
|
|
$self->{devid} = $dest->[0]; |
|
126
|
|
|
|
|
|
|
} else { |
|
127
|
0
|
|
|
|
|
|
$self->{host} = undef; |
|
128
|
|
|
|
|
|
|
} |
|
129
|
|
|
|
|
|
|
} |
|
130
|
|
|
|
|
|
|
|
|
131
|
0
|
|
|
|
|
|
_fail("unable to open socket to storage node (tried: @down_hosts): $!"); |
|
132
|
|
|
|
|
|
|
} |
|
133
|
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
# abstracted read; implements what ends up being a blocking read but |
|
135
|
|
|
|
|
|
|
# does it in terms of non-blocking operations. |
|
136
|
|
|
|
|
|
|
sub _getline { |
|
137
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
138
|
0
|
|
0
|
|
|
|
my $timeout = shift || 3; |
|
139
|
0
|
0
|
|
|
|
|
return undef unless $self->{sock}; |
|
140
|
|
|
|
|
|
|
|
|
141
|
|
|
|
|
|
|
# short cut if we already have data read |
|
142
|
0
|
0
|
|
|
|
|
if ($self->{data_in} =~ s/^(.*?\r?\n)//) { |
|
143
|
0
|
|
|
|
|
|
return $1; |
|
144
|
|
|
|
|
|
|
} |
|
145
|
|
|
|
|
|
|
|
|
146
|
0
|
|
|
|
|
|
my $rin = ''; |
|
147
|
0
|
|
|
|
|
|
vec($rin, fileno($self->{sock}), 1) = 1; |
|
148
|
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
# nope, we have to read a line |
|
150
|
0
|
|
|
|
|
|
my $nfound; |
|
151
|
0
|
|
|
|
|
|
my $t1 = Time::HiRes::time(); |
|
152
|
0
|
|
|
|
|
|
while ($nfound = select($rin, undef, undef, $timeout)) { |
|
153
|
0
|
|
|
|
|
|
my $data; |
|
154
|
0
|
|
|
|
|
|
my $bytesin = sysread($self->{sock}, $data, 1024); |
|
155
|
0
|
0
|
|
|
|
|
if (defined $bytesin) { |
|
156
|
|
|
|
|
|
|
# we can also get 0 here, which means EOF. no error, but no data. |
|
157
|
0
|
0
|
|
|
|
|
$self->{data_in} .= $data if $bytesin; |
|
158
|
|
|
|
|
|
|
} else { |
|
159
|
0
|
0
|
|
|
|
|
next if $! == EAGAIN; |
|
160
|
0
|
|
|
|
|
|
_fail("error reading from node for device $self->{devid}: $!"); |
|
161
|
|
|
|
|
|
|
} |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
# return a line if we got one |
|
164
|
0
|
0
|
|
|
|
|
if ($self->{data_in} =~ s/^(.*?\r?\n)//) { |
|
165
|
0
|
|
|
|
|
|
return $1; |
|
166
|
|
|
|
|
|
|
} |
|
167
|
|
|
|
|
|
|
|
|
168
|
|
|
|
|
|
|
# and if we got no data, it's time to return EOF |
|
169
|
0
|
0
|
|
|
|
|
unless ($bytesin) { |
|
170
|
0
|
|
|
|
|
|
$@ = "\$bytesin is 0"; |
|
171
|
0
|
|
|
|
|
|
return undef; |
|
172
|
|
|
|
|
|
|
} |
|
173
|
|
|
|
|
|
|
} |
|
174
|
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
# if we got here, nothing was readable in our time limit |
|
176
|
0
|
|
|
|
|
|
my $t2 = Time::HiRes::time(); |
|
177
|
0
|
|
|
|
|
|
$@ = sprintf("not readable in %0.02f seconds", $t2-$t1); |
|
178
|
0
|
|
|
|
|
|
return undef; |
|
179
|
|
|
|
|
|
|
} |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
# abstracted write function that uses non-blocking I/O and checking for |
|
182
|
|
|
|
|
|
|
# writeability to ensure that we don't get stuck doing a write if the |
|
183
|
|
|
|
|
|
|
# node we're talking to goes down. also handles logic to fall back to |
|
184
|
|
|
|
|
|
|
# a backup node if we're on our first write and the first node is down. |
|
185
|
|
|
|
|
|
|
# this entire function is a blocking function, it just uses intelligent |
|
186
|
|
|
|
|
|
|
# non-blocking write functionality. |
|
187
|
|
|
|
|
|
|
# |
|
188
|
|
|
|
|
|
|
# this function returns success (1) or it croaks on failure. |
|
189
|
|
|
|
|
|
|
sub _write { |
|
190
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
191
|
0
|
0
|
|
|
|
|
return undef unless $self->{sock}; |
|
192
|
|
|
|
|
|
|
|
|
193
|
0
|
|
|
|
|
|
my $win = ''; |
|
194
|
0
|
|
|
|
|
|
vec($win, fileno($self->{sock}), 1) = 1; |
|
195
|
|
|
|
|
|
|
|
|
196
|
|
|
|
|
|
|
# setup data and counters |
|
197
|
0
|
|
|
|
|
|
my $data = shift(); |
|
198
|
0
|
|
|
|
|
|
my $bytesleft = length($data); |
|
199
|
0
|
|
|
|
|
|
my $bytessent = 0; |
|
200
|
|
|
|
|
|
|
|
|
201
|
|
|
|
|
|
|
# main sending loop for data, will keep looping until all of the data |
|
202
|
|
|
|
|
|
|
# we've been asked to send is sent |
|
203
|
0
|
|
|
|
|
|
my $nfound; |
|
204
|
0
|
|
0
|
|
|
|
while ($bytesleft && ($nfound = select(undef, $win, undef, 3))) { |
|
205
|
0
|
|
|
|
|
|
my $bytesout = syswrite($self->{sock}, $data, $bytesleft, $bytessent); |
|
206
|
0
|
0
|
|
|
|
|
if (defined $bytesout) { |
|
207
|
|
|
|
|
|
|
# update our myriad counters |
|
208
|
0
|
|
|
|
|
|
$bytessent += $bytesout; |
|
209
|
0
|
|
|
|
|
|
$self->{bytes_out} += $bytesout; |
|
210
|
0
|
|
|
|
|
|
$bytesleft -= $bytesout; |
|
211
|
|
|
|
|
|
|
} else { |
|
212
|
|
|
|
|
|
|
# if we get EAGAIN, restart the select loop, else fail |
|
213
|
0
|
0
|
|
|
|
|
next if $! == EAGAIN; |
|
214
|
0
|
|
|
|
|
|
_fail("error writing to node for device $self->{devid}: $!"); |
|
215
|
|
|
|
|
|
|
} |
|
216
|
|
|
|
|
|
|
} |
|
217
|
0
|
0
|
|
|
|
|
return 1 unless $bytesleft; |
|
218
|
|
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
# at this point, we had a socket error, since we have bytes left, and |
|
220
|
|
|
|
|
|
|
# the loop above didn't finish sending them. if this was our first |
|
221
|
|
|
|
|
|
|
# write, let's try to fall back to a different host. |
|
222
|
0
|
0
|
|
|
|
|
unless ($self->{bytes_out}) { |
|
223
|
0
|
0
|
|
|
|
|
if (my $dest = shift @{$self->{backup_dests}}) { |
|
|
0
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
# dest is [$devid,$path] |
|
225
|
0
|
0
|
|
|
|
|
$self->_parse_url($dest->[1]) or _fail("bogus URL"); |
|
226
|
0
|
|
|
|
|
|
$self->{devid} = $dest->[0]; |
|
227
|
0
|
|
|
|
|
|
$self->_connect_sock; |
|
228
|
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
# now repass this write to try again |
|
230
|
0
|
|
|
|
|
|
return $self->_write($data); |
|
231
|
|
|
|
|
|
|
} |
|
232
|
|
|
|
|
|
|
} |
|
233
|
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
# total failure (croak) |
|
235
|
0
|
|
|
|
|
|
$self->{sock} = undef; |
|
236
|
0
|
|
|
|
|
|
_fail(sprintf("unable to write to any allocated storage node, last tried dev %s on host %s uri %s. Had sent %s bytes, %s bytes left", $self->{devid}, $self->{host}, $self->{uri}, $self->{bytes_out}, $bytesleft)); |
|
237
|
|
|
|
|
|
|
} |
|
238
|
|
|
|
|
|
|
|
|
239
|
|
|
|
|
|
|
sub PRINT { |
|
240
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
241
|
|
|
|
|
|
|
|
|
242
|
|
|
|
|
|
|
# get data to send to server |
|
243
|
0
|
|
|
|
|
|
my $data = shift; |
|
244
|
0
|
|
|
|
|
|
my $newlen = length $data; |
|
245
|
0
|
|
|
|
|
|
$self->{pos} += $newlen; |
|
246
|
|
|
|
|
|
|
|
|
247
|
|
|
|
|
|
|
# now make socket if we don't have one |
|
248
|
0
|
0
|
0
|
|
|
|
if (!$self->{sock} && $self->{content_length}) { |
|
249
|
0
|
|
|
|
|
|
$self->_connect_sock; |
|
250
|
0
|
|
|
|
|
|
$self->_write("PUT $self->{uri} HTTP/1.0\r\nContent-length: $self->{content_length}\r\n\r\n"); |
|
251
|
|
|
|
|
|
|
} |
|
252
|
|
|
|
|
|
|
|
|
253
|
|
|
|
|
|
|
# write some data to our socket |
|
254
|
0
|
0
|
|
|
|
|
if ($self->{sock}) { |
|
255
|
|
|
|
|
|
|
# save the first 1024 bytes of data so that we can seek back to it |
|
256
|
|
|
|
|
|
|
# and do some work later |
|
257
|
0
|
0
|
|
|
|
|
if ($self->{length} < 1024) { |
|
258
|
0
|
0
|
|
|
|
|
if ($self->{length} + $newlen > 1024) { |
|
259
|
0
|
|
|
|
|
|
$self->{length} = 1024; |
|
260
|
0
|
|
|
|
|
|
$self->{data} .= substr($data, 0, 1024 - $self->{length}); |
|
261
|
|
|
|
|
|
|
} else { |
|
262
|
0
|
|
|
|
|
|
$self->{length} += $newlen; |
|
263
|
0
|
|
|
|
|
|
$self->{data} .= $data; |
|
264
|
|
|
|
|
|
|
} |
|
265
|
|
|
|
|
|
|
} |
|
266
|
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
# actually write |
|
268
|
0
|
|
|
|
|
|
$self->_write($data); |
|
269
|
|
|
|
|
|
|
} else { |
|
270
|
|
|
|
|
|
|
# or not, just stick it on our queued data |
|
271
|
0
|
|
|
|
|
|
$self->{data} .= $data; |
|
272
|
0
|
|
|
|
|
|
$self->{length} += $newlen; |
|
273
|
|
|
|
|
|
|
} |
|
274
|
|
|
|
|
|
|
} |
|
275
|
|
|
|
|
|
|
*print = *PRINT; |
|
276
|
|
|
|
|
|
|
|
|
277
|
|
|
|
|
|
|
sub CLOSE { |
|
278
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
279
|
|
|
|
|
|
|
|
|
280
|
|
|
|
|
|
|
# if we're closed and we have no sock... |
|
281
|
0
|
0
|
|
|
|
|
unless ($self->{sock}) { |
|
282
|
0
|
|
|
|
|
|
$self->_connect_sock; |
|
283
|
0
|
|
|
|
|
|
$self->_write("PUT $self->{uri} HTTP/1.0\r\nContent-length: $self->{length}\r\n\r\n"); |
|
284
|
0
|
|
|
|
|
|
$self->_write($self->{data}); |
|
285
|
|
|
|
|
|
|
} |
|
286
|
|
|
|
|
|
|
|
|
287
|
|
|
|
|
|
|
# set a message in $! and $@ |
|
288
|
|
|
|
|
|
|
my $err = sub { |
|
289
|
0
|
|
|
0
|
|
|
$@ = "$_[0]\n"; |
|
290
|
0
|
|
|
|
|
|
return undef; |
|
291
|
0
|
|
|
|
|
|
}; |
|
292
|
|
|
|
|
|
|
|
|
293
|
|
|
|
|
|
|
# get response from put |
|
294
|
0
|
0
|
|
|
|
|
if ($self->{sock}) { |
|
295
|
0
|
|
|
|
|
|
my $line = $self->_getline(6); # wait up to 6 seconds for response to PUT. |
|
296
|
|
|
|
|
|
|
|
|
297
|
0
|
0
|
|
|
|
|
return $err->("Unable to read response line from server ($self->{sock}) after PUT of $self->{length} to $self->{uri}. _getline says: $@") |
|
298
|
|
|
|
|
|
|
unless defined $line; |
|
299
|
|
|
|
|
|
|
|
|
300
|
0
|
0
|
|
|
|
|
if ($line =~ m!^HTTP/\d+\.\d+\s+(\d+)!) { |
|
301
|
|
|
|
|
|
|
# all 2xx responses are success |
|
302
|
0
|
0
|
0
|
|
|
|
unless ($1 >= 200 && $1 <= 299) { |
|
303
|
0
|
|
|
|
|
|
my $errcode = $1; |
|
304
|
|
|
|
|
|
|
# read through to the body |
|
305
|
0
|
|
|
|
|
|
my ($found_header, $body); |
|
306
|
0
|
|
|
|
|
|
while (defined (my $l = $self->_getline)) { |
|
307
|
|
|
|
|
|
|
# remove trailing stuff |
|
308
|
0
|
|
|
|
|
|
$l =~ s/[\r\n\s]+$//g; |
|
309
|
0
|
0
|
|
|
|
|
$found_header = 1 unless $l; |
|
310
|
0
|
0
|
|
|
|
|
next unless $found_header; |
|
311
|
|
|
|
|
|
|
|
|
312
|
|
|
|
|
|
|
# add line to the body, with a space for readability |
|
313
|
0
|
|
|
|
|
|
$body .= " $l"; |
|
314
|
|
|
|
|
|
|
} |
|
315
|
0
|
0
|
|
|
|
|
$body = substr($body, 0, 512) if length $body > 512; |
|
316
|
0
|
|
|
|
|
|
return $err->("HTTP response $errcode from upload of $self->{uri} to $self->{sock}: $body"); |
|
317
|
|
|
|
|
|
|
} |
|
318
|
|
|
|
|
|
|
} else { |
|
319
|
0
|
|
|
|
|
|
return $err->("Response line not understood from $self->{sock}: $line"); |
|
320
|
|
|
|
|
|
|
} |
|
321
|
0
|
|
|
|
|
|
$self->{sock}->close; |
|
322
|
|
|
|
|
|
|
} |
|
323
|
|
|
|
|
|
|
|
|
324
|
0
|
|
|
|
|
|
my MogileFS $mg = $self->{mg}; |
|
325
|
0
|
|
|
|
|
|
my $domain = $mg->{domain}; |
|
326
|
|
|
|
|
|
|
|
|
327
|
0
|
|
|
|
|
|
my $fid = $self->{fid}; |
|
328
|
0
|
|
|
|
|
|
my $devid = $self->{devid}; |
|
329
|
0
|
|
|
|
|
|
my $path = $self->{path}; |
|
330
|
|
|
|
|
|
|
|
|
331
|
0
|
|
|
|
|
|
my $create_close_args = $self->{create_close_args}; |
|
332
|
|
|
|
|
|
|
|
|
333
|
0
|
|
0
|
|
|
|
my $key = shift || $self->{key}; |
|
334
|
|
|
|
|
|
|
|
|
335
|
0
|
0
|
|
|
|
|
my $rv = $mg->{backend}->do_request |
|
336
|
|
|
|
|
|
|
("create_close", { |
|
337
|
|
|
|
|
|
|
%$create_close_args, |
|
338
|
|
|
|
|
|
|
fid => $fid, |
|
339
|
|
|
|
|
|
|
devid => $devid, |
|
340
|
|
|
|
|
|
|
domain => $domain, |
|
341
|
|
|
|
|
|
|
size => $self->{content_length} ? $self->{content_length} : $self->{length}, |
|
342
|
|
|
|
|
|
|
key => $key, |
|
343
|
|
|
|
|
|
|
path => $path, |
|
344
|
|
|
|
|
|
|
}); |
|
345
|
0
|
0
|
|
|
|
|
unless ($rv) { |
|
346
|
|
|
|
|
|
|
# set $@, as our callers expect $@ to contain the error message that |
|
347
|
|
|
|
|
|
|
# failed during a close. since we failed in the backend, we have to |
|
348
|
|
|
|
|
|
|
# do this manually. |
|
349
|
0
|
|
|
|
|
|
return $err->("$mg->{backend}->{lasterr}: $mg->{backend}->{lasterrstr}"); |
|
350
|
|
|
|
|
|
|
} |
|
351
|
|
|
|
|
|
|
|
|
352
|
0
|
|
|
|
|
|
return 1; |
|
353
|
|
|
|
|
|
|
} |
|
354
|
|
|
|
|
|
|
*close = *CLOSE; |
|
355
|
|
|
|
|
|
|
|
|
356
|
|
|
|
|
|
|
sub TELL { |
|
357
|
|
|
|
|
|
|
# return our current pos |
|
358
|
0
|
|
|
0
|
|
|
return $_[0]->{pos}; |
|
359
|
|
|
|
|
|
|
} |
|
360
|
|
|
|
|
|
|
*tell = *TELL; |
|
361
|
|
|
|
|
|
|
|
|
362
|
|
|
|
|
|
|
sub SEEK { |
|
363
|
|
|
|
|
|
|
# simply set pos... |
|
364
|
0
|
0
|
|
0
|
|
|
_fail("seek past end of file") if $_[1] > $_[0]->{length}; |
|
365
|
0
|
|
|
|
|
|
$_[0]->{pos} = $_[1]; |
|
366
|
|
|
|
|
|
|
} |
|
367
|
|
|
|
|
|
|
*seek = *SEEK; |
|
368
|
|
|
|
|
|
|
|
|
369
|
|
|
|
|
|
|
sub EOF { |
|
370
|
0
|
0
|
|
0
|
|
|
return ($_[0]->{pos} >= $_[0]->{length}) ? 1 : 0; |
|
371
|
|
|
|
|
|
|
} |
|
372
|
|
|
|
|
|
|
*eof = *EOF; |
|
373
|
|
|
|
|
|
|
|
|
374
|
0
|
|
|
0
|
|
|
sub BINMODE { |
|
375
|
|
|
|
|
|
|
# no-op, we're always in binary mode |
|
376
|
|
|
|
|
|
|
} |
|
377
|
|
|
|
|
|
|
*binmode = *BINMODE; |
|
378
|
|
|
|
|
|
|
|
|
379
|
|
|
|
|
|
|
sub READ { |
|
380
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
381
|
0
|
|
|
|
|
|
my $count = $_[1] + 0; |
|
382
|
|
|
|
|
|
|
|
|
383
|
0
|
|
|
|
|
|
my $max = $self->{length} - $self->{pos}; |
|
384
|
0
|
0
|
|
|
|
|
$max = $count if $count < $max; |
|
385
|
|
|
|
|
|
|
|
|
386
|
0
|
|
|
|
|
|
$_[0] = substr($self->{data}, $self->{pos}, $max); |
|
387
|
0
|
|
|
|
|
|
$self->{pos} += $max; |
|
388
|
|
|
|
|
|
|
|
|
389
|
0
|
|
|
|
|
|
return $max; |
|
390
|
|
|
|
|
|
|
} |
|
391
|
|
|
|
|
|
|
*read = *READ; |
|
392
|
|
|
|
|
|
|
|
|
393
|
|
|
|
|
|
|
|
|
394
|
|
|
|
|
|
|
################################################################################ |
|
395
|
|
|
|
|
|
|
# MogileFS::NewHTTPFile class methods |
|
396
|
|
|
|
|
|
|
# |
|
397
|
|
|
|
|
|
|
|
|
398
|
|
|
|
|
|
|
sub _fail { |
|
399
|
0
|
|
|
0
|
|
|
croak "MogileFS::NewHTTPFile: $_[0]"; |
|
400
|
|
|
|
|
|
|
} |
|
401
|
|
|
|
|
|
|
|
|
402
|
|
|
|
|
|
|
sub _debug { |
|
403
|
0
|
|
|
0
|
|
|
MogileFS::Client::_debug(@_); |
|
404
|
|
|
|
|
|
|
} |
|
405
|
|
|
|
|
|
|
|
|
406
|
|
|
|
|
|
|
sub _getset { |
|
407
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
408
|
0
|
|
|
|
|
|
my $what = shift; |
|
409
|
|
|
|
|
|
|
|
|
410
|
0
|
0
|
|
|
|
|
if (@_) { |
|
411
|
|
|
|
|
|
|
# note: we're a TIEHANDLE interface, so we're not QUITE like a |
|
412
|
|
|
|
|
|
|
# normal class... our parameters tend to come in via an arrayref |
|
413
|
0
|
|
|
|
|
|
my $val = shift; |
|
414
|
0
|
0
|
|
|
|
|
$val = shift(@$val) if ref $val eq 'ARRAY'; |
|
415
|
0
|
|
|
|
|
|
return $self->{$what} = $val; |
|
416
|
|
|
|
|
|
|
} else { |
|
417
|
0
|
|
|
|
|
|
return $self->{$what}; |
|
418
|
|
|
|
|
|
|
} |
|
419
|
|
|
|
|
|
|
} |
|
420
|
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
sub _fid { |
|
422
|
0
|
|
|
0
|
|
|
my MogileFS::NewHTTPFile $self = shift; |
|
423
|
0
|
|
|
|
|
|
return $self->{fid}; |
|
424
|
|
|
|
|
|
|
} |
|
425
|
|
|
|
|
|
|
|
|
426
|
|
|
|
|
|
|
1; |