rack 2.2.7 → 3.1.8
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +341 -78
- data/CONTRIBUTING.md +63 -55
- data/MIT-LICENSE +1 -1
- data/README.md +328 -0
- data/SPEC.rdoc +213 -136
- data/lib/rack/auth/abstract/handler.rb +3 -1
- data/lib/rack/auth/abstract/request.rb +3 -1
- data/lib/rack/auth/basic.rb +1 -4
- data/lib/rack/bad_request.rb +8 -0
- data/lib/rack/body_proxy.rb +21 -3
- data/lib/rack/builder.rb +102 -69
- data/lib/rack/cascade.rb +2 -3
- data/lib/rack/common_logger.rb +23 -18
- data/lib/rack/conditional_get.rb +18 -15
- data/lib/rack/constants.rb +67 -0
- data/lib/rack/content_length.rb +12 -16
- data/lib/rack/content_type.rb +8 -5
- data/lib/rack/deflater.rb +40 -26
- data/lib/rack/directory.rb +9 -3
- data/lib/rack/etag.rb +14 -23
- data/lib/rack/events.rb +4 -0
- data/lib/rack/files.rb +15 -17
- data/lib/rack/head.rb +9 -8
- data/lib/rack/headers.rb +238 -0
- data/lib/rack/lint.rb +866 -681
- data/lib/rack/lock.rb +2 -5
- data/lib/rack/logger.rb +3 -0
- data/lib/rack/media_type.rb +9 -4
- data/lib/rack/method_override.rb +5 -1
- data/lib/rack/mime.rb +14 -5
- data/lib/rack/mock.rb +1 -271
- data/lib/rack/mock_request.rb +161 -0
- data/lib/rack/mock_response.rb +124 -0
- data/lib/rack/multipart/generator.rb +7 -5
- data/lib/rack/multipart/parser.rb +217 -91
- data/lib/rack/multipart/uploaded_file.rb +4 -0
- data/lib/rack/multipart.rb +53 -40
- data/lib/rack/null_logger.rb +9 -0
- data/lib/rack/query_parser.rb +81 -102
- data/lib/rack/recursive.rb +2 -0
- data/lib/rack/reloader.rb +0 -2
- data/lib/rack/request.rb +260 -123
- data/lib/rack/response.rb +151 -66
- data/lib/rack/rewindable_input.rb +24 -5
- data/lib/rack/runtime.rb +7 -6
- data/lib/rack/sendfile.rb +30 -25
- data/lib/rack/show_exceptions.rb +21 -4
- data/lib/rack/show_status.rb +17 -7
- data/lib/rack/static.rb +8 -8
- data/lib/rack/tempfile_reaper.rb +15 -4
- data/lib/rack/urlmap.rb +3 -1
- data/lib/rack/utils.rb +240 -237
- data/lib/rack/version.rb +1 -9
- data/lib/rack.rb +13 -89
- metadata +15 -41
- data/README.rdoc +0 -320
- data/Rakefile +0 -130
- data/bin/rackup +0 -5
- data/contrib/rack.png +0 -0
- data/contrib/rack.svg +0 -150
- data/contrib/rack_logo.svg +0 -164
- data/contrib/rdoc.css +0 -412
- data/example/lobster.ru +0 -6
- data/example/protectedlobster.rb +0 -16
- data/example/protectedlobster.ru +0 -10
- data/lib/rack/auth/digest/md5.rb +0 -131
- data/lib/rack/auth/digest/nonce.rb +0 -54
- data/lib/rack/auth/digest/params.rb +0 -54
- data/lib/rack/auth/digest/request.rb +0 -43
- data/lib/rack/chunked.rb +0 -117
- data/lib/rack/core_ext/regexp.rb +0 -14
- data/lib/rack/file.rb +0 -7
- data/lib/rack/handler/cgi.rb +0 -59
- data/lib/rack/handler/fastcgi.rb +0 -100
- data/lib/rack/handler/lsws.rb +0 -61
- data/lib/rack/handler/scgi.rb +0 -71
- data/lib/rack/handler/thin.rb +0 -36
- data/lib/rack/handler/webrick.rb +0 -129
- data/lib/rack/handler.rb +0 -104
- data/lib/rack/lobster.rb +0 -70
- data/lib/rack/server.rb +0 -466
- data/lib/rack/session/abstract/id.rb +0 -523
- data/lib/rack/session/cookie.rb +0 -203
- data/lib/rack/session/memcache.rb +0 -10
- data/lib/rack/session/pool.rb +0 -85
- data/rack.gemspec +0 -46
@@ -0,0 +1,124 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'cgi/cookie'
|
4
|
+
require 'time'
|
5
|
+
|
6
|
+
require_relative 'response'
|
7
|
+
|
8
|
+
module Rack
|
9
|
+
# Rack::MockResponse provides useful helpers for testing your apps.
|
10
|
+
# Usually, you don't create the MockResponse on your own, but use
|
11
|
+
# MockRequest.
|
12
|
+
|
13
|
+
class MockResponse < Rack::Response
|
14
|
+
class << self
|
15
|
+
alias [] new
|
16
|
+
end
|
17
|
+
|
18
|
+
# Headers
|
19
|
+
attr_reader :original_headers, :cookies
|
20
|
+
|
21
|
+
# Errors
|
22
|
+
attr_accessor :errors
|
23
|
+
|
24
|
+
def initialize(status, headers, body, errors = nil)
|
25
|
+
@original_headers = headers
|
26
|
+
|
27
|
+
if errors
|
28
|
+
@errors = errors.string if errors.respond_to?(:string)
|
29
|
+
else
|
30
|
+
@errors = ""
|
31
|
+
end
|
32
|
+
|
33
|
+
super(body, status, headers)
|
34
|
+
|
35
|
+
@cookies = parse_cookies_from_header
|
36
|
+
buffered_body!
|
37
|
+
end
|
38
|
+
|
39
|
+
def =~(other)
|
40
|
+
body =~ other
|
41
|
+
end
|
42
|
+
|
43
|
+
def match(other)
|
44
|
+
body.match other
|
45
|
+
end
|
46
|
+
|
47
|
+
def body
|
48
|
+
return @buffered_body if defined?(@buffered_body)
|
49
|
+
|
50
|
+
# FIXME: apparently users of MockResponse expect the return value of
|
51
|
+
# MockResponse#body to be a string. However, the real response object
|
52
|
+
# returns the body as a list.
|
53
|
+
#
|
54
|
+
# See spec_showstatus.rb:
|
55
|
+
#
|
56
|
+
# should "not replace existing messages" do
|
57
|
+
# ...
|
58
|
+
# res.body.should == "foo!"
|
59
|
+
# end
|
60
|
+
buffer = @buffered_body = String.new
|
61
|
+
|
62
|
+
@body.each do |chunk|
|
63
|
+
buffer << chunk
|
64
|
+
end
|
65
|
+
|
66
|
+
return buffer
|
67
|
+
end
|
68
|
+
|
69
|
+
def empty?
|
70
|
+
[201, 204, 304].include? status
|
71
|
+
end
|
72
|
+
|
73
|
+
def cookie(name)
|
74
|
+
cookies.fetch(name, nil)
|
75
|
+
end
|
76
|
+
|
77
|
+
private
|
78
|
+
|
79
|
+
def parse_cookies_from_header
|
80
|
+
cookies = Hash.new
|
81
|
+
set_cookie_header = headers['set-cookie']
|
82
|
+
if set_cookie_header && !set_cookie_header.empty?
|
83
|
+
Array(set_cookie_header).each do |cookie|
|
84
|
+
cookie_name, cookie_filling = cookie.split('=', 2)
|
85
|
+
cookie_attributes = identify_cookie_attributes cookie_filling
|
86
|
+
parsed_cookie = CGI::Cookie.new(
|
87
|
+
'name' => cookie_name.strip,
|
88
|
+
'value' => cookie_attributes.fetch('value'),
|
89
|
+
'path' => cookie_attributes.fetch('path', nil),
|
90
|
+
'domain' => cookie_attributes.fetch('domain', nil),
|
91
|
+
'expires' => cookie_attributes.fetch('expires', nil),
|
92
|
+
'secure' => cookie_attributes.fetch('secure', false)
|
93
|
+
)
|
94
|
+
cookies.store(cookie_name, parsed_cookie)
|
95
|
+
end
|
96
|
+
end
|
97
|
+
cookies
|
98
|
+
end
|
99
|
+
|
100
|
+
def identify_cookie_attributes(cookie_filling)
|
101
|
+
cookie_bits = cookie_filling.split(';')
|
102
|
+
cookie_attributes = Hash.new
|
103
|
+
cookie_attributes.store('value', cookie_bits[0].strip)
|
104
|
+
cookie_bits.drop(1).each do |bit|
|
105
|
+
if bit.include? '='
|
106
|
+
cookie_attribute, attribute_value = bit.split('=', 2)
|
107
|
+
cookie_attributes.store(cookie_attribute.strip.downcase, attribute_value.strip)
|
108
|
+
end
|
109
|
+
if bit.include? 'secure'
|
110
|
+
cookie_attributes.store('secure', true)
|
111
|
+
end
|
112
|
+
end
|
113
|
+
|
114
|
+
if cookie_attributes.key? 'max-age'
|
115
|
+
cookie_attributes.store('expires', Time.now + cookie_attributes['max-age'].to_i)
|
116
|
+
elsif cookie_attributes.key? 'expires'
|
117
|
+
cookie_attributes.store('expires', Time.httpdate(cookie_attributes['expires']))
|
118
|
+
end
|
119
|
+
|
120
|
+
cookie_attributes
|
121
|
+
end
|
122
|
+
|
123
|
+
end
|
124
|
+
end
|
@@ -1,5 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require_relative 'uploaded_file'
|
4
|
+
|
3
5
|
module Rack
|
4
6
|
module Multipart
|
5
7
|
class Generator
|
@@ -74,12 +76,12 @@ module Rack
|
|
74
76
|
|
75
77
|
def content_for_tempfile(io, file, name)
|
76
78
|
length = ::File.stat(file.path).size if file.path
|
77
|
-
filename = "; filename=\"#{Utils.
|
79
|
+
filename = "; filename=\"#{Utils.escape_path(file.original_filename)}\""
|
78
80
|
<<-EOF
|
79
81
|
--#{MULTIPART_BOUNDARY}\r
|
80
|
-
|
81
|
-
|
82
|
-
#{"
|
82
|
+
content-disposition: form-data; name="#{name}"#{filename}\r
|
83
|
+
content-type: #{file.content_type}\r
|
84
|
+
#{"content-length: #{length}\r\n" if length}\r
|
83
85
|
#{io.read}\r
|
84
86
|
EOF
|
85
87
|
end
|
@@ -87,7 +89,7 @@ EOF
|
|
87
89
|
def content_for_other(file, name)
|
88
90
|
<<-EOF
|
89
91
|
--#{MULTIPART_BOUNDARY}\r
|
90
|
-
|
92
|
+
content-disposition: form-data; name="#{name}"\r
|
91
93
|
\r
|
92
94
|
#{file}\r
|
93
95
|
EOF
|
@@ -2,21 +2,48 @@
|
|
2
2
|
|
3
3
|
require 'strscan'
|
4
4
|
|
5
|
+
require_relative '../utils'
|
6
|
+
require_relative '../bad_request'
|
7
|
+
|
5
8
|
module Rack
|
6
9
|
module Multipart
|
7
|
-
class MultipartPartLimitError < Errno::EMFILE
|
8
|
-
|
10
|
+
class MultipartPartLimitError < Errno::EMFILE
|
11
|
+
include BadRequest
|
12
|
+
end
|
9
13
|
|
10
|
-
class
|
11
|
-
|
14
|
+
class MultipartTotalPartLimitError < StandardError
|
15
|
+
include BadRequest
|
16
|
+
end
|
17
|
+
|
18
|
+
# Use specific error class when parsing multipart request
|
19
|
+
# that ends early.
|
20
|
+
class EmptyContentError < ::EOFError
|
21
|
+
include BadRequest
|
22
|
+
end
|
23
|
+
|
24
|
+
# Base class for multipart exceptions that do not subclass from
|
25
|
+
# other exception classes for backwards compatibility.
|
26
|
+
class BoundaryTooLongError < StandardError
|
27
|
+
include BadRequest
|
28
|
+
end
|
12
29
|
|
30
|
+
# Prefer to use the BoundaryTooLongError class or Rack::BadRequest.
|
31
|
+
Error = BoundaryTooLongError
|
32
|
+
|
33
|
+
EOL = "\r\n"
|
34
|
+
MULTIPART = %r|\Amultipart/.*boundary=\"?([^\";,]+)\"?|ni
|
35
|
+
MULTIPART_CONTENT_TYPE = /Content-Type: (.*)#{EOL}/ni
|
36
|
+
MULTIPART_CONTENT_DISPOSITION = /Content-Disposition:(.*)(?=#{EOL}(\S|\z))/ni
|
37
|
+
MULTIPART_CONTENT_ID = /Content-ID:\s*([^#{EOL}]*)/ni
|
38
|
+
|
39
|
+
class Parser
|
13
40
|
BUFSIZE = 1_048_576
|
14
41
|
TEXT_PLAIN = "text/plain"
|
15
42
|
TEMPFILE_FACTORY = lambda { |filename, content_type|
|
16
|
-
|
17
|
-
}
|
43
|
+
extension = ::File.extname(filename.gsub("\0", '%00'))[0, 129]
|
18
44
|
|
19
|
-
|
45
|
+
Tempfile.new(["RackMultipart", extension])
|
46
|
+
}
|
20
47
|
|
21
48
|
class BoundedIO # :nodoc:
|
22
49
|
def initialize(io, content_length)
|
@@ -39,16 +66,12 @@ module Rack
|
|
39
66
|
if str
|
40
67
|
@cursor += str.bytesize
|
41
68
|
else
|
42
|
-
# Raise an error for mismatching
|
69
|
+
# Raise an error for mismatching content-length and actual contents
|
43
70
|
raise EOFError, "bad content body"
|
44
71
|
end
|
45
72
|
|
46
73
|
str
|
47
74
|
end
|
48
|
-
|
49
|
-
def rewind
|
50
|
-
@io.rewind
|
51
|
-
end
|
52
75
|
end
|
53
76
|
|
54
77
|
MultipartInfo = Struct.new :params, :tmp_files
|
@@ -67,18 +90,17 @@ module Rack
|
|
67
90
|
boundary = parse_boundary content_type
|
68
91
|
return EMPTY unless boundary
|
69
92
|
|
93
|
+
if boundary.length > 70
|
94
|
+
# RFC 1521 Section 7.2.1 imposes a 70 character maximum for the boundary.
|
95
|
+
# Most clients use no more than 55 characters.
|
96
|
+
raise BoundaryTooLongError, "multipart boundary size too large (#{boundary.length} characters)"
|
97
|
+
end
|
98
|
+
|
70
99
|
io = BoundedIO.new(io, content_length) if content_length
|
71
|
-
outbuf = String.new
|
72
100
|
|
73
101
|
parser = new(boundary, tmpfile, bufsize, qp)
|
74
|
-
parser.
|
75
|
-
|
76
|
-
loop do
|
77
|
-
break if parser.state == :DONE
|
78
|
-
parser.on_read io.read(bufsize, outbuf)
|
79
|
-
end
|
102
|
+
parser.parse(io)
|
80
103
|
|
81
|
-
io.rewind
|
82
104
|
parser.result
|
83
105
|
end
|
84
106
|
|
@@ -178,32 +200,48 @@ module Rack
|
|
178
200
|
def initialize(boundary, tempfile, bufsize, query_parser)
|
179
201
|
@query_parser = query_parser
|
180
202
|
@params = query_parser.make_params
|
181
|
-
@boundary = "--#{boundary}"
|
182
203
|
@bufsize = bufsize
|
183
204
|
|
184
|
-
@full_boundary = @boundary
|
185
|
-
@end_boundary = @boundary + '--'
|
186
205
|
@state = :FAST_FORWARD
|
187
206
|
@mime_index = 0
|
188
207
|
@collector = Collector.new tempfile
|
189
208
|
|
190
209
|
@sbuf = StringScanner.new("".dup)
|
191
|
-
@body_regex = /(?:#{EOL})
|
192
|
-
@
|
210
|
+
@body_regex = /(?:#{EOL}|\A)--#{Regexp.quote(boundary)}(?:#{EOL}|--)/m
|
211
|
+
@body_regex_at_end = /#{@body_regex}\z/m
|
212
|
+
@end_boundary_size = boundary.bytesize + 4 # (-- at start, -- at finish)
|
213
|
+
@rx_max_size = boundary.bytesize + 6 # (\r\n-- at start, either \r\n or -- at finish)
|
193
214
|
@head_regex = /(.*?#{EOL})#{EOL}/m
|
194
215
|
end
|
195
216
|
|
196
|
-
def
|
197
|
-
|
198
|
-
|
199
|
-
|
217
|
+
def parse(io)
|
218
|
+
outbuf = String.new
|
219
|
+
read_data(io, outbuf)
|
220
|
+
|
221
|
+
loop do
|
222
|
+
status =
|
223
|
+
case @state
|
224
|
+
when :FAST_FORWARD
|
225
|
+
handle_fast_forward
|
226
|
+
when :CONSUME_TOKEN
|
227
|
+
handle_consume_token
|
228
|
+
when :MIME_HEAD
|
229
|
+
handle_mime_head
|
230
|
+
when :MIME_BODY
|
231
|
+
handle_mime_body
|
232
|
+
else # when :DONE
|
233
|
+
return
|
234
|
+
end
|
235
|
+
|
236
|
+
read_data(io, outbuf) if status == :want_read
|
237
|
+
end
|
200
238
|
end
|
201
239
|
|
202
240
|
def result
|
203
241
|
@collector.each do |part|
|
204
242
|
part.get_data do |data|
|
205
243
|
tag_multipart_encoding(part.filename, part.content_type, part.name, data)
|
206
|
-
@query_parser.normalize_params(@params, part.name, data
|
244
|
+
@query_parser.normalize_params(@params, part.name, data)
|
207
245
|
end
|
208
246
|
end
|
209
247
|
MultipartInfo.new @params.to_params_hash, @collector.find_all(&:file?).map(&:body)
|
@@ -211,29 +249,45 @@ module Rack
|
|
211
249
|
|
212
250
|
private
|
213
251
|
|
214
|
-
def
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
break if handle_mime_body == :want_read
|
225
|
-
when :DONE
|
226
|
-
break
|
227
|
-
end
|
228
|
-
end
|
252
|
+
def dequote(str) # From WEBrick::HTTPUtils
|
253
|
+
ret = (/\A"(.*)"\Z/ =~ str) ? $1 : str.dup
|
254
|
+
ret.gsub!(/\\(.)/, "\\1")
|
255
|
+
ret
|
256
|
+
end
|
257
|
+
|
258
|
+
def read_data(io, outbuf)
|
259
|
+
content = io.read(@bufsize, outbuf)
|
260
|
+
handle_empty_content!(content)
|
261
|
+
@sbuf.concat(content)
|
229
262
|
end
|
230
263
|
|
264
|
+
# This handles the initial parser state. We read until we find the starting
|
265
|
+
# boundary, then we can transition to the next state. If we find the ending
|
266
|
+
# boundary, this is an invalid multipart upload, but keep scanning for opening
|
267
|
+
# boundary in that case. If no boundary found, we need to keep reading data
|
268
|
+
# and retry. It's highly unlikely the initial read will not consume the
|
269
|
+
# boundary. The client would have to deliberately craft a response
|
270
|
+
# with the opening boundary beyond the buffer size for that to happen.
|
231
271
|
def handle_fast_forward
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
272
|
+
while true
|
273
|
+
case consume_boundary
|
274
|
+
when :BOUNDARY
|
275
|
+
# found opening boundary, transition to next state
|
276
|
+
@state = :MIME_HEAD
|
277
|
+
return
|
278
|
+
when :END_BOUNDARY
|
279
|
+
# invalid multipart upload
|
280
|
+
if @sbuf.pos == @end_boundary_size && @sbuf.rest == EOL
|
281
|
+
# stop parsing a buffer if a buffer is only an end boundary.
|
282
|
+
@state = :DONE
|
283
|
+
return
|
284
|
+
end
|
285
|
+
|
286
|
+
# retry for opening boundary
|
287
|
+
else
|
288
|
+
# no boundary found, keep reading data
|
289
|
+
return :want_read
|
290
|
+
end
|
237
291
|
end
|
238
292
|
end
|
239
293
|
|
@@ -247,17 +301,101 @@ module Rack
|
|
247
301
|
end
|
248
302
|
end
|
249
303
|
|
304
|
+
CONTENT_DISPOSITION_MAX_PARAMS = 16
|
305
|
+
CONTENT_DISPOSITION_MAX_BYTES = 1536
|
250
306
|
def handle_mime_head
|
251
307
|
if @sbuf.scan_until(@head_regex)
|
252
308
|
head = @sbuf[1]
|
253
309
|
content_type = head[MULTIPART_CONTENT_TYPE, 1]
|
254
|
-
if
|
255
|
-
|
310
|
+
if (disposition = head[MULTIPART_CONTENT_DISPOSITION, 1]) &&
|
311
|
+
disposition.bytesize <= CONTENT_DISPOSITION_MAX_BYTES
|
312
|
+
|
313
|
+
# ignore actual content-disposition value (should always be form-data)
|
314
|
+
i = disposition.index(';')
|
315
|
+
disposition.slice!(0, i+1)
|
316
|
+
param = nil
|
317
|
+
num_params = 0
|
318
|
+
|
319
|
+
# Parse parameter list
|
320
|
+
while i = disposition.index('=')
|
321
|
+
# Only parse up to max parameters, to avoid potential denial of service
|
322
|
+
num_params += 1
|
323
|
+
break if num_params > CONTENT_DISPOSITION_MAX_PARAMS
|
324
|
+
|
325
|
+
# Found end of parameter name, ensure forward progress in loop
|
326
|
+
param = disposition.slice!(0, i+1)
|
327
|
+
|
328
|
+
# Remove ending equals and preceding whitespace from parameter name
|
329
|
+
param.chomp!('=')
|
330
|
+
param.lstrip!
|
331
|
+
|
332
|
+
if disposition[0] == '"'
|
333
|
+
# Parameter value is quoted, parse it, handling backslash escapes
|
334
|
+
disposition.slice!(0, 1)
|
335
|
+
value = String.new
|
336
|
+
|
337
|
+
while i = disposition.index(/(["\\])/)
|
338
|
+
c = $1
|
339
|
+
|
340
|
+
# Append all content until ending quote or escape
|
341
|
+
value << disposition.slice!(0, i)
|
342
|
+
|
343
|
+
# Remove either backslash or ending quote,
|
344
|
+
# ensures forward progress in loop
|
345
|
+
disposition.slice!(0, 1)
|
346
|
+
|
347
|
+
# stop parsing parameter value if found ending quote
|
348
|
+
break if c == '"'
|
349
|
+
|
350
|
+
escaped_char = disposition.slice!(0, 1)
|
351
|
+
if param == 'filename' && escaped_char != '"'
|
352
|
+
# Possible IE uploaded filename, append both escape backslash and value
|
353
|
+
value << c << escaped_char
|
354
|
+
else
|
355
|
+
# Other only append escaped value
|
356
|
+
value << escaped_char
|
357
|
+
end
|
358
|
+
end
|
359
|
+
else
|
360
|
+
if i = disposition.index(';')
|
361
|
+
# Parameter value unquoted (which may be invalid), value ends at semicolon
|
362
|
+
value = disposition.slice!(0, i)
|
363
|
+
else
|
364
|
+
# If no ending semicolon, assume remainder of line is value and stop
|
365
|
+
# parsing
|
366
|
+
disposition.strip!
|
367
|
+
value = disposition
|
368
|
+
disposition = ''
|
369
|
+
end
|
370
|
+
end
|
371
|
+
|
372
|
+
case param
|
373
|
+
when 'name'
|
374
|
+
name = value
|
375
|
+
when 'filename'
|
376
|
+
filename = value
|
377
|
+
when 'filename*'
|
378
|
+
filename_star = value
|
379
|
+
# else
|
380
|
+
# ignore other parameters
|
381
|
+
end
|
382
|
+
|
383
|
+
# skip trailing semicolon, to proceed to next parameter
|
384
|
+
if i = disposition.index(';')
|
385
|
+
disposition.slice!(0, i+1)
|
386
|
+
end
|
387
|
+
end
|
256
388
|
else
|
257
389
|
name = head[MULTIPART_CONTENT_ID, 1]
|
258
390
|
end
|
259
391
|
|
260
|
-
|
392
|
+
if filename_star
|
393
|
+
encoding, _, filename = filename_star.split("'", 3)
|
394
|
+
filename = normalize_filename(filename || '')
|
395
|
+
filename.force_encoding(find_encoding(encoding))
|
396
|
+
elsif filename
|
397
|
+
filename = normalize_filename(filename)
|
398
|
+
end
|
261
399
|
|
262
400
|
if name.nil? || name.empty?
|
263
401
|
name = filename || "#{content_type || TEXT_PLAIN}[]".dup
|
@@ -272,7 +410,7 @@ module Rack
|
|
272
410
|
|
273
411
|
def handle_mime_body
|
274
412
|
if (body_with_boundary = @sbuf.check_until(@body_regex)) # check but do not advance the pointer yet
|
275
|
-
body = body_with_boundary.sub(
|
413
|
+
body = body_with_boundary.sub(@body_regex_at_end, '') # remove the boundary from the string
|
276
414
|
@collector.on_mime_body @mime_index, body
|
277
415
|
@sbuf.pos += body.length + 2 # skip \r\n after the content
|
278
416
|
@state = :CONSUME_TOKEN
|
@@ -289,54 +427,31 @@ module Rack
|
|
289
427
|
end
|
290
428
|
end
|
291
429
|
|
292
|
-
|
293
|
-
|
430
|
+
# Scan until the we find the start or end of the boundary.
|
431
|
+
# If we find it, return the appropriate symbol for the start or
|
432
|
+
# end of the boundary. If we don't find the start or end of the
|
433
|
+
# boundary, clear the buffer and return nil.
|
294
434
|
def consume_boundary
|
295
|
-
|
296
|
-
|
297
|
-
|
298
|
-
|
299
|
-
|
300
|
-
return if @sbuf.eos?
|
435
|
+
if read_buffer = @sbuf.scan_until(@body_regex)
|
436
|
+
read_buffer.end_with?(EOL) ? :BOUNDARY : :END_BOUNDARY
|
437
|
+
else
|
438
|
+
@sbuf.terminate
|
439
|
+
nil
|
301
440
|
end
|
302
441
|
end
|
303
442
|
|
304
|
-
def
|
305
|
-
filename = nil
|
306
|
-
case head
|
307
|
-
when RFC2183
|
308
|
-
params = Hash[*head.scan(DISPPARM).flat_map(&:compact)]
|
309
|
-
|
310
|
-
if filename = params['filename']
|
311
|
-
filename = $1 if filename =~ /^"(.*)"$/
|
312
|
-
elsif filename = params['filename*']
|
313
|
-
encoding, _, filename = filename.split("'", 3)
|
314
|
-
end
|
315
|
-
when BROKEN
|
316
|
-
filename = $1
|
317
|
-
filename = $1 if filename =~ /^"(.*)"$/
|
318
|
-
end
|
319
|
-
|
320
|
-
return unless filename
|
321
|
-
|
443
|
+
def normalize_filename(filename)
|
322
444
|
if filename.scan(/%.?.?/).all? { |s| /%[0-9a-fA-F]{2}/.match?(s) }
|
323
445
|
filename = Utils.unescape_path(filename)
|
324
446
|
end
|
325
447
|
|
326
448
|
filename.scrub!
|
327
449
|
|
328
|
-
|
329
|
-
filename = filename.gsub(/\\(.)/, '\1')
|
330
|
-
end
|
331
|
-
|
332
|
-
if encoding
|
333
|
-
filename.force_encoding ::Encoding.find(encoding)
|
334
|
-
end
|
335
|
-
|
336
|
-
filename
|
450
|
+
filename.split(/[\/\\]/).last || String.new
|
337
451
|
end
|
338
452
|
|
339
453
|
CHARSET = "charset"
|
454
|
+
deprecate_constant :CHARSET
|
340
455
|
|
341
456
|
def tag_multipart_encoding(filename, content_type, name, body)
|
342
457
|
name = name.to_s
|
@@ -357,7 +472,9 @@ module Rack
|
|
357
472
|
k.strip!
|
358
473
|
v.strip!
|
359
474
|
v = v[1..-2] if v.start_with?('"') && v.end_with?('"')
|
360
|
-
|
475
|
+
if k == "charset"
|
476
|
+
encoding = find_encoding(v)
|
477
|
+
end
|
361
478
|
end
|
362
479
|
end
|
363
480
|
end
|
@@ -366,9 +483,18 @@ module Rack
|
|
366
483
|
body.force_encoding(encoding)
|
367
484
|
end
|
368
485
|
|
486
|
+
# Return the related Encoding object. However, because
|
487
|
+
# enc is submitted by the user, it may be invalid, so
|
488
|
+
# use a binary encoding in that case.
|
489
|
+
def find_encoding(enc)
|
490
|
+
Encoding.find enc
|
491
|
+
rescue ArgumentError
|
492
|
+
Encoding::BINARY
|
493
|
+
end
|
494
|
+
|
369
495
|
def handle_empty_content!(content)
|
370
496
|
if content.nil? || content.empty?
|
371
|
-
raise
|
497
|
+
raise EmptyContentError
|
372
498
|
end
|
373
499
|
end
|
374
500
|
end
|