integration.rb 16.0 KB
Newer Older
1 2
require 'stringio'
require 'uri'
3
require 'active_support/core_ext/kernel/singleton_class'
4
require 'active_support/core_ext/object/inclusion'
5
require 'active_support/core_ext/object/try'
J
Joshua Peek 已提交
6
require 'rack/test'
7
require 'test/unit/assertions'
8 9

module ActionDispatch
D
David Heinemeier Hansson 已提交
10
  module Integration #:nodoc:
11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
    module RequestHelpers
      # Performs a GET request with the given parameters.
      #
      # - +path+: The URI (as a String) on which you want to perform a GET
      #   request.
      # - +parameters+: The HTTP parameters that you want to pass. This may
      #   be +nil+,
      #   a Hash, or a String that is appropriately encoded
      #   (<tt>application/x-www-form-urlencoded</tt> or
      #   <tt>multipart/form-data</tt>).
      # - +headers+: Additional HTTP headers to pass, as a Hash. The keys will
      #   automatically be upcased, with the prefix 'HTTP_' added if needed.
      #
      # This method returns an Response object, which one can use to
      # inspect the details of the response. Furthermore, if this method was
26
      # called from an ActionDispatch::IntegrationTest object, then that
27 28 29
      # object's <tt>@response</tt> instance variable will point to the same
      # response object.
      #
30 31
      # You can also perform POST, PUT, DELETE, and HEAD requests with +#post+,
      # +#put+, +#delete+, and +#head+.
32 33 34 35
      def get(path, parameters = nil, headers = nil)
        process :get, path, parameters, headers
      end

36
      # Performs a POST request with the given parameters. See +#get+ for more
37 38 39 40 41
      # details.
      def post(path, parameters = nil, headers = nil)
        process :post, path, parameters, headers
      end

42
      # Performs a PUT request with the given parameters. See +#get+ for more
43 44 45 46 47
      # details.
      def put(path, parameters = nil, headers = nil)
        process :put, path, parameters, headers
      end

48
      # Performs a DELETE request with the given parameters. See +#get+ for
49 50 51 52 53
      # more details.
      def delete(path, parameters = nil, headers = nil)
        process :delete, path, parameters, headers
      end

54
      # Performs a HEAD request with the given parameters. See +#get+ for more
55 56 57 58 59 60 61 62
      # details.
      def head(path, parameters = nil, headers = nil)
        process :head, path, parameters, headers
      end

      # Performs an XMLHttpRequest request with the given parameters, mirroring
      # a request from the Prototype library.
      #
63
      # The request_method is +:get+, +:post+, +:put+, +:delete+ or +:head+; the
64
      # parameters are +nil+, a hash, or a url-encoded or multipart string;
65
      # the headers are a hash. Keys are automatically upcased and prefixed
66 67 68
      # with 'HTTP_' if not already.
      def xml_http_request(request_method, path, parameters = nil, headers = nil)
        headers ||= {}
69 70
        headers['HTTP_X_REQUESTED_WITH'] = 'XMLHttpRequest'
        headers['HTTP_ACCEPT'] ||= [Mime::JS, Mime::HTML, Mime::XML, 'text/xml', Mime::ALL].join(', ')
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
        process(request_method, path, parameters, headers)
      end
      alias xhr :xml_http_request

      # Follow a single redirect response. If the last response was not a
      # redirect, an exception will be raised. Otherwise, the redirect is
      # performed on the location header.
      def follow_redirect!
        raise "not a redirect! #{status} #{status_message}" unless redirect?
        get(response.location)
        status
      end

      # Performs a request using the specified method, following any subsequent
      # redirect. Note that the redirects are followed until the response is
      # not a redirect--this means you may run into an infinite loop if your
      # redirect loops back to itself.
      def request_via_redirect(http_method, path, parameters = nil, headers = nil)
        process(http_method, path, parameters, headers)
        follow_redirect! while redirect?
        status
      end

      # Performs a GET request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
      def get_via_redirect(path, parameters = nil, headers = nil)
        request_via_redirect(:get, path, parameters, headers)
      end

      # Performs a POST request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
      def post_via_redirect(path, parameters = nil, headers = nil)
        request_via_redirect(:post, path, parameters, headers)
      end

      # Performs a PUT request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
      def put_via_redirect(path, parameters = nil, headers = nil)
        request_via_redirect(:put, path, parameters, headers)
      end

      # Performs a DELETE request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
      def delete_via_redirect(path, parameters = nil, headers = nil)
        request_via_redirect(:delete, path, parameters, headers)
      end
    end

119 120
    # An instance of this class represents a set of requests and responses
    # performed sequentially by a test process. Because you can instantiate
121 122 123
    # multiple sessions and run them side-by-side, you can also mimic (to some
    # limited extent) multiple simultaneous users interacting with your system.
    #
124 125 126
    # Typically, you will instantiate a new session using
    # IntegrationTest#open_session, rather than instantiating
    # Integration::Session directly.
127
    class Session
128 129
      DEFAULT_HOST = "www.example.com"

130
      include Test::Unit::Assertions
J
Joshua Peek 已提交
131
      include TestProcess, RequestHelpers, Assertions
132

133 134 135
      %w( status status_message headers body redirect? ).each do |method|
        delegate method, :to => :response, :allow_nil => true
      end
136

137 138 139
      %w( path ).each do |method|
        delegate method, :to => :request, :allow_nil => true
      end
140 141

      # The hostname used in the last request.
142 143 144 145
      def host
        @host || DEFAULT_HOST
      end
      attr_writer :host
146 147 148

      # The remote_addr used in the last request.
      attr_accessor :remote_addr
149

150 151 152
      # The Accept header to send.
      attr_accessor :accept

153 154
      # A map of the cookies returned by the last response, and which will be
      # sent with the next request.
155
      def cookies
156
        _mock_session.cookie_jar
157
      end
158 159 160 161 162 163 164 165 166 167

      # A reference to the controller instance used by the last request.
      attr_reader :controller

      # A reference to the request instance used by the last request.
      attr_reader :request

      # A reference to the response instance used by the last request.
      attr_reader :response

168 169 170
      # A running counter of the number of requests processed.
      attr_accessor :request_count

171 172
      include ActionDispatch::Routing::UrlFor

P
Pratik Naik 已提交
173
      # Create and initialize a new Session instance.
174
      def initialize(app)
J
José Valim 已提交
175
        super()
176
        @app = app
177 178 179 180 181 182 183

        # If the app is a Rails app, make url_helpers available on the session
        # This makes app.url_for and app.foo_path available in the console
        if app.respond_to?(:routes) && app.routes.respond_to?(:url_helpers)
          singleton_class.class_eval { include app.routes.url_helpers }
        end

184 185 186
        reset!
      end

187
      remove_method :default_url_options
188 189
      def default_url_options
        { :host => host, :protocol => https? ? "https" : "http" }
190 191
      end

192 193 194 195 196 197 198 199
      # Resets the instance. This can be used to reset the state information
      # in an existing session instance, so it can be used from a clean-slate
      # condition.
      #
      #   session.reset!
      def reset!
        @https = false
        @controller = @request = @response = nil
200
        @_mock_session = nil
201
        @request_count = 0
202

203
        self.host        = DEFAULT_HOST
204
        self.remote_addr = "127.0.0.1"
205 206 207
        self.accept      = "text/xml,application/xml,application/xhtml+xml," +
                           "text/html;q=0.9,text/plain;q=0.8,image/png," +
                           "*/*;q=0.5"
J
Jamis Buck 已提交
208

209
        unless defined? @named_routes_configured
J
Jamis Buck 已提交
210 211 212 213
          # the helpers are made protected by default--we make them public for
          # easier access during testing and troubleshooting.
          @named_routes_configured = true
        end
214 215 216 217 218 219
      end

      # Specify whether or not the session should mimic a secure HTTPS request.
      #
      #   session.https!
      #   session.https!(false)
220
      def https!(flag = true)
221
        @https = flag
222 223
      end

P
Pratik Naik 已提交
224
      # Return +true+ if the session is mimicking a secure HTTPS request.
225 226 227 228 229 230 231 232 233 234
      #
      #   if session.https?
      #     ...
      #   end
      def https?
        @https
      end

      # Set the host name to use in the next request.
      #
235
      #   session.host! "www.example.com"
A
Aaron Patterson 已提交
236
      alias :host! :host=
237 238

      private
239 240 241
        def _mock_session
          @_mock_session ||= Rack::MockSession.new(@app, host)
        end
242

243
        # Performs the actual request.
244 245
        def process(method, path, parameters = nil, env = nil)
          env ||= {}
246 247 248 249 250 251
          if path =~ %r{://}
            location = URI.parse(path)
            https! URI::HTTPS === location if location.scheme
            host! location.host if location.host
            path = location.query ? "#{location.path}?#{location.query}" : location.path
          end
252

253 254 255
          unless ActionController::Base < ActionController::Testing
            ActionController::Base.class_eval do
              include ActionController::Testing
256
            end
257 258
          end

259
          hostname, port = host.split(':')
260

261
          default_env = {
262
            :method => method,
263
            :params => parameters,
264

265
            "SERVER_NAME"     => hostname,
266
            "SERVER_PORT"     => port || (https? ? "443" : "80"),
267 268 269
            "HTTPS"           => https? ? "on" : "off",
            "rack.url_scheme" => https? ? "https" : "http",

270
            "REQUEST_URI"    => path,
271
            "HTTP_HOST"      => host,
272
            "REMOTE_ADDR"    => remote_addr,
273
            "CONTENT_TYPE"   => "application/x-www-form-urlencoded",
274
            "HTTP_ACCEPT"    => accept
275
          }
276

277
          session = Rack::Test::Session.new(_mock_session)
278

279
          env.reverse_merge!(default_env)
280

281 282 283 284 285 286 287 288 289
          # NOTE: rack-test v0.5 doesn't build a default uri correctly
          # Make sure requested path is always a full uri
          uri = URI.parse('/')
          uri.scheme ||= env['rack.url_scheme']
          uri.host   ||= env['SERVER_NAME']
          uri.port   ||= env['SERVER_PORT'].try(:to_i)
          uri += path

          session.request(uri.to_s, env)
290

291
          @request_count += 1
292
          @request  = ActionDispatch::Request.new(session.last_request.env)
293
          response = _mock_session.last_response
294
          @response = ActionDispatch::TestResponse.new(response.status, response.headers, response.body)
295
          @html_document = nil
296

297 298
          @controller = session.last_request.env['action_controller.instance']

299
          return response.status
300 301 302
        end
    end

303
    module Runner
304 305
      include ActionDispatch::Assertions

306
      def app
307
        @app ||= nil
308 309
      end

310 311 312
      # Reset the current session. This is useful for testing multiple sessions
      # in a single test case.
      def reset!
313
        @integration_session = Integration::Session.new(app)
314 315 316
      end

      %w(get post put head delete cookies assigns
317
         xml_http_request xhr get_via_redirect post_via_redirect).each do |method|
318
        define_method(method) do |*args|
319
          reset! unless integration_session
320
          # reset the html_document variable, but only for new get/post calls
321
          @html_document = nil unless method.in?(["cookies", "assigns"])
322
          integration_session.__send__(method, *args).tap do
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
            copy_session_variables!
          end
        end
      end

      # Open a new session instance. If a block is given, the new session is
      # yielded to the block before being returned.
      #
      #   session = open_session do |sess|
      #     sess.extend(CustomAssertions)
      #   end
      #
      # By default, a single session is automatically created for you, but you
      # can use this method to open multiple sessions that ought to be tested
      # simultaneously.
338
      def open_session(app = nil)
339 340
        dup.tap do |session|
          yield session if block_given?
341 342 343 344 345 346
        end
      end

      # Copy the instance variables from the current session instance into the
      # test instance.
      def copy_session_variables! #:nodoc:
347
        return unless integration_session
348
        %w(controller response request).each do |var|
349
          instance_variable_set("@#{var}", @integration_session.__send__(var))
350 351 352
        end
      end

353 354 355
      extend ActiveSupport::Concern
      include ActionDispatch::Routing::UrlFor

356
      def url_options
357 358
        reset! unless integration_session
        integration_session.url_options
359 360
      end

361
      def respond_to?(method, include_private = false)
362
        integration_session.respond_to?(method, include_private) || super
363 364
      end

365 366
      # Delegate unhandled messages to the current session instance.
      def method_missing(sym, *args, &block)
367 368 369
        reset! unless integration_session
        if integration_session.respond_to?(sym)
          integration_session.__send__(sym, *args, &block).tap do
370 371 372 373
            copy_session_variables!
          end
        else
          super
374 375
        end
      end
376 377 378 379 380

      private
        def integration_session
          @integration_session ||= nil
        end
381
    end
382 383
  end

384
  # An integration test spans multiple controllers and actions,
385 386 387 388
  # tying them all together to ensure they work together as expected. It tests
  # more completely than either unit or functional tests do, exercising the
  # entire stack, from the dispatcher to the database.
  #
389
  # At its simplest, you simply extend <tt>IntegrationTest</tt> and write your tests
390 391
  # using the get/post methods:
  #
392
  #   require "test_helper"
393
  #
394
  #   class ExampleTest < ActionDispatch::IntegrationTest
395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413
  #     fixtures :people
  #
  #     def test_login
  #       # get the login page
  #       get "/login"
  #       assert_equal 200, status
  #
  #       # post the login and follow through to the home page
  #       post "/login", :username => people(:jamis).username,
  #         :password => people(:jamis).password
  #       follow_redirect!
  #       assert_equal 200, status
  #       assert_equal "/home", path
  #     end
  #   end
  #
  # However, you can also have multiple session instances open per test, and
  # even extend those instances with assertions and methods to create a very
  # powerful testing DSL that is specific for your application. You can even
414
  # reference any named routes you happen to have defined.
415
  #
416
  #   require "test_helper"
417
  #
418
  #   class AdvancedTest < ActionDispatch::IntegrationTest
419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458
  #     fixtures :people, :rooms
  #
  #     def test_login_and_speak
  #       jamis, david = login(:jamis), login(:david)
  #       room = rooms(:office)
  #
  #       jamis.enter(room)
  #       jamis.speak(room, "anybody home?")
  #
  #       david.enter(room)
  #       david.speak(room, "hello!")
  #     end
  #
  #     private
  #
  #       module CustomAssertions
  #         def enter(room)
  #           # reference a named route, for maximum internal consistency!
  #           get(room_url(:id => room.id))
  #           assert(...)
  #           ...
  #         end
  #
  #         def speak(room, message)
  #           xml_http_request "/say/#{room.id}", :message => message
  #           assert(...)
  #           ...
  #         end
  #       end
  #
  #       def login(who)
  #         open_session do |sess|
  #           sess.extend(CustomAssertions)
  #           who = people(who)
  #           sess.post "/login", :username => who.username,
  #             :password => who.password
  #           assert(...)
  #         end
  #       end
  #   end
459
  class IntegrationTest < ActiveSupport::TestCase
460
    include Integration::Runner
461
    include ActionController::TemplateAssertions
462 463 464 465

    @@app = nil

    def self.app
466 467
      # DEPRECATE Rails application fallback
      # This should be set by the initializer
468
      @@app || (defined?(Rails.application) && Rails.application) || nil
469 470 471 472 473 474 475 476 477
    end

    def self.app=(app)
      @@app = app
    end

    def app
      super || self.class.app
    end
478
  end
479
end