2009-12-18 20:16:58 -06:00
|
|
|
require 'stringio'
|
|
|
|
require 'time' # for Time#httpdate
|
2010-06-13 23:09:24 -05:00
|
|
|
require 'rack/deflater'
|
|
|
|
require 'rack/mock'
|
2009-12-18 20:16:58 -06:00
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
describe Rack::Deflater do
|
2009-12-18 20:16:58 -06:00
|
|
|
def build_response(status, body, accept_encoding, headers = {})
|
|
|
|
body = [body] if body.respond_to? :to_str
|
|
|
|
app = lambda { |env| [status, {}, body] }
|
|
|
|
request = Rack::MockRequest.env_for("", headers.merge("HTTP_ACCEPT_ENCODING" => accept_encoding))
|
|
|
|
response = Rack::Deflater.new(app).call(request)
|
|
|
|
|
|
|
|
return response
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "be able to deflate bodies that respond to each" do
|
2009-12-18 20:16:58 -06:00
|
|
|
body = Object.new
|
|
|
|
class << body; def each; yield("foo"); yield("bar"); end; end
|
|
|
|
|
|
|
|
response = build_response(200, body, "deflate")
|
|
|
|
|
|
|
|
response[0].should.equal(200)
|
|
|
|
response[1].should.equal({
|
|
|
|
"Content-Encoding" => "deflate",
|
|
|
|
"Vary" => "Accept-Encoding"
|
|
|
|
})
|
|
|
|
buf = ''
|
|
|
|
response[2].each { |part| buf << part }
|
|
|
|
buf.should.equal("K\313\317OJ,\002\000")
|
|
|
|
end
|
|
|
|
|
|
|
|
# TODO: This is really just a special case of the above...
|
2010-06-13 23:09:24 -05:00
|
|
|
should "be able to deflate String bodies" do
|
2009-12-18 20:16:58 -06:00
|
|
|
response = build_response(200, "Hello world!", "deflate")
|
|
|
|
|
|
|
|
response[0].should.equal(200)
|
|
|
|
response[1].should.equal({
|
|
|
|
"Content-Encoding" => "deflate",
|
|
|
|
"Vary" => "Accept-Encoding"
|
|
|
|
})
|
|
|
|
buf = ''
|
|
|
|
response[2].each { |part| buf << part }
|
|
|
|
buf.should.equal("\363H\315\311\311W(\317/\312IQ\004\000")
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "be able to gzip bodies that respond to each" do
|
2009-12-18 20:16:58 -06:00
|
|
|
body = Object.new
|
|
|
|
class << body; def each; yield("foo"); yield("bar"); end; end
|
|
|
|
|
|
|
|
response = build_response(200, body, "gzip")
|
|
|
|
|
|
|
|
response[0].should.equal(200)
|
|
|
|
response[1].should.equal({
|
|
|
|
"Content-Encoding" => "gzip",
|
|
|
|
"Vary" => "Accept-Encoding",
|
|
|
|
})
|
|
|
|
|
|
|
|
buf = ''
|
|
|
|
response[2].each { |part| buf << part }
|
|
|
|
io = StringIO.new(buf)
|
|
|
|
gz = Zlib::GzipReader.new(io)
|
|
|
|
gz.read.should.equal("foobar")
|
|
|
|
gz.close
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "be able to fallback to no deflation" do
|
2009-12-18 20:16:58 -06:00
|
|
|
response = build_response(200, "Hello world!", "superzip")
|
|
|
|
|
|
|
|
response[0].should.equal(200)
|
|
|
|
response[1].should.equal({ "Vary" => "Accept-Encoding" })
|
|
|
|
response[2].should.equal(["Hello world!"])
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "be able to skip when there is no response entity body" do
|
2009-12-18 20:16:58 -06:00
|
|
|
response = build_response(304, [], "gzip")
|
|
|
|
|
|
|
|
response[0].should.equal(304)
|
|
|
|
response[1].should.equal({})
|
|
|
|
response[2].should.equal([])
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "handle the lack of an acceptable encoding" do
|
2009-12-18 20:16:58 -06:00
|
|
|
response1 = build_response(200, "Hello world!", "identity;q=0", "PATH_INFO" => "/")
|
|
|
|
response1[0].should.equal(406)
|
|
|
|
response1[1].should.equal({"Content-Type" => "text/plain", "Content-Length" => "71"})
|
|
|
|
response1[2].should.equal(["An acceptable encoding for the requested resource / could not be found."])
|
|
|
|
|
|
|
|
response2 = build_response(200, "Hello world!", "identity;q=0", "SCRIPT_NAME" => "/foo", "PATH_INFO" => "/bar")
|
|
|
|
response2[0].should.equal(406)
|
|
|
|
response2[1].should.equal({"Content-Type" => "text/plain", "Content-Length" => "78"})
|
|
|
|
response2[2].should.equal(["An acceptable encoding for the requested resource /foo/bar could not be found."])
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "handle gzip response with Last-Modified header" do
|
2009-12-18 20:16:58 -06:00
|
|
|
last_modified = Time.now.httpdate
|
|
|
|
|
|
|
|
app = lambda { |env| [200, { "Last-Modified" => last_modified }, ["Hello World!"]] }
|
|
|
|
request = Rack::MockRequest.env_for("", "HTTP_ACCEPT_ENCODING" => "gzip")
|
|
|
|
response = Rack::Deflater.new(app).call(request)
|
|
|
|
|
|
|
|
response[0].should.equal(200)
|
|
|
|
response[1].should.equal({
|
|
|
|
"Content-Encoding" => "gzip",
|
|
|
|
"Vary" => "Accept-Encoding",
|
|
|
|
"Last-Modified" => last_modified
|
|
|
|
})
|
|
|
|
|
|
|
|
buf = ''
|
|
|
|
response[2].each { |part| buf << part }
|
|
|
|
io = StringIO.new(buf)
|
|
|
|
gz = Zlib::GzipReader.new(io)
|
|
|
|
gz.read.should.equal("Hello World!")
|
|
|
|
gz.close
|
|
|
|
end
|
|
|
|
|
2010-06-13 23:09:24 -05:00
|
|
|
should "do nothing when no-transform Cache-Control directive present" do
|
2009-12-18 20:16:58 -06:00
|
|
|
app = lambda { |env| [200, {'Cache-Control' => 'no-transform'}, ['Hello World!']] }
|
|
|
|
request = Rack::MockRequest.env_for("", "HTTP_ACCEPT_ENCODING" => "gzip")
|
|
|
|
response = Rack::Deflater.new(app).call(request)
|
|
|
|
|
|
|
|
response[0].should.equal(200)
|
|
|
|
response[1].should.not.include "Content-Encoding"
|
|
|
|
response[2].join.should.equal("Hello World!")
|
|
|
|
end
|
|
|
|
end
|