rack 2.0.3 → 2.2.4

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 (190) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +708 -0
  3. data/CONTRIBUTING.md +136 -0
  4. data/{COPYING → MIT-LICENSE} +4 -2
  5. data/README.rdoc +152 -150
  6. data/Rakefile +37 -23
  7. data/{SPEC → SPEC.rdoc} +35 -10
  8. data/bin/rackup +1 -0
  9. data/example/lobster.ru +2 -0
  10. data/example/protectedlobster.rb +3 -1
  11. data/example/protectedlobster.ru +2 -0
  12. data/lib/rack/auth/abstract/handler.rb +3 -1
  13. data/lib/rack/auth/abstract/request.rb +1 -1
  14. data/lib/rack/auth/basic.rb +7 -4
  15. data/lib/rack/auth/digest/md5.rb +13 -11
  16. data/lib/rack/auth/digest/nonce.rb +6 -3
  17. data/lib/rack/auth/digest/params.rb +4 -2
  18. data/lib/rack/auth/digest/request.rb +5 -3
  19. data/lib/rack/body_proxy.rb +15 -14
  20. data/lib/rack/builder.rb +116 -23
  21. data/lib/rack/cascade.rb +28 -12
  22. data/lib/rack/chunked.rb +68 -20
  23. data/lib/rack/common_logger.rb +36 -25
  24. data/lib/rack/conditional_get.rb +20 -16
  25. data/lib/rack/config.rb +2 -0
  26. data/lib/rack/content_length.rb +8 -7
  27. data/lib/rack/content_type.rb +5 -4
  28. data/lib/rack/core_ext/regexp.rb +14 -0
  29. data/lib/rack/deflater.rb +59 -34
  30. data/lib/rack/directory.rb +84 -64
  31. data/lib/rack/etag.rb +8 -5
  32. data/lib/rack/events.rb +19 -20
  33. data/lib/rack/file.rb +4 -173
  34. data/lib/rack/files.rb +218 -0
  35. data/lib/rack/handler/cgi.rb +2 -3
  36. data/lib/rack/handler/fastcgi.rb +4 -4
  37. data/lib/rack/handler/lsws.rb +3 -3
  38. data/lib/rack/handler/scgi.rb +9 -8
  39. data/lib/rack/handler/thin.rb +3 -3
  40. data/lib/rack/handler/webrick.rb +15 -6
  41. data/lib/rack/handler.rb +7 -2
  42. data/lib/rack/head.rb +1 -1
  43. data/lib/rack/lint.rb +72 -26
  44. data/lib/rack/lobster.rb +10 -10
  45. data/lib/rack/lock.rb +14 -4
  46. data/lib/rack/logger.rb +2 -0
  47. data/lib/rack/media_type.rb +10 -5
  48. data/lib/rack/method_override.rb +9 -3
  49. data/lib/rack/mime.rb +9 -1
  50. data/lib/rack/mock.rb +98 -21
  51. data/lib/rack/multipart/generator.rb +17 -13
  52. data/lib/rack/multipart/parser.rb +61 -63
  53. data/lib/rack/multipart/uploaded_file.rb +15 -7
  54. data/lib/rack/multipart.rb +5 -4
  55. data/lib/rack/null_logger.rb +2 -0
  56. data/lib/rack/query_parser.rb +59 -30
  57. data/lib/rack/recursive.rb +7 -5
  58. data/lib/rack/reloader.rb +8 -4
  59. data/lib/rack/request.rb +228 -56
  60. data/lib/rack/response.rb +127 -44
  61. data/lib/rack/rewindable_input.rb +4 -3
  62. data/lib/rack/runtime.rb +6 -4
  63. data/lib/rack/sendfile.rb +13 -9
  64. data/lib/rack/server.rb +95 -24
  65. data/lib/rack/session/abstract/id.rb +100 -22
  66. data/lib/rack/session/cookie.rb +22 -14
  67. data/lib/rack/session/memcache.rb +4 -87
  68. data/lib/rack/session/pool.rb +18 -9
  69. data/lib/rack/show_exceptions.rb +22 -18
  70. data/lib/rack/show_status.rb +9 -9
  71. data/lib/rack/static.rb +23 -11
  72. data/lib/rack/tempfile_reaper.rb +1 -1
  73. data/lib/rack/urlmap.rb +12 -6
  74. data/lib/rack/utils.rb +107 -111
  75. data/lib/rack/version.rb +29 -0
  76. data/lib/rack.rb +67 -73
  77. data/rack.gemspec +40 -29
  78. metadata +25 -181
  79. data/HISTORY.md +0 -505
  80. data/test/builder/an_underscore_app.rb +0 -5
  81. data/test/builder/anything.rb +0 -5
  82. data/test/builder/comment.ru +0 -4
  83. data/test/builder/end.ru +0 -5
  84. data/test/builder/line.ru +0 -1
  85. data/test/builder/options.ru +0 -2
  86. data/test/cgi/assets/folder/test.js +0 -1
  87. data/test/cgi/assets/fonts/font.eot +0 -1
  88. data/test/cgi/assets/images/image.png +0 -1
  89. data/test/cgi/assets/index.html +0 -1
  90. data/test/cgi/assets/javascripts/app.js +0 -1
  91. data/test/cgi/assets/stylesheets/app.css +0 -1
  92. data/test/cgi/lighttpd.conf +0 -26
  93. data/test/cgi/rackup_stub.rb +0 -6
  94. data/test/cgi/sample_rackup.ru +0 -5
  95. data/test/cgi/test +0 -9
  96. data/test/cgi/test+directory/test+file +0 -1
  97. data/test/cgi/test.fcgi +0 -9
  98. data/test/cgi/test.gz +0 -0
  99. data/test/cgi/test.ru +0 -5
  100. data/test/gemloader.rb +0 -10
  101. data/test/helper.rb +0 -34
  102. data/test/multipart/bad_robots +0 -259
  103. data/test/multipart/binary +0 -0
  104. data/test/multipart/content_type_and_no_filename +0 -6
  105. data/test/multipart/empty +0 -10
  106. data/test/multipart/fail_16384_nofile +0 -814
  107. data/test/multipart/file1.txt +0 -1
  108. data/test/multipart/filename_and_modification_param +0 -7
  109. data/test/multipart/filename_and_no_name +0 -6
  110. data/test/multipart/filename_with_encoded_words +0 -7
  111. data/test/multipart/filename_with_escaped_quotes +0 -6
  112. data/test/multipart/filename_with_escaped_quotes_and_modification_param +0 -7
  113. data/test/multipart/filename_with_null_byte +0 -7
  114. data/test/multipart/filename_with_percent_escaped_quotes +0 -6
  115. data/test/multipart/filename_with_single_quote +0 -7
  116. data/test/multipart/filename_with_unescaped_percentages +0 -6
  117. data/test/multipart/filename_with_unescaped_percentages2 +0 -6
  118. data/test/multipart/filename_with_unescaped_percentages3 +0 -6
  119. data/test/multipart/filename_with_unescaped_quotes +0 -6
  120. data/test/multipart/ie +0 -6
  121. data/test/multipart/invalid_character +0 -6
  122. data/test/multipart/mixed_files +0 -21
  123. data/test/multipart/nested +0 -10
  124. data/test/multipart/none +0 -9
  125. data/test/multipart/quoted +0 -15
  126. data/test/multipart/rack-logo.png +0 -0
  127. data/test/multipart/semicolon +0 -6
  128. data/test/multipart/text +0 -15
  129. data/test/multipart/three_files_three_fields +0 -31
  130. data/test/multipart/unity3d_wwwform +0 -11
  131. data/test/multipart/webkit +0 -32
  132. data/test/rackup/config.ru +0 -31
  133. data/test/registering_handler/rack/handler/registering_myself.rb +0 -8
  134. data/test/spec_auth_basic.rb +0 -89
  135. data/test/spec_auth_digest.rb +0 -260
  136. data/test/spec_body_proxy.rb +0 -85
  137. data/test/spec_builder.rb +0 -233
  138. data/test/spec_cascade.rb +0 -63
  139. data/test/spec_cgi.rb +0 -84
  140. data/test/spec_chunked.rb +0 -103
  141. data/test/spec_common_logger.rb +0 -95
  142. data/test/spec_conditional_get.rb +0 -103
  143. data/test/spec_config.rb +0 -23
  144. data/test/spec_content_length.rb +0 -86
  145. data/test/spec_content_type.rb +0 -46
  146. data/test/spec_deflater.rb +0 -375
  147. data/test/spec_directory.rb +0 -148
  148. data/test/spec_etag.rb +0 -108
  149. data/test/spec_events.rb +0 -133
  150. data/test/spec_fastcgi.rb +0 -85
  151. data/test/spec_file.rb +0 -264
  152. data/test/spec_handler.rb +0 -57
  153. data/test/spec_head.rb +0 -46
  154. data/test/spec_lint.rb +0 -515
  155. data/test/spec_lobster.rb +0 -59
  156. data/test/spec_lock.rb +0 -194
  157. data/test/spec_logger.rb +0 -24
  158. data/test/spec_media_type.rb +0 -42
  159. data/test/spec_method_override.rb +0 -96
  160. data/test/spec_mime.rb +0 -51
  161. data/test/spec_mock.rb +0 -359
  162. data/test/spec_multipart.rb +0 -722
  163. data/test/spec_null_logger.rb +0 -21
  164. data/test/spec_recursive.rb +0 -75
  165. data/test/spec_request.rb +0 -1393
  166. data/test/spec_response.rb +0 -510
  167. data/test/spec_rewindable_input.rb +0 -128
  168. data/test/spec_runtime.rb +0 -50
  169. data/test/spec_sendfile.rb +0 -125
  170. data/test/spec_server.rb +0 -193
  171. data/test/spec_session_abstract_id.rb +0 -31
  172. data/test/spec_session_abstract_session_hash.rb +0 -45
  173. data/test/spec_session_cookie.rb +0 -442
  174. data/test/spec_session_memcache.rb +0 -320
  175. data/test/spec_session_pool.rb +0 -210
  176. data/test/spec_show_exceptions.rb +0 -80
  177. data/test/spec_show_status.rb +0 -104
  178. data/test/spec_static.rb +0 -184
  179. data/test/spec_tempfile_reaper.rb +0 -64
  180. data/test/spec_thin.rb +0 -96
  181. data/test/spec_urlmap.rb +0 -237
  182. data/test/spec_utils.rb +0 -742
  183. data/test/spec_version.rb +0 -11
  184. data/test/spec_webrick.rb +0 -209
  185. data/test/static/another/index.html +0 -1
  186. data/test/static/foo.html +0 -1
  187. data/test/static/index.html +0 -1
  188. data/test/testrequest.rb +0 -78
  189. data/test/unregistered_handler/rack/handler/unregistered.rb +0 -7
  190. 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