rack 2.0.6 → 2.1.0

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


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

Files changed (187) hide show
  1. checksums.yaml +4 -4
  2. data/{HISTORY.md → CHANGELOG.md} +220 -155
  3. data/{COPYING → MIT-LICENSE} +4 -2
  4. data/README.rdoc +77 -117
  5. data/Rakefile +25 -18
  6. data/SPEC +3 -4
  7. data/bin/rackup +1 -0
  8. data/example/lobster.ru +2 -0
  9. data/example/protectedlobster.rb +3 -1
  10. data/example/protectedlobster.ru +2 -0
  11. data/lib/rack/auth/abstract/handler.rb +3 -1
  12. data/lib/rack/auth/abstract/request.rb +2 -0
  13. data/lib/rack/auth/basic.rb +4 -1
  14. data/lib/rack/auth/digest/md5.rb +9 -7
  15. data/lib/rack/auth/digest/nonce.rb +6 -3
  16. data/lib/rack/auth/digest/params.rb +4 -2
  17. data/lib/rack/auth/digest/request.rb +2 -0
  18. data/lib/rack/body_proxy.rb +3 -6
  19. data/lib/rack/builder.rb +38 -15
  20. data/lib/rack/cascade.rb +6 -5
  21. data/lib/rack/chunked.rb +29 -6
  22. data/lib/rack/common_logger.rb +9 -8
  23. data/lib/rack/conditional_get.rb +3 -1
  24. data/lib/rack/config.rb +2 -0
  25. data/lib/rack/content_length.rb +3 -1
  26. data/lib/rack/content_type.rb +3 -1
  27. data/lib/rack/core_ext/regexp.rb +14 -0
  28. data/lib/rack/deflater.rb +28 -17
  29. data/lib/rack/directory.rb +17 -14
  30. data/lib/rack/etag.rb +3 -1
  31. data/lib/rack/events.rb +5 -3
  32. data/lib/rack/file.rb +5 -173
  33. data/lib/rack/files.rb +178 -0
  34. data/lib/rack/handler/cgi.rb +3 -1
  35. data/lib/rack/handler/fastcgi.rb +4 -2
  36. data/lib/rack/handler/lsws.rb +3 -1
  37. data/lib/rack/handler/scgi.rb +9 -6
  38. data/lib/rack/handler/thin.rb +3 -1
  39. data/lib/rack/handler/webrick.rb +4 -2
  40. data/lib/rack/handler.rb +7 -2
  41. data/lib/rack/head.rb +2 -0
  42. data/lib/rack/lint.rb +14 -11
  43. data/lib/rack/lobster.rb +7 -5
  44. data/lib/rack/lock.rb +2 -0
  45. data/lib/rack/logger.rb +2 -0
  46. data/lib/rack/media_type.rb +10 -5
  47. data/lib/rack/method_override.rb +4 -2
  48. data/lib/rack/mime.rb +9 -1
  49. data/lib/rack/mock.rb +74 -15
  50. data/lib/rack/multipart/generator.rb +6 -7
  51. data/lib/rack/multipart/parser.rb +55 -52
  52. data/lib/rack/multipart/uploaded_file.rb +2 -0
  53. data/lib/rack/multipart.rb +5 -3
  54. data/lib/rack/null_logger.rb +2 -0
  55. data/lib/rack/query_parser.rb +51 -25
  56. data/lib/rack/recursive.rb +7 -5
  57. data/lib/rack/reloader.rb +10 -4
  58. data/lib/rack/request.rb +80 -27
  59. data/lib/rack/response.rb +71 -31
  60. data/lib/rack/rewindable_input.rb +4 -2
  61. data/lib/rack/runtime.rb +4 -2
  62. data/lib/rack/sendfile.rb +15 -8
  63. data/lib/rack/server.rb +88 -16
  64. data/lib/rack/session/abstract/id.rb +104 -21
  65. data/lib/rack/session/cookie.rb +21 -11
  66. data/lib/rack/session/memcache.rb +4 -87
  67. data/lib/rack/session/pool.rb +17 -8
  68. data/lib/rack/show_exceptions.rb +15 -9
  69. data/lib/rack/show_status.rb +4 -2
  70. data/lib/rack/static.rb +15 -10
  71. data/lib/rack/tempfile_reaper.rb +2 -0
  72. data/lib/rack/urlmap.rb +11 -2
  73. data/lib/rack/utils.rb +55 -70
  74. data/lib/rack.rb +63 -60
  75. data/rack.gemspec +17 -7
  76. metadata +30 -171
  77. data/test/builder/an_underscore_app.rb +0 -5
  78. data/test/builder/anything.rb +0 -5
  79. data/test/builder/comment.ru +0 -4
  80. data/test/builder/end.ru +0 -5
  81. data/test/builder/line.ru +0 -1
  82. data/test/builder/options.ru +0 -2
  83. data/test/cgi/assets/folder/test.js +0 -1
  84. data/test/cgi/assets/fonts/font.eot +0 -1
  85. data/test/cgi/assets/images/image.png +0 -1
  86. data/test/cgi/assets/index.html +0 -1
  87. data/test/cgi/assets/javascripts/app.js +0 -1
  88. data/test/cgi/assets/stylesheets/app.css +0 -1
  89. data/test/cgi/lighttpd.conf +0 -26
  90. data/test/cgi/rackup_stub.rb +0 -6
  91. data/test/cgi/sample_rackup.ru +0 -5
  92. data/test/cgi/test +0 -9
  93. data/test/cgi/test+directory/test+file +0 -1
  94. data/test/cgi/test.fcgi +0 -9
  95. data/test/cgi/test.gz +0 -0
  96. data/test/cgi/test.ru +0 -5
  97. data/test/gemloader.rb +0 -10
  98. data/test/helper.rb +0 -34
  99. data/test/multipart/bad_robots +0 -259
  100. data/test/multipart/binary +0 -0
  101. data/test/multipart/content_type_and_no_filename +0 -6
  102. data/test/multipart/empty +0 -10
  103. data/test/multipart/fail_16384_nofile +0 -814
  104. data/test/multipart/file1.txt +0 -1
  105. data/test/multipart/filename_and_modification_param +0 -7
  106. data/test/multipart/filename_and_no_name +0 -6
  107. data/test/multipart/filename_with_encoded_words +0 -7
  108. data/test/multipart/filename_with_escaped_quotes +0 -6
  109. data/test/multipart/filename_with_escaped_quotes_and_modification_param +0 -7
  110. data/test/multipart/filename_with_null_byte +0 -7
  111. data/test/multipart/filename_with_percent_escaped_quotes +0 -6
  112. data/test/multipart/filename_with_single_quote +0 -7
  113. data/test/multipart/filename_with_unescaped_percentages +0 -6
  114. data/test/multipart/filename_with_unescaped_percentages2 +0 -6
  115. data/test/multipart/filename_with_unescaped_percentages3 +0 -6
  116. data/test/multipart/filename_with_unescaped_quotes +0 -6
  117. data/test/multipart/ie +0 -6
  118. data/test/multipart/invalid_character +0 -6
  119. data/test/multipart/mixed_files +0 -21
  120. data/test/multipart/nested +0 -10
  121. data/test/multipart/none +0 -9
  122. data/test/multipart/quoted +0 -15
  123. data/test/multipart/rack-logo.png +0 -0
  124. data/test/multipart/semicolon +0 -6
  125. data/test/multipart/text +0 -15
  126. data/test/multipart/three_files_three_fields +0 -31
  127. data/test/multipart/unity3d_wwwform +0 -11
  128. data/test/multipart/webkit +0 -32
  129. data/test/rackup/config.ru +0 -31
  130. data/test/registering_handler/rack/handler/registering_myself.rb +0 -8
  131. data/test/spec_auth_basic.rb +0 -89
  132. data/test/spec_auth_digest.rb +0 -260
  133. data/test/spec_body_proxy.rb +0 -85
  134. data/test/spec_builder.rb +0 -233
  135. data/test/spec_cascade.rb +0 -63
  136. data/test/spec_cgi.rb +0 -84
  137. data/test/spec_chunked.rb +0 -103
  138. data/test/spec_common_logger.rb +0 -95
  139. data/test/spec_conditional_get.rb +0 -103
  140. data/test/spec_config.rb +0 -23
  141. data/test/spec_content_length.rb +0 -86
  142. data/test/spec_content_type.rb +0 -46
  143. data/test/spec_deflater.rb +0 -375
  144. data/test/spec_directory.rb +0 -148
  145. data/test/spec_etag.rb +0 -108
  146. data/test/spec_events.rb +0 -133
  147. data/test/spec_fastcgi.rb +0 -85
  148. data/test/spec_file.rb +0 -264
  149. data/test/spec_handler.rb +0 -57
  150. data/test/spec_head.rb +0 -46
  151. data/test/spec_lint.rb +0 -515
  152. data/test/spec_lobster.rb +0 -59
  153. data/test/spec_lock.rb +0 -204
  154. data/test/spec_logger.rb +0 -24
  155. data/test/spec_media_type.rb +0 -42
  156. data/test/spec_method_override.rb +0 -110
  157. data/test/spec_mime.rb +0 -51
  158. data/test/spec_mock.rb +0 -359
  159. data/test/spec_multipart.rb +0 -722
  160. data/test/spec_null_logger.rb +0 -21
  161. data/test/spec_recursive.rb +0 -75
  162. data/test/spec_request.rb +0 -1398
  163. data/test/spec_response.rb +0 -510
  164. data/test/spec_rewindable_input.rb +0 -128
  165. data/test/spec_runtime.rb +0 -50
  166. data/test/spec_sendfile.rb +0 -125
  167. data/test/spec_server.rb +0 -193
  168. data/test/spec_session_abstract_id.rb +0 -31
  169. data/test/spec_session_abstract_session_hash.rb +0 -45
  170. data/test/spec_session_cookie.rb +0 -442
  171. data/test/spec_session_memcache.rb +0 -320
  172. data/test/spec_session_pool.rb +0 -210
  173. data/test/spec_show_exceptions.rb +0 -93
  174. data/test/spec_show_status.rb +0 -104
  175. data/test/spec_static.rb +0 -184
  176. data/test/spec_tempfile_reaper.rb +0 -64
  177. data/test/spec_thin.rb +0 -96
  178. data/test/spec_urlmap.rb +0 -237
  179. data/test/spec_utils.rb +0 -742
  180. data/test/spec_version.rb +0 -11
  181. data/test/spec_webrick.rb +0 -206
  182. data/test/static/another/index.html +0 -1
  183. data/test/static/foo.html +0 -1
  184. data/test/static/index.html +0 -1
  185. data/test/testrequest.rb +0 -78
  186. data/test/unregistered_handler/rack/handler/unregistered.rb +0 -7
  187. data/test/unregistered_handler/rack/handler/unregistered_long_one.rb +0 -7
data/test/spec_lint.rb DELETED
@@ -1,515 +0,0 @@
1
- require 'minitest/autorun'
2
- require 'stringio'
3
- require 'tempfile'
4
- require 'rack/lint'
5
- require 'rack/mock'
6
-
7
- describe Rack::Lint do
8
- def env(*args)
9
- Rack::MockRequest.env_for("/", *args)
10
- end
11
-
12
- it "pass valid request" do
13
- Rack::Lint.new(lambda { |env|
14
- [200, {"Content-type" => "test/plain", "Content-length" => "3"}, ["foo"]]
15
- }).call(env({})).first.must_equal 200
16
- end
17
-
18
- it "notice fatal errors" do
19
- lambda { Rack::Lint.new(nil).call }.must_raise(Rack::Lint::LintError).
20
- message.must_match(/No env given/)
21
- end
22
-
23
- it "notice environment errors" do
24
- lambda { Rack::Lint.new(nil).call 5 }.must_raise(Rack::Lint::LintError).
25
- message.must_match(/not a Hash/)
26
-
27
- lambda {
28
- e = env
29
- e.delete("REQUEST_METHOD")
30
- Rack::Lint.new(nil).call(e)
31
- }.must_raise(Rack::Lint::LintError).
32
- message.must_match(/missing required key REQUEST_METHOD/)
33
-
34
- lambda {
35
- e = env
36
- e.delete("SERVER_NAME")
37
- Rack::Lint.new(nil).call(e)
38
- }.must_raise(Rack::Lint::LintError).
39
- message.must_match(/missing required key SERVER_NAME/)
40
-
41
-
42
- lambda {
43
- Rack::Lint.new(nil).call(env("HTTP_CONTENT_TYPE" => "text/plain"))
44
- }.must_raise(Rack::Lint::LintError).
45
- message.must_match(/contains HTTP_CONTENT_TYPE/)
46
-
47
- lambda {
48
- Rack::Lint.new(nil).call(env("HTTP_CONTENT_LENGTH" => "42"))
49
- }.must_raise(Rack::Lint::LintError).
50
- message.must_match(/contains HTTP_CONTENT_LENGTH/)
51
-
52
- lambda {
53
- Rack::Lint.new(nil).call(env("FOO" => Object.new))
54
- }.must_raise(Rack::Lint::LintError).
55
- message.must_match(/non-string value/)
56
-
57
- lambda {
58
- Rack::Lint.new(nil).call(env("rack.version" => "0.2"))
59
- }.must_raise(Rack::Lint::LintError).
60
- message.must_match(/must be an Array/)
61
-
62
- lambda {
63
- Rack::Lint.new(nil).call(env("rack.url_scheme" => "gopher"))
64
- }.must_raise(Rack::Lint::LintError).
65
- message.must_match(/url_scheme unknown/)
66
-
67
- lambda {
68
- Rack::Lint.new(nil).call(env("rack.session" => []))
69
- }.must_raise(Rack::Lint::LintError).
70
- message.must_equal "session [] must respond to store and []="
71
-
72
- lambda {
73
- Rack::Lint.new(nil).call(env("rack.logger" => []))
74
- }.must_raise(Rack::Lint::LintError).
75
- message.must_equal "logger [] must respond to info"
76
-
77
- lambda {
78
- Rack::Lint.new(nil).call(env("rack.multipart.buffer_size" => 0))
79
- }.must_raise(Rack::Lint::LintError).
80
- message.must_equal "rack.multipart.buffer_size must be an Integer > 0 if specified"
81
-
82
- lambda {
83
- Rack::Lint.new(nil).call(env("rack.multipart.tempfile_factory" => Tempfile))
84
- }.must_raise(Rack::Lint::LintError).
85
- message.must_equal "rack.multipart.tempfile_factory must respond to #call"
86
-
87
- lambda {
88
- Rack::Lint.new(lambda { |env|
89
- env['rack.multipart.tempfile_factory'].call("testfile", "text/plain")
90
- }).call(env("rack.multipart.tempfile_factory" => lambda { |filename, content_type| Object.new }))
91
- }.must_raise(Rack::Lint::LintError).
92
- message.must_equal "rack.multipart.tempfile_factory return value must respond to #<<"
93
-
94
- lambda {
95
- Rack::Lint.new(nil).call(env("REQUEST_METHOD" => "FUCKUP?"))
96
- }.must_raise(Rack::Lint::LintError).
97
- message.must_match(/REQUEST_METHOD/)
98
-
99
- lambda {
100
- Rack::Lint.new(nil).call(env("SCRIPT_NAME" => "howdy"))
101
- }.must_raise(Rack::Lint::LintError).
102
- message.must_match(/must start with/)
103
-
104
- lambda {
105
- Rack::Lint.new(nil).call(env("PATH_INFO" => "../foo"))
106
- }.must_raise(Rack::Lint::LintError).
107
- message.must_match(/must start with/)
108
-
109
- lambda {
110
- Rack::Lint.new(nil).call(env("CONTENT_LENGTH" => "xcii"))
111
- }.must_raise(Rack::Lint::LintError).
112
- message.must_match(/Invalid CONTENT_LENGTH/)
113
-
114
- lambda {
115
- e = env
116
- e.delete("PATH_INFO")
117
- e.delete("SCRIPT_NAME")
118
- Rack::Lint.new(nil).call(e)
119
- }.must_raise(Rack::Lint::LintError).
120
- message.must_match(/One of .* must be set/)
121
-
122
- lambda {
123
- Rack::Lint.new(nil).call(env("SCRIPT_NAME" => "/"))
124
- }.must_raise(Rack::Lint::LintError).
125
- message.must_match(/cannot be .* make it ''/)
126
- end
127
-
128
- it "notice input errors" do
129
- lambda {
130
- Rack::Lint.new(nil).call(env("rack.input" => ""))
131
- }.must_raise(Rack::Lint::LintError).
132
- message.must_match(/does not respond to #gets/)
133
-
134
- lambda {
135
- input = Object.new
136
- def input.binmode?
137
- false
138
- end
139
- Rack::Lint.new(nil).call(env("rack.input" => input))
140
- }.must_raise(Rack::Lint::LintError).
141
- message.must_match(/is not opened in binary mode/)
142
-
143
- lambda {
144
- input = Object.new
145
- def input.external_encoding
146
- result = Object.new
147
- def result.name
148
- "US-ASCII"
149
- end
150
- result
151
- end
152
- Rack::Lint.new(nil).call(env("rack.input" => input))
153
- }.must_raise(Rack::Lint::LintError).
154
- message.must_match(/does not have ASCII-8BIT as its external encoding/)
155
- end
156
-
157
- it "notice error errors" do
158
- lambda {
159
- Rack::Lint.new(nil).call(env("rack.errors" => ""))
160
- }.must_raise(Rack::Lint::LintError).
161
- message.must_match(/does not respond to #puts/)
162
- end
163
-
164
- it "notice status errors" do
165
- lambda {
166
- Rack::Lint.new(lambda { |env|
167
- ["cc", {}, ""]
168
- }).call(env({}))
169
- }.must_raise(Rack::Lint::LintError).
170
- message.must_match(/must be >=100 seen as integer/)
171
-
172
- lambda {
173
- Rack::Lint.new(lambda { |env|
174
- [42, {}, ""]
175
- }).call(env({}))
176
- }.must_raise(Rack::Lint::LintError).
177
- message.must_match(/must be >=100 seen as integer/)
178
- end
179
-
180
- it "notice header errors" do
181
- lambda {
182
- Rack::Lint.new(lambda { |env|
183
- [200, Object.new, []]
184
- }).call(env({}))
185
- }.must_raise(Rack::Lint::LintError).
186
- message.must_equal "headers object should respond to #each, but doesn't (got Object as headers)"
187
-
188
- lambda {
189
- Rack::Lint.new(lambda { |env|
190
- [200, {true=>false}, []]
191
- }).call(env({}))
192
- }.must_raise(Rack::Lint::LintError).
193
- message.must_equal "header key must be a string, was TrueClass"
194
-
195
- lambda {
196
- Rack::Lint.new(lambda { |env|
197
- [200, {"Status" => "404"}, []]
198
- }).call(env({}))
199
- }.must_raise(Rack::Lint::LintError).
200
- message.must_match(/must not contain Status/)
201
-
202
- # From RFC 7230:<F24><F25>
203
- # Most HTTP header field values are defined using common syntax
204
- # components (token, quoted-string, and comment) separated by
205
- # whitespace or specific delimiting characters. Delimiters are chosen
206
- # from the set of US-ASCII visual characters not allowed in a token
207
- # (DQUOTE and "(),/:;<=>?@[\]{}").
208
- #
209
- # token = 1*tchar
210
- #
211
- # tchar = "!" / "#" / "$" / "%" / "&" / "'" / "*"
212
- # / "+" / "-" / "." / "^" / "_" / "`" / "|" / "~"
213
- # / DIGIT / ALPHA
214
- # ; any VCHAR, except delimiters
215
- invalid_headers = 0.upto(31).map(&:chr) + %W<( ) , / : ; < = > ? @ [ \\ ] { } \x7F>
216
- invalid_headers.each do |invalid_header|
217
- lambda {
218
- Rack::Lint.new(lambda { |env|
219
- [200, {invalid_header => "text/plain"}, []]
220
- }).call(env({}))
221
- }.must_raise(Rack::Lint::LintError, "on invalid header: #{invalid_header}").
222
- message.must_equal("invalid header name: #{invalid_header}")
223
- end
224
- valid_headers = 0.upto(127).map(&:chr) - invalid_headers
225
- valid_headers.each do |valid_header|
226
- Rack::Lint.new(lambda { |env|
227
- [200, {valid_header => "text/plain"}, []]
228
- }).call(env({})).first.must_equal 200
229
- end
230
-
231
- lambda {
232
- Rack::Lint.new(lambda { |env|
233
- [200, {"Foo" => Object.new}, []]
234
- }).call(env({}))
235
- }.must_raise(Rack::Lint::LintError).
236
- message.must_equal "a header value must be a String, but the value of 'Foo' is a Object"
237
-
238
- lambda {
239
- Rack::Lint.new(lambda { |env|
240
- [200, {"Foo" => [1, 2, 3]}, []]
241
- }).call(env({}))
242
- }.must_raise(Rack::Lint::LintError).
243
- message.must_equal "a header value must be a String, but the value of 'Foo' is a Array"
244
-
245
-
246
- lambda {
247
- Rack::Lint.new(lambda { |env|
248
- [200, {"Foo-Bar" => "text\000plain"}, []]
249
- }).call(env({}))
250
- }.must_raise(Rack::Lint::LintError).
251
- message.must_match(/invalid header/)
252
-
253
- # line ends (010).must_be :allowed in header values.?
254
- Rack::Lint.new(lambda { |env|
255
- [200, {"Foo-Bar" => "one\ntwo\nthree", "Content-Length" => "0", "Content-Type" => "text/plain" }, []]
256
- }).call(env({})).first.must_equal 200
257
-
258
- # non-Hash header responses.must_be :allowed?
259
- Rack::Lint.new(lambda { |env|
260
- [200, [%w(Content-Type text/plain), %w(Content-Length 0)], []]
261
- }).call(env({})).first.must_equal 200
262
- end
263
-
264
- it "notice content-type errors" do
265
- # lambda {
266
- # Rack::Lint.new(lambda { |env|
267
- # [200, {"Content-length" => "0"}, []]
268
- # }).call(env({}))
269
- # }.must_raise(Rack::Lint::LintError).
270
- # message.must_match(/No Content-Type/)
271
-
272
- [100, 101, 204, 304].each do |status|
273
- lambda {
274
- Rack::Lint.new(lambda { |env|
275
- [status, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
276
- }).call(env({}))
277
- }.must_raise(Rack::Lint::LintError).
278
- message.must_match(/Content-Type header found/)
279
- end
280
- end
281
-
282
- it "notice content-length errors" do
283
- [100, 101, 204, 304].each do |status|
284
- lambda {
285
- Rack::Lint.new(lambda { |env|
286
- [status, {"Content-length" => "0"}, []]
287
- }).call(env({}))
288
- }.must_raise(Rack::Lint::LintError).
289
- message.must_match(/Content-Length header found/)
290
- end
291
-
292
- lambda {
293
- Rack::Lint.new(lambda { |env|
294
- [200, {"Content-type" => "text/plain", "Content-Length" => "1"}, []]
295
- }).call(env({}))[2].each { }
296
- }.must_raise(Rack::Lint::LintError).
297
- message.must_match(/Content-Length header was 1, but should be 0/)
298
- end
299
-
300
- it "notice body errors" do
301
- lambda {
302
- body = Rack::Lint.new(lambda { |env|
303
- [200, {"Content-type" => "text/plain","Content-length" => "3"}, [1,2,3]]
304
- }).call(env({}))[2]
305
- body.each { |part| }
306
- }.must_raise(Rack::Lint::LintError).
307
- message.must_match(/yielded non-string/)
308
- end
309
-
310
- it "notice input handling errors" do
311
- lambda {
312
- Rack::Lint.new(lambda { |env|
313
- env["rack.input"].gets("\r\n")
314
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
315
- }).call(env({}))
316
- }.must_raise(Rack::Lint::LintError).
317
- message.must_match(/gets called with arguments/)
318
-
319
- lambda {
320
- Rack::Lint.new(lambda { |env|
321
- env["rack.input"].read(1, 2, 3)
322
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
323
- }).call(env({}))
324
- }.must_raise(Rack::Lint::LintError).
325
- message.must_match(/read called with too many arguments/)
326
-
327
- lambda {
328
- Rack::Lint.new(lambda { |env|
329
- env["rack.input"].read("foo")
330
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
331
- }).call(env({}))
332
- }.must_raise(Rack::Lint::LintError).
333
- message.must_match(/read called with non-integer and non-nil length/)
334
-
335
- lambda {
336
- Rack::Lint.new(lambda { |env|
337
- env["rack.input"].read(-1)
338
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
339
- }).call(env({}))
340
- }.must_raise(Rack::Lint::LintError).
341
- message.must_match(/read called with a negative length/)
342
-
343
- lambda {
344
- Rack::Lint.new(lambda { |env|
345
- env["rack.input"].read(nil, nil)
346
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
347
- }).call(env({}))
348
- }.must_raise(Rack::Lint::LintError).
349
- message.must_match(/read called with non-String buffer/)
350
-
351
- lambda {
352
- Rack::Lint.new(lambda { |env|
353
- env["rack.input"].read(nil, 1)
354
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
355
- }).call(env({}))
356
- }.must_raise(Rack::Lint::LintError).
357
- message.must_match(/read called with non-String buffer/)
358
-
359
- lambda {
360
- Rack::Lint.new(lambda { |env|
361
- env["rack.input"].rewind(0)
362
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
363
- }).call(env({}))
364
- }.must_raise(Rack::Lint::LintError).
365
- message.must_match(/rewind called with arguments/)
366
-
367
- weirdio = Object.new
368
- class << weirdio
369
- def gets
370
- 42
371
- end
372
-
373
- def read
374
- 23
375
- end
376
-
377
- def each
378
- yield 23
379
- yield 42
380
- end
381
-
382
- def rewind
383
- raise Errno::ESPIPE, "Errno::ESPIPE"
384
- end
385
- end
386
-
387
- eof_weirdio = Object.new
388
- class << eof_weirdio
389
- def gets
390
- nil
391
- end
392
-
393
- def read(*args)
394
- nil
395
- end
396
-
397
- def each
398
- end
399
-
400
- def rewind
401
- end
402
- end
403
-
404
- lambda {
405
- Rack::Lint.new(lambda { |env|
406
- env["rack.input"].gets
407
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
408
- }).call(env("rack.input" => weirdio))
409
- }.must_raise(Rack::Lint::LintError).
410
- message.must_match(/gets didn't return a String/)
411
-
412
- lambda {
413
- Rack::Lint.new(lambda { |env|
414
- env["rack.input"].each { |x| }
415
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
416
- }).call(env("rack.input" => weirdio))
417
- }.must_raise(Rack::Lint::LintError).
418
- message.must_match(/each didn't yield a String/)
419
-
420
- lambda {
421
- Rack::Lint.new(lambda { |env|
422
- env["rack.input"].read
423
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
424
- }).call(env("rack.input" => weirdio))
425
- }.must_raise(Rack::Lint::LintError).
426
- message.must_match(/read didn't return nil or a String/)
427
-
428
- lambda {
429
- Rack::Lint.new(lambda { |env|
430
- env["rack.input"].read
431
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
432
- }).call(env("rack.input" => eof_weirdio))
433
- }.must_raise(Rack::Lint::LintError).
434
- message.must_match(/read\(nil\) returned nil on EOF/)
435
-
436
- lambda {
437
- Rack::Lint.new(lambda { |env|
438
- env["rack.input"].rewind
439
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
440
- }).call(env("rack.input" => weirdio))
441
- }.must_raise(Rack::Lint::LintError).
442
- message.must_match(/rewind raised Errno::ESPIPE/)
443
-
444
-
445
- lambda {
446
- Rack::Lint.new(lambda { |env|
447
- env["rack.input"].close
448
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
449
- }).call(env({}))
450
- }.must_raise(Rack::Lint::LintError).
451
- message.must_match(/close must not be called/)
452
- end
453
-
454
- it "notice error handling errors" do
455
- lambda {
456
- Rack::Lint.new(lambda { |env|
457
- env["rack.errors"].write(42)
458
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
459
- }).call(env({}))
460
- }.must_raise(Rack::Lint::LintError).
461
- message.must_match(/write not called with a String/)
462
-
463
- lambda {
464
- Rack::Lint.new(lambda { |env|
465
- env["rack.errors"].close
466
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
467
- }).call(env({}))
468
- }.must_raise(Rack::Lint::LintError).
469
- message.must_match(/close must not be called/)
470
- end
471
-
472
- it "notice HEAD errors" do
473
- Rack::Lint.new(lambda { |env|
474
- [200, {"Content-type" => "test/plain", "Content-length" => "3"}, []]
475
- }).call(env({"REQUEST_METHOD" => "HEAD"})).first.must_equal 200
476
-
477
- lambda {
478
- Rack::Lint.new(lambda { |env|
479
- [200, {"Content-type" => "test/plain", "Content-length" => "3"}, ["foo"]]
480
- }).call(env({"REQUEST_METHOD" => "HEAD"}))[2].each { }
481
- }.must_raise(Rack::Lint::LintError).
482
- message.must_match(/body was given for HEAD/)
483
- end
484
-
485
- def assert_lint(*args)
486
- hello_str = "hello world"
487
- hello_str.force_encoding(Encoding::ASCII_8BIT)
488
-
489
- Rack::Lint.new(lambda { |env|
490
- env["rack.input"].send(:read, *args)
491
- [201, {"Content-type" => "text/plain", "Content-length" => "0"}, []]
492
- }).call(env({"rack.input" => StringIO.new(hello_str)})).
493
- first.must_equal 201
494
- end
495
-
496
- it "pass valid read calls" do
497
- assert_lint
498
- assert_lint 0
499
- assert_lint 1
500
- assert_lint nil
501
- assert_lint nil, ''
502
- assert_lint 1, ''
503
- end
504
- end
505
-
506
- describe "Rack::Lint::InputWrapper" do
507
- it "delegate :rewind to underlying IO object" do
508
- io = StringIO.new("123")
509
- wrapper = Rack::Lint::InputWrapper.new(io)
510
- wrapper.read.must_equal "123"
511
- wrapper.read.must_equal ""
512
- wrapper.rewind
513
- wrapper.read.must_equal "123"
514
- end
515
- end
data/test/spec_lobster.rb DELETED
@@ -1,59 +0,0 @@
1
- require 'minitest/autorun'
2
- require 'rack/lobster'
3
- require 'rack/lint'
4
- require 'rack/mock'
5
-
6
- module LobsterHelpers
7
- def lobster
8
- Rack::MockRequest.new Rack::Lint.new(Rack::Lobster.new)
9
- end
10
-
11
- def lambda_lobster
12
- Rack::MockRequest.new Rack::Lint.new(Rack::Lobster::LambdaLobster)
13
- end
14
- end
15
-
16
- describe Rack::Lobster::LambdaLobster do
17
- include LobsterHelpers
18
-
19
- it "be a single lambda" do
20
- Rack::Lobster::LambdaLobster.must_be_kind_of Proc
21
- end
22
-
23
- it "look like a lobster" do
24
- res = lambda_lobster.get("/")
25
- res.must_be :ok?
26
- res.body.must_include "(,(,,(,,,("
27
- res.body.must_include "?flip"
28
- end
29
-
30
- it "be flippable" do
31
- res = lambda_lobster.get("/?flip")
32
- res.must_be :ok?
33
- res.body.must_include "(,,,(,,(,("
34
- end
35
- end
36
-
37
- describe Rack::Lobster do
38
- include LobsterHelpers
39
-
40
- it "look like a lobster" do
41
- res = lobster.get("/")
42
- res.must_be :ok?
43
- res.body.must_include "(,(,,(,,,("
44
- res.body.must_include "?flip"
45
- res.body.must_include "crash"
46
- end
47
-
48
- it "be flippable" do
49
- res = lobster.get("/?flip=left")
50
- res.must_be :ok?
51
- res.body.must_include "),,,),,),)"
52
- end
53
-
54
- it "provide crashing for testing purposes" do
55
- lambda {
56
- lobster.get("/?flip=crash")
57
- }.must_raise RuntimeError
58
- end
59
- end