integration.rb 17.6 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/try'
J
Joshua Peek 已提交
5
require 'rack/test'
6
require 'minitest/unit'
7 8

module ActionDispatch
D
David Heinemeier Hansson 已提交
9
  module Integration #:nodoc:
10 11 12 13 14 15 16 17 18 19
    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>).
20
      # - +headers_or_env+: Additional headers to pass, as a Hash. The headers will be
21
      #   merged into the Rack env hash.
22
      #
R
Robin Dupret 已提交
23
      # This method returns a Response object, which one can use to
24
      # inspect the details of the response. Furthermore, if this method was
25
      # called from an ActionDispatch::IntegrationTest object, then that
26 27 28
      # object's <tt>@response</tt> instance variable will point to the same
      # response object.
      #
29 30
      # You can also perform POST, PATCH, PUT, DELETE, and HEAD requests with
      # +#post+, +#patch+, +#put+, +#delete+, and +#head+.
31 32
      def get(path, parameters = nil, headers_or_env = nil)
        process :get, path, parameters, headers_or_env
33 34
      end

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

41 42
      # Performs a PATCH request with the given parameters. See +#get+ for more
      # details.
43 44
      def patch(path, parameters = nil, headers_or_env = nil)
        process :patch, path, parameters, headers_or_env
45 46
      end

47
      # Performs a PUT request with the given parameters. See +#get+ for more
48
      # details.
49 50
      def put(path, parameters = nil, headers_or_env = nil)
        process :put, path, parameters, headers_or_env
51 52
      end

53
      # Performs a DELETE request with the given parameters. See +#get+ for
54
      # more details.
55 56
      def delete(path, parameters = nil, headers_or_env = nil)
        process :delete, path, parameters, headers_or_env
57 58
      end

59
      # Performs a HEAD request with the given parameters. See +#get+ for more
60
      # details.
61 62
      def head(path, parameters = nil, headers_or_env = nil)
        process :head, path, parameters, headers_or_env
63 64
      end

65 66
      # Performs a OPTIONS request with the given parameters. See +#get+ for
      # more details.
67 68
      def options(path, parameters = nil, headers_or_env = nil)
        process :options, path, parameters, headers_or_env
69 70
      end

71 72 73
      # Performs an XMLHttpRequest request with the given parameters, mirroring
      # a request from the Prototype library.
      #
74 75
      # The request_method is +:get+, +:post+, +:patch+, +:put+, +:delete+ or
      # +:head+; the parameters are +nil+, a hash, or a url-encoded or multipart
76
      # string; the headers are a hash.
77 78 79 80 81
      def xml_http_request(request_method, path, parameters = nil, headers_or_env = nil)
        headers_or_env ||= {}
        headers_or_env['HTTP_X_REQUESTED_WITH'] = 'XMLHttpRequest'
        headers_or_env['HTTP_ACCEPT'] ||= [Mime::JS, Mime::HTML, Mime::XML, 'text/xml', Mime::ALL].join(', ')
        process(request_method, path, parameters, headers_or_env)
82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
      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.
98 99
      def request_via_redirect(http_method, path, parameters = nil, headers_or_env = nil)
        process(http_method, path, parameters, headers_or_env)
100 101 102 103 104 105
        follow_redirect! while redirect?
        status
      end

      # Performs a GET request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
106 107
      def get_via_redirect(path, parameters = nil, headers_or_env = nil)
        request_via_redirect(:get, path, parameters, headers_or_env)
108 109 110 111
      end

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

116 117
      # Performs a PATCH request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
118 119
      def patch_via_redirect(path, parameters = nil, headers_or_env = nil)
        request_via_redirect(:patch, path, parameters, headers_or_env)
120 121
      end

122 123
      # Performs a PUT request, following any subsequent redirect.
      # See +request_via_redirect+ for more information.
124 125
      def put_via_redirect(path, parameters = nil, headers_or_env = nil)
        request_via_redirect(:put, path, parameters, headers_or_env)
126 127 128 129
      end

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

135 136
    # An instance of this class represents a set of requests and responses
    # performed sequentially by a test process. Because you can instantiate
137 138 139
    # multiple sessions and run them side-by-side, you can also mimic (to some
    # limited extent) multiple simultaneous users interacting with your system.
    #
140 141 142
    # Typically, you will instantiate a new session using
    # IntegrationTest#open_session, rather than instantiating
    # Integration::Session directly.
143
    class Session
144 145
      DEFAULT_HOST = "www.example.com"

146
      include MiniTest::Assertions
J
Joshua Peek 已提交
147
      include TestProcess, RequestHelpers, Assertions
148

149 150 151
      %w( status status_message headers body redirect? ).each do |method|
        delegate method, :to => :response, :allow_nil => true
      end
152

153 154 155
      %w( path ).each do |method|
        delegate method, :to => :request, :allow_nil => true
      end
156 157

      # The hostname used in the last request.
158 159 160 161
      def host
        @host || DEFAULT_HOST
      end
      attr_writer :host
162 163 164

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

166 167 168
      # The Accept header to send.
      attr_accessor :accept

169 170
      # A map of the cookies returned by the last response, and which will be
      # sent with the next request.
171
      def cookies
172
        _mock_session.cookie_jar
173
      end
174 175 176 177 178 179 180 181 182 183

      # 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

184 185 186
      # A running counter of the number of requests processed.
      attr_accessor :request_count

187 188
      include ActionDispatch::Routing::UrlFor

P
Pratik Naik 已提交
189
      # Create and initialize a new Session instance.
190
      def initialize(app)
J
José Valim 已提交
191
        super()
192
        @app = app
193 194 195

        # 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
196 197 198 199 200
        if app.respond_to?(:routes)
          singleton_class.class_eval do
            include app.routes.url_helpers if app.routes.respond_to?(:url_helpers)
            include app.routes.mounted_helpers if app.routes.respond_to?(:mounted_helpers)
          end
201 202
        end

203 204 205
        reset!
      end

206 207 208 209 210 211 212 213 214 215
      def url_options
        @url_options ||= default_url_options.dup.tap do |url_options|
          url_options.reverse_merge!(controller.url_options) if controller

          if @app.respond_to?(:routes) && @app.routes.respond_to?(:default_url_options)
            url_options.reverse_merge!(@app.routes.default_url_options)
          end

          url_options.reverse_merge!(:host => host, :protocol => https? ? "https" : "http")
        end
216 217
      end

218 219 220 221 222 223 224 225
      # 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
226
        @_mock_session = nil
227
        @request_count = 0
228
        @url_options = nil
229

230
        self.host        = DEFAULT_HOST
231
        self.remote_addr = "127.0.0.1"
232 233 234
        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 已提交
235

236
        unless defined? @named_routes_configured
J
Jamis Buck 已提交
237 238 239 240
          # the helpers are made protected by default--we make them public for
          # easier access during testing and troubleshooting.
          @named_routes_configured = true
        end
241 242 243 244 245 246
      end

      # Specify whether or not the session should mimic a secure HTTPS request.
      #
      #   session.https!
      #   session.https!(false)
247
      def https!(flag = true)
248
        @https = flag
249 250
      end

P
Pratik Naik 已提交
251
      # Return +true+ if the session is mimicking a secure HTTPS request.
252 253 254 255 256 257 258 259 260 261
      #
      #   if session.https?
      #     ...
      #   end
      def https?
        @https
      end

      # Set the host name to use in the next request.
      #
262
      #   session.host! "www.example.com"
A
Aaron Patterson 已提交
263
      alias :host! :host=
264 265

      private
266 267 268
        def _mock_session
          @_mock_session ||= Rack::MockSession.new(@app, host)
        end
269

270
        # Performs the actual request.
271
        def process(method, path, parameters = nil, headers_or_env = nil)
272 273 274
          if path =~ %r{://}
            location = URI.parse(path)
            https! URI::HTTPS === location if location.scheme
275
            host! "#{location.host}:#{location.port}" if location.host
276 277
            path = location.query ? "#{location.path}?#{location.query}" : location.path
          end
278

279 280 281
          unless ActionController::Base < ActionController::Testing
            ActionController::Base.class_eval do
              include ActionController::Testing
282
            end
283 284
          end

285
          hostname, port = host.split(':')
286

287
          env = {
288
            :method => method,
289
            :params => parameters,
290

291
            "SERVER_NAME"     => hostname,
292
            "SERVER_PORT"     => port || (https? ? "443" : "80"),
293 294 295
            "HTTPS"           => https? ? "on" : "off",
            "rack.url_scheme" => https? ? "https" : "http",

296
            "REQUEST_URI"    => path,
297
            "HTTP_HOST"      => host,
298
            "REMOTE_ADDR"    => remote_addr,
299
            "CONTENT_TYPE"   => "application/x-www-form-urlencoded",
300
            "HTTP_ACCEPT"    => accept
301
          }
302 303
          # this modifies the passed env directly
          Http::Headers.new(env).merge!(headers_or_env || {})
304

305
          session = Rack::Test::Session.new(_mock_session)
306

307
          env.merge!(env)
308

309 310 311 312 313 314 315 316 317
          # 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)
318

319
          @request_count += 1
320
          @request  = ActionDispatch::Request.new(session.last_request.env)
321
          response = _mock_session.last_response
322
          @response = ActionDispatch::TestResponse.new(response.status, response.headers, response.body)
323
          @html_document = nil
324
          @url_options = nil
325

326 327
          @controller = session.last_request.env['action_controller.instance']

328
          return response.status
329 330 331
        end
    end

332
    module Runner
333 334
      include ActionDispatch::Assertions

335
      def app
336
        @app ||= nil
337 338
      end

339 340 341
      # Reset the current session. This is useful for testing multiple sessions
      # in a single test case.
      def reset!
342
        @integration_session = Integration::Session.new(app)
343 344
      end

345
      %w(get post patch put head delete options cookies assigns
346
         xml_http_request xhr get_via_redirect post_via_redirect).each do |method|
347
        define_method(method) do |*args|
348
          reset! unless integration_session
349
          # reset the html_document variable, but only for new get/post calls
350
          @html_document = nil unless method == 'cookies' || method == 'assigns'
351
          integration_session.__send__(method, *args).tap do
352 353 354 355 356 357 358 359 360 361 362 363 364 365 366
            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.
367
      def open_session(app = nil)
368 369
        dup.tap do |session|
          yield session if block_given?
370 371 372 373 374 375
        end
      end

      # Copy the instance variables from the current session instance into the
      # test instance.
      def copy_session_variables! #:nodoc:
376
        return unless integration_session
377
        %w(controller response request).each do |var|
378
          instance_variable_set("@#{var}", @integration_session.__send__(var))
379 380 381
        end
      end

382 383 384 385
      def default_url_options
        reset! unless integration_session
        integration_session.default_url_options
      end
386

387
      def default_url_options=(options)
388
        reset! unless integration_session
389
        integration_session.default_url_options = options
390 391
      end

392
      def respond_to?(method, include_private = false)
393
        integration_session.respond_to?(method, include_private) || super
394 395
      end

396 397
      # Delegate unhandled messages to the current session instance.
      def method_missing(sym, *args, &block)
398 399 400
        reset! unless integration_session
        if integration_session.respond_to?(sym)
          integration_session.__send__(sym, *args, &block).tap do
401 402 403 404
            copy_session_variables!
          end
        else
          super
405 406
        end
      end
407 408 409 410 411

      private
        def integration_session
          @integration_session ||= nil
        end
412
    end
413 414
  end

415
  # An integration test spans multiple controllers and actions,
416 417 418 419
  # 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.
  #
420
  # At its simplest, you simply extend <tt>IntegrationTest</tt> and write your tests
421 422
  # using the get/post methods:
  #
423
  #   require "test_helper"
424
  #
425
  #   class ExampleTest < ActionDispatch::IntegrationTest
426 427 428 429 430 431 432 433
  #     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
A
AvnerCohen 已提交
434 435
  #       post "/login", username: people(:jamis).username,
  #         password: people(:jamis).password
436 437 438 439 440 441 442 443 444
  #       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
445
  # reference any named routes you happen to have defined.
446
  #
447
  #   require "test_helper"
448
  #
449
  #   class AdvancedTest < ActionDispatch::IntegrationTest
450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467
  #     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!
A
AvnerCohen 已提交
468
  #           get(room_url(id: room.id))
469 470 471 472 473
  #           assert(...)
  #           ...
  #         end
  #
  #         def speak(room, message)
A
AvnerCohen 已提交
474
  #           xml_http_request "/say/#{room.id}", message: message
475 476 477 478 479 480 481 482 483
  #           assert(...)
  #           ...
  #         end
  #       end
  #
  #       def login(who)
  #         open_session do |sess|
  #           sess.extend(CustomAssertions)
  #           who = people(who)
A
AvnerCohen 已提交
484 485
  #           sess.post "/login", username: who.username,
  #             password: who.password
486 487 488 489
  #           assert(...)
  #         end
  #       end
  #   end
490
  class IntegrationTest < ActiveSupport::TestCase
491
    include Integration::Runner
492
    include ActionController::TemplateAssertions
493
    include ActionDispatch::Routing::UrlFor
494 495 496 497

    @@app = nil

    def self.app
498
      if !@@app && !ActionDispatch.test_app
499
        ActiveSupport::Deprecation.warn "Rails application fallback is deprecated and no longer works, please set ActionDispatch.test_app"
500 501 502
      end

      @@app || ActionDispatch.test_app
503 504 505 506 507 508 509 510 511
    end

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

    def app
      super || self.class.app
    end
512 513 514 515 516

    def url_options
      reset! unless integration_session
      integration_session.url_options
    end
517
  end
518
end