rack 2.2.15 → 3.0.0.beta1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of rack might be problematic. Click here for more details.

Files changed (85) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +134 -132
  3. data/CONTRIBUTING.md +53 -47
  4. data/MIT-LICENSE +1 -1
  5. data/README.md +287 -0
  6. data/Rakefile +40 -7
  7. data/SPEC.rdoc +166 -125
  8. data/contrib/LICENSE.md +7 -0
  9. data/contrib/logo.webp +0 -0
  10. data/lib/rack/auth/abstract/handler.rb +3 -1
  11. data/lib/rack/auth/abstract/request.rb +3 -1
  12. data/lib/rack/auth/basic.rb +2 -1
  13. data/lib/rack/auth/digest/md5.rb +1 -131
  14. data/lib/rack/auth/digest/nonce.rb +1 -53
  15. data/lib/rack/auth/digest/params.rb +1 -54
  16. data/lib/rack/auth/digest/request.rb +1 -43
  17. data/lib/rack/auth/digest.rb +256 -0
  18. data/lib/rack/body_proxy.rb +3 -1
  19. data/lib/rack/builder.rb +60 -42
  20. data/lib/rack/cascade.rb +2 -0
  21. data/lib/rack/chunked.rb +16 -13
  22. data/lib/rack/common_logger.rb +24 -20
  23. data/lib/rack/conditional_get.rb +18 -15
  24. data/lib/rack/constants.rb +62 -0
  25. data/lib/rack/content_length.rb +12 -16
  26. data/lib/rack/content_type.rb +8 -5
  27. data/lib/rack/deflater.rb +40 -26
  28. data/lib/rack/directory.rb +9 -3
  29. data/lib/rack/etag.rb +14 -23
  30. data/lib/rack/events.rb +4 -0
  31. data/lib/rack/file.rb +2 -0
  32. data/lib/rack/files.rb +15 -17
  33. data/lib/rack/head.rb +9 -8
  34. data/lib/rack/headers.rb +154 -0
  35. data/lib/rack/lint.rb +740 -649
  36. data/lib/rack/lock.rb +2 -5
  37. data/lib/rack/logger.rb +2 -0
  38. data/lib/rack/media_type.rb +4 -9
  39. data/lib/rack/method_override.rb +5 -1
  40. data/lib/rack/mime.rb +8 -0
  41. data/lib/rack/mock.rb +1 -300
  42. data/lib/rack/mock_request.rb +166 -0
  43. data/lib/rack/mock_response.rb +124 -0
  44. data/lib/rack/multipart/generator.rb +7 -5
  45. data/lib/rack/multipart/parser.rb +123 -85
  46. data/lib/rack/multipart/uploaded_file.rb +4 -0
  47. data/lib/rack/multipart.rb +20 -40
  48. data/lib/rack/null_logger.rb +9 -0
  49. data/lib/rack/query_parser.rb +78 -89
  50. data/lib/rack/recursive.rb +2 -0
  51. data/lib/rack/reloader.rb +0 -2
  52. data/lib/rack/request.rb +189 -91
  53. data/lib/rack/response.rb +131 -61
  54. data/lib/rack/rewindable_input.rb +24 -5
  55. data/lib/rack/runtime.rb +7 -6
  56. data/lib/rack/sendfile.rb +31 -26
  57. data/lib/rack/show_exceptions.rb +15 -2
  58. data/lib/rack/show_status.rb +17 -7
  59. data/lib/rack/static.rb +9 -10
  60. data/lib/rack/tempfile_reaper.rb +15 -4
  61. data/lib/rack/urlmap.rb +4 -2
  62. data/lib/rack/utils.rb +212 -202
  63. data/lib/rack/version.rb +9 -4
  64. data/lib/rack.rb +5 -76
  65. data/rack.gemspec +6 -6
  66. metadata +22 -31
  67. data/README.rdoc +0 -347
  68. data/bin/rackup +0 -5
  69. data/contrib/rack.png +0 -0
  70. data/contrib/rack.svg +0 -150
  71. data/contrib/rack_logo.svg +0 -164
  72. data/lib/rack/core_ext/regexp.rb +0 -14
  73. data/lib/rack/handler/cgi.rb +0 -59
  74. data/lib/rack/handler/fastcgi.rb +0 -100
  75. data/lib/rack/handler/lsws.rb +0 -61
  76. data/lib/rack/handler/scgi.rb +0 -71
  77. data/lib/rack/handler/thin.rb +0 -36
  78. data/lib/rack/handler/webrick.rb +0 -129
  79. data/lib/rack/handler.rb +0 -104
  80. data/lib/rack/lobster.rb +0 -70
  81. data/lib/rack/server.rb +0 -466
  82. data/lib/rack/session/abstract/id.rb +0 -523
  83. data/lib/rack/session/cookie.rb +0 -203
  84. data/lib/rack/session/memcache.rb +0 -10
  85. data/lib/rack/session/pool.rb +0 -90
data/lib/rack/lock.rb CHANGED
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'thread'
3
+ require_relative 'body_proxy'
4
4
 
5
5
  module Rack
6
6
  # Rack::Lock locks every request inside a mutex, so that every request
@@ -12,10 +12,8 @@ module Rack
12
12
 
13
13
  def call(env)
14
14
  @mutex.lock
15
- @env = env
16
- @old_rack_multithread = env[RACK_MULTITHREAD]
17
15
  begin
18
- response = @app.call(env.merge!(RACK_MULTITHREAD => false))
16
+ response = @app.call(env)
19
17
  returned = response << BodyProxy.new(response.pop) { unlock }
20
18
  ensure
21
19
  unlock unless returned
@@ -26,7 +24,6 @@ module Rack
26
24
 
27
25
  def unlock
28
26
  @mutex.unlock
29
- @env[RACK_MULTITHREAD] = @old_rack_multithread
30
27
  end
31
28
  end
32
29
  end
data/lib/rack/logger.rb CHANGED
@@ -2,6 +2,8 @@
2
2
 
3
3
  require 'logger'
4
4
 
5
+ require_relative 'constants'
6
+
5
7
  module Rack
6
8
  # Sets up rack.logger to write to rack.errors stream
7
9
  class Logger
@@ -4,7 +4,7 @@ module Rack
4
4
  # Rack::MediaType parse media type and parameters out of content_type string
5
5
 
6
6
  class MediaType
7
- SPLIT_PATTERN = /[;,]/
7
+ SPLIT_PATTERN = %r{\s*[;,]\s*}
8
8
 
9
9
  class << self
10
10
  # The media type (type/subtype) portion of the CONTENT_TYPE header
@@ -15,11 +15,7 @@ module Rack
15
15
  # http://www.w3.org/Protocols/rfc2616/rfc2616-sec3.html#sec3.7
16
16
  def type(content_type)
17
17
  return nil unless content_type
18
- if type = content_type.split(SPLIT_PATTERN, 2).first
19
- type.rstrip!
20
- type.downcase!
21
- type
22
- end
18
+ content_type.split(SPLIT_PATTERN, 2).first.tap(&:downcase!)
23
19
  end
24
20
 
25
21
  # The media type parameters provided in CONTENT_TYPE as a Hash, or
@@ -31,10 +27,9 @@ module Rack
31
27
  return {} if content_type.nil?
32
28
 
33
29
  content_type.split(SPLIT_PATTERN)[1..-1].each_with_object({}) do |s, hsh|
34
- s.strip!
35
30
  k, v = s.split('=', 2)
36
- k.downcase!
37
- hsh[k] = strip_doublequotes(v)
31
+
32
+ hsh[k.tap(&:downcase!)] = strip_doublequotes(v)
38
33
  end
39
34
  end
40
35
 
@@ -1,5 +1,9 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require_relative 'constants'
4
+ require_relative 'request'
5
+ require_relative 'utils'
6
+
3
7
  module Rack
4
8
  class MethodOverride
5
9
  HTTP_METHODS = %w[GET HEAD PUT POST DELETE OPTIONS PATCH LINK UNLINK]
@@ -43,7 +47,7 @@ module Rack
43
47
 
44
48
  def method_override_param(req)
45
49
  req.POST[METHOD_OVERRIDE_PARAM_KEY]
46
- rescue Utils::InvalidParameterError, Utils::ParameterTypeError, QueryParser::ParamsTooDeepError
50
+ rescue Utils::InvalidParameterError, Utils::ParameterTypeError
47
51
  req.get_header(RACK_ERRORS).puts "Invalid or incomplete POST params"
48
52
  rescue EOFError
49
53
  req.get_header(RACK_ERRORS).puts "Bad request content body"
data/lib/rack/mime.rb CHANGED
@@ -63,6 +63,7 @@ module Rack
63
63
  ".aif" => "audio/x-aiff",
64
64
  ".aiff" => "audio/x-aiff",
65
65
  ".ami" => "application/vnd.amiga.ami",
66
+ ".apng" => "image/apng",
66
67
  ".appcache" => "text/cache-manifest",
67
68
  ".apr" => "application/vnd.lotus-approach",
68
69
  ".asc" => "application/pgp-signature",
@@ -77,6 +78,7 @@ module Rack
77
78
  ".atx" => "application/vnd.antix.game-component",
78
79
  ".au" => "audio/basic",
79
80
  ".avi" => "video/x-msvideo",
81
+ ".avif" => "image/avif",
80
82
  ".bat" => "application/x-msdownload",
81
83
  ".bcpio" => "application/x-bcpio",
82
84
  ".bdm" => "application/vnd.syncml.dm+wbxml",
@@ -197,6 +199,7 @@ module Rack
197
199
  ".fe_launch" => "application/vnd.denovo.fcselayout-link",
198
200
  ".fg5" => "application/vnd.fujitsu.oasysgp",
199
201
  ".fli" => "video/x-fli",
202
+ ".flif" => "image/flif",
200
203
  ".flo" => "application/vnd.micrografx.flo",
201
204
  ".flv" => "video/x-flv",
202
205
  ".flw" => "application/vnd.kde.kivio",
@@ -237,6 +240,10 @@ module Rack
237
240
  ".h264" => "video/h264",
238
241
  ".hbci" => "application/vnd.hbci",
239
242
  ".hdf" => "application/x-hdf",
243
+ ".heic" => "image/heic",
244
+ ".heics" => "image/heic-sequence",
245
+ ".heif" => "image/heif",
246
+ ".heifs" => "image/heif-sequence",
240
247
  ".hh" => "text/x-c",
241
248
  ".hlp" => "application/winhlp",
242
249
  ".hpgl" => "application/vnd.hp-hpgl",
@@ -617,6 +624,7 @@ module Rack
617
624
  ".wbs" => "application/vnd.criticaltools.wbs+xml",
618
625
  ".wbxml" => "application/vnd.wap.wbxml",
619
626
  ".webm" => "video/webm",
627
+ ".webp" => "image/webp",
620
628
  ".wm" => "video/x-ms-wm",
621
629
  ".wma" => "audio/x-ms-wma",
622
630
  ".wmd" => "application/x-ms-wmd",
data/lib/rack/mock.rb CHANGED
@@ -1,302 +1,3 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'uri'
4
- require 'stringio'
5
- require_relative '../rack'
6
-
7
- module Rack
8
- # Rack::MockRequest helps testing your Rack application without
9
- # actually using HTTP.
10
- #
11
- # After performing a request on a URL with get/post/put/patch/delete, it
12
- # returns a MockResponse with useful helper methods for effective
13
- # testing.
14
- #
15
- # You can pass a hash with additional configuration to the
16
- # get/post/put/patch/delete.
17
- # <tt>:input</tt>:: A String or IO-like to be used as rack.input.
18
- # <tt>:fatal</tt>:: Raise a FatalWarning if the app writes to rack.errors.
19
- # <tt>:lint</tt>:: If true, wrap the application in a Rack::Lint.
20
-
21
- class MockRequest
22
- class FatalWarning < RuntimeError
23
- end
24
-
25
- class FatalWarner
26
- def puts(warning)
27
- raise FatalWarning, warning
28
- end
29
-
30
- def write(warning)
31
- raise FatalWarning, warning
32
- end
33
-
34
- def flush
35
- end
36
-
37
- def string
38
- ""
39
- end
40
- end
41
-
42
- DEFAULT_ENV = {
43
- RACK_VERSION => Rack::VERSION,
44
- RACK_INPUT => StringIO.new,
45
- RACK_ERRORS => StringIO.new,
46
- RACK_MULTITHREAD => true,
47
- RACK_MULTIPROCESS => true,
48
- RACK_RUNONCE => false,
49
- }.freeze
50
-
51
- def initialize(app)
52
- @app = app
53
- end
54
-
55
- # Make a GET request and return a MockResponse. See #request.
56
- def get(uri, opts = {}) request(GET, uri, opts) end
57
- # Make a POST request and return a MockResponse. See #request.
58
- def post(uri, opts = {}) request(POST, uri, opts) end
59
- # Make a PUT request and return a MockResponse. See #request.
60
- def put(uri, opts = {}) request(PUT, uri, opts) end
61
- # Make a PATCH request and return a MockResponse. See #request.
62
- def patch(uri, opts = {}) request(PATCH, uri, opts) end
63
- # Make a DELETE request and return a MockResponse. See #request.
64
- def delete(uri, opts = {}) request(DELETE, uri, opts) end
65
- # Make a HEAD request and return a MockResponse. See #request.
66
- def head(uri, opts = {}) request(HEAD, uri, opts) end
67
- # Make an OPTIONS request and return a MockResponse. See #request.
68
- def options(uri, opts = {}) request(OPTIONS, uri, opts) end
69
-
70
- # Make a request using the given request method for the given
71
- # uri to the rack application and return a MockResponse.
72
- # Options given are passed to MockRequest.env_for.
73
- def request(method = GET, uri = "", opts = {})
74
- env = self.class.env_for(uri, opts.merge(method: method))
75
-
76
- if opts[:lint]
77
- app = Rack::Lint.new(@app)
78
- else
79
- app = @app
80
- end
81
-
82
- errors = env[RACK_ERRORS]
83
- status, headers, body = app.call(env)
84
- MockResponse.new(status, headers, body, errors)
85
- ensure
86
- body.close if body.respond_to?(:close)
87
- end
88
-
89
- # For historical reasons, we're pinning to RFC 2396.
90
- # URI::Parser = URI::RFC2396_Parser
91
- def self.parse_uri_rfc2396(uri)
92
- @parser ||= URI::Parser.new
93
- @parser.parse(uri)
94
- end
95
-
96
- # Return the Rack environment used for a request to +uri+.
97
- # All options that are strings are added to the returned environment.
98
- # Options:
99
- # :fatal :: Whether to raise an exception if request outputs to rack.errors
100
- # :input :: The rack.input to set
101
- # :method :: The HTTP request method to use
102
- # :params :: The params to use
103
- # :script_name :: The SCRIPT_NAME to set
104
- def self.env_for(uri = "", opts = {})
105
- uri = parse_uri_rfc2396(uri)
106
- uri.path = "/#{uri.path}" unless uri.path[0] == ?/
107
-
108
- env = DEFAULT_ENV.dup
109
-
110
- env[REQUEST_METHOD] = (opts[:method] ? opts[:method].to_s.upcase : GET).b
111
- env[SERVER_NAME] = (uri.host || "example.org").b
112
- env[SERVER_PORT] = (uri.port ? uri.port.to_s : "80").b
113
- env[QUERY_STRING] = (uri.query.to_s).b
114
- env[PATH_INFO] = ((!uri.path || uri.path.empty?) ? "/" : uri.path).b
115
- env[RACK_URL_SCHEME] = (uri.scheme || "http").b
116
- env[HTTPS] = (env[RACK_URL_SCHEME] == "https" ? "on" : "off").b
117
-
118
- env[SCRIPT_NAME] = opts[:script_name] || ""
119
-
120
- if opts[:fatal]
121
- env[RACK_ERRORS] = FatalWarner.new
122
- else
123
- env[RACK_ERRORS] = StringIO.new
124
- end
125
-
126
- if params = opts[:params]
127
- if env[REQUEST_METHOD] == GET
128
- params = Utils.parse_nested_query(params) if params.is_a?(String)
129
- params.update(Utils.parse_nested_query(env[QUERY_STRING]))
130
- env[QUERY_STRING] = Utils.build_nested_query(params)
131
- elsif !opts.has_key?(:input)
132
- opts["CONTENT_TYPE"] = "application/x-www-form-urlencoded"
133
- if params.is_a?(Hash)
134
- if data = Rack::Multipart.build_multipart(params)
135
- opts[:input] = data
136
- opts["CONTENT_LENGTH"] ||= data.length.to_s
137
- opts["CONTENT_TYPE"] = "multipart/form-data; boundary=#{Rack::Multipart::MULTIPART_BOUNDARY}"
138
- else
139
- opts[:input] = Utils.build_nested_query(params)
140
- end
141
- else
142
- opts[:input] = params
143
- end
144
- end
145
- end
146
-
147
- empty_str = String.new
148
- opts[:input] ||= empty_str
149
- if String === opts[:input]
150
- rack_input = StringIO.new(opts[:input])
151
- else
152
- rack_input = opts[:input]
153
- end
154
-
155
- rack_input.set_encoding(Encoding::BINARY)
156
- env[RACK_INPUT] = rack_input
157
-
158
- env["CONTENT_LENGTH"] ||= env[RACK_INPUT].size.to_s if env[RACK_INPUT].respond_to?(:size)
159
-
160
- opts.each { |field, value|
161
- env[field] = value if String === field
162
- }
163
-
164
- env
165
- end
166
- end
167
-
168
- # Rack::MockResponse provides useful helpers for testing your apps.
169
- # Usually, you don't create the MockResponse on your own, but use
170
- # MockRequest.
171
-
172
- class MockResponse < Rack::Response
173
- begin
174
- # Recent versions of the CGI gem may not provide `CGI::Cookie`.
175
- require 'cgi/cookie'
176
- Cookie = CGI::Cookie
177
- rescue LoadError
178
- class Cookie
179
- attr_reader :name, :value, :path, :domain, :expires, :secure
180
-
181
- def initialize(args)
182
- @name = args["name"]
183
- @value = args["value"]
184
- @path = args["path"]
185
- @domain = args["domain"]
186
- @expires = args["expires"]
187
- @secure = args["secure"]
188
- end
189
-
190
- def method_missing(method_name, *args, &block)
191
- @value.send(method_name, *args, &block)
192
- end
193
- # :nocov:
194
- ruby2_keywords(:method_missing) if respond_to?(:ruby2_keywords, true)
195
- # :nocov:
196
-
197
- def respond_to_missing?(method_name, include_all = false)
198
- @value.respond_to?(method_name, include_all) || super
199
- end
200
- end
201
- end
202
-
203
- class << self
204
- alias [] new
205
- end
206
-
207
- # Headers
208
- attr_reader :original_headers, :cookies
209
-
210
- # Errors
211
- attr_accessor :errors
212
-
213
- def initialize(status, headers, body, errors = StringIO.new(""))
214
- @original_headers = headers
215
- @errors = errors.string if errors.respond_to?(:string)
216
- @cookies = parse_cookies_from_header
217
-
218
- super(body, status, headers)
219
-
220
- buffered_body!
221
- end
222
-
223
- def =~(other)
224
- body =~ other
225
- end
226
-
227
- def match(other)
228
- body.match other
229
- end
230
-
231
- def body
232
- # FIXME: apparently users of MockResponse expect the return value of
233
- # MockResponse#body to be a string. However, the real response object
234
- # returns the body as a list.
235
- #
236
- # See spec_showstatus.rb:
237
- #
238
- # should "not replace existing messages" do
239
- # ...
240
- # res.body.should == "foo!"
241
- # end
242
- buffer = String.new
243
-
244
- super.each do |chunk|
245
- buffer << chunk
246
- end
247
-
248
- return buffer
249
- end
250
-
251
- def empty?
252
- [201, 204, 304].include? status
253
- end
254
-
255
- def cookie(name)
256
- cookies.fetch(name, nil)
257
- end
258
-
259
- private
260
-
261
- def parse_cookies_from_header
262
- cookies = Hash.new
263
- if original_headers.has_key? 'Set-Cookie'
264
- set_cookie_header = original_headers.fetch('Set-Cookie')
265
- set_cookie_header.split("\n").each do |cookie|
266
- cookie_name, cookie_filling = cookie.split('=', 2)
267
- cookie_attributes = identify_cookie_attributes cookie_filling
268
- parsed_cookie = CGI::Cookie.new(
269
- 'name' => cookie_name.strip,
270
- 'value' => cookie_attributes.fetch('value'),
271
- 'path' => cookie_attributes.fetch('path', nil),
272
- 'domain' => cookie_attributes.fetch('domain', nil),
273
- 'expires' => cookie_attributes.fetch('expires', nil),
274
- 'secure' => cookie_attributes.fetch('secure', false)
275
- )
276
- cookies.store(cookie_name, parsed_cookie)
277
- end
278
- end
279
- cookies
280
- end
281
-
282
- def identify_cookie_attributes(cookie_filling)
283
- cookie_bits = cookie_filling.split(';')
284
- cookie_attributes = Hash.new
285
- cookie_attributes.store('value', cookie_bits[0].strip)
286
- cookie_bits.each do |bit|
287
- if bit.include? '='
288
- cookie_attribute, attribute_value = bit.split('=')
289
- cookie_attributes.store(cookie_attribute.strip, attribute_value.strip)
290
- if cookie_attribute.include? 'max-age'
291
- cookie_attributes.store('expires', Time.now + attribute_value.strip.to_i)
292
- end
293
- end
294
- if bit.include? 'secure'
295
- cookie_attributes.store('secure', true)
296
- end
297
- end
298
- cookie_attributes
299
- end
300
-
301
- end
302
- end
3
+ require_relative 'mock_request'
@@ -0,0 +1,166 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'uri'
4
+ require 'stringio'
5
+
6
+ require_relative 'constants'
7
+ require_relative 'mock_response'
8
+
9
+ module Rack
10
+ # Rack::MockRequest helps testing your Rack application without
11
+ # actually using HTTP.
12
+ #
13
+ # After performing a request on a URL with get/post/put/patch/delete, it
14
+ # returns a MockResponse with useful helper methods for effective
15
+ # testing.
16
+ #
17
+ # You can pass a hash with additional configuration to the
18
+ # get/post/put/patch/delete.
19
+ # <tt>:input</tt>:: A String or IO-like to be used as rack.input.
20
+ # <tt>:fatal</tt>:: Raise a FatalWarning if the app writes to rack.errors.
21
+ # <tt>:lint</tt>:: If true, wrap the application in a Rack::Lint.
22
+
23
+ class MockRequest
24
+ class FatalWarning < RuntimeError
25
+ end
26
+
27
+ class FatalWarner
28
+ def puts(warning)
29
+ raise FatalWarning, warning
30
+ end
31
+
32
+ def write(warning)
33
+ raise FatalWarning, warning
34
+ end
35
+
36
+ def flush
37
+ end
38
+
39
+ def string
40
+ ""
41
+ end
42
+ end
43
+
44
+ DEFAULT_ENV = {
45
+ RACK_INPUT => StringIO.new,
46
+ RACK_ERRORS => StringIO.new,
47
+ }.freeze
48
+
49
+ def initialize(app)
50
+ @app = app
51
+ end
52
+
53
+ # Make a GET request and return a MockResponse. See #request.
54
+ def get(uri, opts = {}) request(GET, uri, opts) end
55
+ # Make a POST request and return a MockResponse. See #request.
56
+ def post(uri, opts = {}) request(POST, uri, opts) end
57
+ # Make a PUT request and return a MockResponse. See #request.
58
+ def put(uri, opts = {}) request(PUT, uri, opts) end
59
+ # Make a PATCH request and return a MockResponse. See #request.
60
+ def patch(uri, opts = {}) request(PATCH, uri, opts) end
61
+ # Make a DELETE request and return a MockResponse. See #request.
62
+ def delete(uri, opts = {}) request(DELETE, uri, opts) end
63
+ # Make a HEAD request and return a MockResponse. See #request.
64
+ def head(uri, opts = {}) request(HEAD, uri, opts) end
65
+ # Make an OPTIONS request and return a MockResponse. See #request.
66
+ def options(uri, opts = {}) request(OPTIONS, uri, opts) end
67
+
68
+ # Make a request using the given request method for the given
69
+ # uri to the rack application and return a MockResponse.
70
+ # Options given are passed to MockRequest.env_for.
71
+ def request(method = GET, uri = "", opts = {})
72
+ env = self.class.env_for(uri, opts.merge(method: method))
73
+
74
+ if opts[:lint]
75
+ app = Rack::Lint.new(@app)
76
+ else
77
+ app = @app
78
+ end
79
+
80
+ errors = env[RACK_ERRORS]
81
+ status, headers, body = app.call(env)
82
+ MockResponse.new(status, headers, body, errors)
83
+ ensure
84
+ body.close if body.respond_to?(:close)
85
+ end
86
+
87
+ # For historical reasons, we're pinning to RFC 2396.
88
+ # URI::Parser = URI::RFC2396_Parser
89
+ def self.parse_uri_rfc2396(uri)
90
+ @parser ||= URI::Parser.new
91
+ @parser.parse(uri)
92
+ end
93
+
94
+ # Return the Rack environment used for a request to +uri+.
95
+ # All options that are strings are added to the returned environment.
96
+ # Options:
97
+ # :fatal :: Whether to raise an exception if request outputs to rack.errors
98
+ # :input :: The rack.input to set
99
+ # :http_version :: The SERVER_PROTOCOL to set
100
+ # :method :: The HTTP request method to use
101
+ # :params :: The params to use
102
+ # :script_name :: The SCRIPT_NAME to set
103
+ def self.env_for(uri = "", opts = {})
104
+ uri = parse_uri_rfc2396(uri)
105
+ uri.path = "/#{uri.path}" unless uri.path[0] == ?/
106
+
107
+ env = DEFAULT_ENV.dup
108
+
109
+ env[REQUEST_METHOD] = (opts[:method] ? opts[:method].to_s.upcase : GET).b
110
+ env[SERVER_NAME] = (uri.host || "example.org").b
111
+ env[SERVER_PORT] = (uri.port ? uri.port.to_s : "80").b
112
+ env[SERVER_PROTOCOL] = opts[:http_version] || 'HTTP/1.1'
113
+ env[QUERY_STRING] = (uri.query.to_s).b
114
+ env[PATH_INFO] = (uri.path).b
115
+ env[RACK_URL_SCHEME] = (uri.scheme || "http").b
116
+ env[HTTPS] = (env[RACK_URL_SCHEME] == "https" ? "on" : "off").b
117
+
118
+ env[SCRIPT_NAME] = opts[:script_name] || ""
119
+
120
+ if opts[:fatal]
121
+ env[RACK_ERRORS] = FatalWarner.new
122
+ else
123
+ env[RACK_ERRORS] = StringIO.new
124
+ end
125
+
126
+ if params = opts[:params]
127
+ if env[REQUEST_METHOD] == GET
128
+ params = Utils.parse_nested_query(params) if params.is_a?(String)
129
+ params.update(Utils.parse_nested_query(env[QUERY_STRING]))
130
+ env[QUERY_STRING] = Utils.build_nested_query(params)
131
+ elsif !opts.has_key?(:input)
132
+ opts["CONTENT_TYPE"] = "application/x-www-form-urlencoded"
133
+ if params.is_a?(Hash)
134
+ if data = Rack::Multipart.build_multipart(params)
135
+ opts[:input] = data
136
+ opts["CONTENT_LENGTH"] ||= data.length.to_s
137
+ opts["CONTENT_TYPE"] = "multipart/form-data; boundary=#{Rack::Multipart::MULTIPART_BOUNDARY}"
138
+ else
139
+ opts[:input] = Utils.build_nested_query(params)
140
+ end
141
+ else
142
+ opts[:input] = params
143
+ end
144
+ end
145
+ end
146
+
147
+ opts[:input] ||= String.new
148
+ if String === opts[:input]
149
+ rack_input = StringIO.new(opts[:input])
150
+ else
151
+ rack_input = opts[:input]
152
+ end
153
+
154
+ rack_input.set_encoding(Encoding::BINARY)
155
+ env[RACK_INPUT] = rack_input
156
+
157
+ env["CONTENT_LENGTH"] ||= env[RACK_INPUT].size.to_s if env[RACK_INPUT].respond_to?(:size)
158
+
159
+ opts.each { |field, value|
160
+ env[field] = value if String === field
161
+ }
162
+
163
+ env
164
+ end
165
+ end
166
+ end