Skip to content

Commit

Permalink
Merge pull request #25771 from kaspth/make-test-response-assign-respo…
Browse files Browse the repository at this point in the history
…nse-parser

Let TestResponse assign a parser.
  • Loading branch information
kaspth committed Jul 13, 2016
1 parent c9b080d commit f890331
Show file tree
Hide file tree
Showing 4 changed files with 72 additions and 54 deletions.
55 changes: 3 additions & 52 deletions actionpack/lib/action_dispatch/testing/integration.rb
Expand Up @@ -6,6 +6,8 @@
require 'rack/test'
require 'minitest'

require 'action_dispatch/testing/request_encoder'

module ActionDispatch
module Integration #:nodoc:
module RequestHelpers
Expand Down Expand Up @@ -385,7 +387,6 @@ def process(method, path, params: nil, headers: nil, env: nil, xhr: false, as: n
response = _mock_session.last_response
@response = ActionDispatch::TestResponse.from_response(response)
@response.request = @request
@response.response_parser = RequestEncoder.parser(@response.content_type)
@html_document = nil
@url_options = nil

Expand All @@ -397,56 +398,6 @@ def process(method, path, params: nil, headers: nil, env: nil, xhr: false, as: n
def build_full_uri(path, env)
"#{env['rack.url_scheme']}://#{env['SERVER_NAME']}:#{env['SERVER_PORT']}#{path}"
end

class RequestEncoder # :nodoc:
@encoders = {}

attr_reader :response_parser

def initialize(mime_name, param_encoder, response_parser, url_encoded_form = false)
@mime = Mime[mime_name]

unless @mime
raise ArgumentError, "Can't register a request encoder for " \
"unregistered MIME Type: #{mime_name}. See `Mime::Type.register`."
end

@url_encoded_form = url_encoded_form
@path_format = ".#{@mime.symbol}" unless @url_encoded_form
@response_parser = response_parser || -> body { body }
@param_encoder = param_encoder || :"to_#{@mime.symbol}".to_proc
end

def append_format_to(path)
path += @path_format unless @url_encoded_form
path
end

def content_type
@mime.to_s
end

def encode_params(params)
@param_encoder.call(params)
end

def self.parser(content_type)
mime = Mime::Type.lookup(content_type)
encoder(mime ? mime.ref : nil).response_parser
end

def self.encoder(name)
@encoders[name] || WWWFormEncoder
end

def self.register_encoder(mime_name, param_encoder: nil, response_parser: nil)
@encoders[mime_name] = new(mime_name, param_encoder, response_parser)
end

register_encoder :json, response_parser: -> body { JSON.parse(body) }

WWWFormEncoder = new(:url_encoded_form, -> params { params }, nil, true)
end
end

module Runner
Expand Down Expand Up @@ -769,7 +720,7 @@ def app=(app)
end

def register_encoder(*args)
Integration::Session::RequestEncoder.register_encoder(*args)
RequestEncoder.register_encoder(*args)
end
end

Expand Down
54 changes: 54 additions & 0 deletions actionpack/lib/action_dispatch/testing/request_encoder.rb
@@ -0,0 +1,54 @@
module ActionDispatch
class RequestEncoder # :nodoc:
@encoders = {}

attr_reader :response_parser

def initialize(mime_name, param_encoder, response_parser, url_encoded_form = false)
@mime = Mime[mime_name]

unless @mime
raise ArgumentError, "Can't register a request encoder for " \
"unregistered MIME Type: #{mime_name}. See `Mime::Type.register`."
end

@url_encoded_form = url_encoded_form
@path_format = ".#{@mime.symbol}" unless @url_encoded_form
@response_parser = response_parser || -> body { body }
@param_encoder = param_encoder || :"to_#{@mime.symbol}".to_proc
end

def append_format_to(path)
if @url_encoded_form
path
else
path + @path_format
end
end

def content_type
@mime.to_s
end

def encode_params(params)
@param_encoder.call(params)
end

def self.parser(content_type)
mime = Mime::Type.lookup(content_type)
encoder(mime ? mime.ref : nil).response_parser
end

def self.encoder(name)
@encoders[name] || WWWFormEncoder
end

def self.register_encoder(mime_name, param_encoder: nil, response_parser: nil)
@encoders[mime_name] = new(mime_name, param_encoder, response_parser)
end

register_encoder :json, response_parser: -> body { JSON.parse(body) }

WWWFormEncoder = new(:url_encoded_form, -> params { params }, nil, true)
end
end
9 changes: 7 additions & 2 deletions actionpack/lib/action_dispatch/testing/test_response.rb
@@ -1,3 +1,5 @@
require 'action_dispatch/testing/request_encoder'

module ActionDispatch
# Integration test methods such as ActionDispatch::Integration::Session#get
# and ActionDispatch::Integration::Session#post return objects of class
Expand All @@ -10,6 +12,11 @@ def self.from_response(response)
new response.status, response.headers, response.body
end

def initialize(*) # :nodoc:
super
@response_parser = RequestEncoder.parser(content_type)
end

# Was the response successful?
alias_method :success?, :successful?

Expand All @@ -19,8 +26,6 @@ def self.from_response(response)
# Was there a server-side error?
alias_method :error?, :server_error?

attr_writer :response_parser # :nodoc:

def parsed_body
@parsed_body ||= @response_parser.call(body)
end
Expand Down
8 changes: 8 additions & 0 deletions actionpack/test/dispatch/test_response_test.rb
Expand Up @@ -17,4 +17,12 @@ def assert_response_code_range(range, predicate)
assert_response_code_range 500..599, :server_error?
assert_response_code_range 400..499, :client_error?
end

test "response parsing" do
response = ActionDispatch::TestResponse.create(200, {}, '')
assert_equal response.body, response.parsed_body

response = ActionDispatch::TestResponse.create(200, { 'Content-Type' => 'application/json' }, '{ "foo": "fighters" }')
assert_equal({ 'foo' => 'fighters' }, response.parsed_body)
end
end

0 comments on commit f890331

Please sign in to comment.