# I advertized that this plugin worked with EdgeRails or 1.1.6 with assert_select, but I had made some changes that required EdgeRails since I last tested with 1.1.6. This is a backwards-compatibility hack. module FormTestHelper def make_request(method, path, params={}, referring_uri=nil, use_xhr=false) if self.kind_of?(ActionController::IntegrationTest) self.send(method, path, params.stringify_keys, {:referer => referring_uri}) else # Have to generate a new request and have it recognized to be backwards-compatible wih 1.1.6 request = ActionController::TestRequest.new request.request_uri = path request.env['REQUEST_METHOD'] = method.to_s.upcase ActionController::Routing::Routes.recognize(request) new_params = request.parameters params.merge!(new_params) if params[:controller] && params[:controller] != current_controller = self.instance_eval("@controller").controller_name raise "Can't follow links outside of current controller (from #{current_controller} to #{params[:controller]})" end self.instance_eval("@request").env["HTTP_REFERER"] ||= referring_uri # facilitate testing of redirect_to :back if use_xhr self.xhr(method, params.delete("action"), params.stringify_keys) else self.send(method, params.delete("action"), params.stringify_keys) end end end end # Include FormEncodedPairParser, which is in Rails 1.2 and this plugin relies upon require 'strscan' class CGIMethods #:nodoc: class FormEncodedPairParser < StringScanner attr_reader :top, :parent, :result def initialize(pairs = []) super('') @result = {} pairs.each { |key, value| parse(key, value) } end KEY_REGEXP = %r{([^\[\]=&]+)} BRACKETED_KEY_REGEXP = %r{\[([^\[\]=&]+)\]} # Parse the query string def parse(key, value) self.string = key @top, @parent = result, nil # First scan the bare key key = scan(KEY_REGEXP) or return key = post_key_check(key) # Then scan as many nestings as present until eos? r = scan(BRACKETED_KEY_REGEXP) or return key = self[1] key = post_key_check(key) end bind(key, value) end private # After we see a key, we must look ahead to determine our next action. Cases: # # [] follows the key. Then the value must be an array. # = follows the key. (A value comes next) # & or the end of string follows the key. Then the key is a flag. # otherwise, a hash follows the key. def post_key_check(key) if scan(/\[\]/) # a[b][] indicates that b is an array container(key, Array) nil elsif check(/\[[^\]]/) # a[b] indicates that a is a hash container(key, Hash) nil else # End of key? We do nothing. key end end # Add a container to the stack. # def container(key, klass) type_conflict! klass, top[key] if top.is_a?(Hash) && top.key?(key) && ! top[key].is_a?(klass) value = bind(key, klass.new) type_conflict! klass, value unless value.is_a?(klass) push(value) end # Push a value onto the 'stack', which is actually only the top 2 items. def push(value) @parent, @top = @top, value end # Bind a key (which may be nil for items in an array) to the provided value. def bind(key, value) if top.is_a? Array if key if top[-1].is_a?(Hash) && ! top[-1].key?(key) top[-1][key] = value else top << {key => value}.with_indifferent_access push top.last end else top << value end elsif top.is_a? Hash key = CGI.unescape(key) parent << (@top = {}) if top.key?(key) && parent.is_a?(Array) return top[key] ||= value else raise ArgumentError, "Don't know what to do: top is #{top.inspect}" end return value end def type_conflict!(klass, value) raise TypeError, "Conflicting types for parameter containers. " + "Expected an instance of #{klass}, but found an instance of #{value.class}. " + "This can be caused by passing Array and Hash based paramters qs[]=value&qs[key]=value. " end end end