diff options
54 files changed, 495 insertions, 413 deletions
| diff --git a/.travis.yml b/.travis.yml index 605d1ff247..c648bd2ca7 100644 --- a/.travis.yml +++ b/.travis.yml @@ -21,7 +21,7 @@ env:      - "GEM=aj:integration"      - "GEM=guides"  rvm: -  - 2.2.2 +  - 2.2.3    - ruby-head  matrix:    allow_failures: diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md index 65c05c5748..078d5f1219 100644 --- a/CODE_OF_CONDUCT.md +++ b/CODE_OF_CONDUCT.md @@ -1,22 +1,12 @@  # Contributor Code of Conduct -As contributors and maintainers of this project, and in the interest of fostering an open and welcoming community, we pledge to respect all people who contribute through reporting issues, posting feature requests, updating documentation, submitting pull requests or patches, and other activities. +The Rails team is committed to fostering a welcoming community. -We are committed to making participation in this project a harassment-free experience for everyone, regardless of level of experience, gender, gender identity and expression, sexual orientation, disability, personal appearance, body size, race, ethnicity, age, religion, or nationality. +**Our Code of Conduct can be found here**: -Examples of unacceptable behavior by participants include: +http://rubyonrails.org/conduct/ -* The use of sexualized language or imagery -* Personal attacks -* Trolling or insulting/derogatory comments -* Public or private harassment -* Publishing other's private information, such as physical or electronic addresses, without explicit permission -* Other unethical or unprofessional conduct. +For a history of updates, see the page history here: -Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct. By adopting this Code of Conduct, project maintainers commit themselves to fairly and consistently applying these principles to every aspect of managing this project. Project maintainers who do not follow or enforce the Code of Conduct may be permanently removed from the project team. +https://github.com/rails/rails.github.com/commits/master/conduct/index.html -This code of conduct applies both within project spaces and in public spaces when an individual is representing the project or its community. - -Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by opening an issue or contacting one or more of the project maintainers. - -This Code of Conduct is adapted from the [Contributor Covenant](http://contributor-covenant.org), version 1.2.0, available at [http://contributor-covenant.org/version/1/2/0/](http://contributor-covenant.org/version/1/2/0/)
\ No newline at end of file diff --git a/Gemfile.lock b/Gemfile.lock index daaa208a74..7cd4832b3a 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -27,7 +27,7 @@ GIT  GIT    remote: git://github.com/rack/rack.git -  revision: 2cb9430fce32da2f2dbccf9de97ab691351c1408 +  revision: c94e22401d4719b4d78378c7b63362cd692f9005    specs:      rack (2.0.0.alpha)        json @@ -79,7 +79,7 @@ GIT  GIT    remote: git://github.com/rails/sprockets.git -  revision: 653de5268ba862ea23457291824180c67ad5dc99 +  revision: a61550db7184fc83964fb983547ff09da9efa9be    branch: master    specs:      sprockets (4.0.0) @@ -203,7 +203,7 @@ GEM      nokogiri (1.6.6.2)        mini_portile (~> 0.6.0)      pg (0.18.2) -    psych (2.0.13) +    psych (2.0.15)      que (0.10.0)      racc (1.4.12)      rack-cache (1.2) @@ -236,7 +236,7 @@ GEM        resque (~> 1.25)        rufus-scheduler (~> 3.0)      rufus-scheduler (3.1.3) -    sass (3.4.16) +    sass (3.4.17)      sdoc (0.4.1)        json (~> 1.7, >= 1.7.7)        rdoc (~> 4.0) @@ -252,7 +252,7 @@ GEM      sigdump (0.2.3)      sinatra (1.0)        rack (>= 1.0) -    sneakers (1.1.0) +    sneakers (1.1.1)        bunny (>= 1.7.0, <= 2.0.0)        serverengine (~> 1.5.5)        thor @@ -76,7 +76,7 @@ and may also be used independently outside Rails.  We encourage you to contribute to Ruby on Rails! Please check out the  [Contributing to Ruby on Rails guide](http://edgeguides.rubyonrails.org/contributing_to_ruby_on_rails.html) for guidelines about how to proceed. [Join us!](http://contributors.rubyonrails.org) -Everyone interacting in Rails and its sub-project’s codebases, issue trackers, chat rooms, and mailing lists is expected to follow the Rails [code of conduct](CODE_OF_CONDUCT.md). +Everyone interacting in Rails and its sub-project’s codebases, issue trackers, chat rooms, and mailing lists is expected to follow the Rails [code of conduct](http://rubyonrails.org/conduct/).  ## Code Status diff --git a/actionpack/lib/action_controller/metal/request_forgery_protection.rb b/actionpack/lib/action_controller/metal/request_forgery_protection.rb index d21a778d8d..e5f3cb8e8d 100644 --- a/actionpack/lib/action_controller/metal/request_forgery_protection.rb +++ b/actionpack/lib/action_controller/metal/request_forgery_protection.rb @@ -136,7 +136,7 @@ module ActionController #:nodoc:          # This is the method that defines the application behavior when a request is found to be unverified.          def handle_unverified_request            request = @controller.request -          request.session = NullSessionHash.new(request.env) +          request.session = NullSessionHash.new(request)            request.env['action_dispatch.request.flash_hash'] = nil            request.env['rack.session.options'] = { skip: true }            request.cookie_jar = NullCookieJar.build(request, {}) @@ -145,8 +145,8 @@ module ActionController #:nodoc:          protected          class NullSessionHash < Rack::Session::Abstract::SessionHash #:nodoc: -          def initialize(env) -            super(nil, env) +          def initialize(req) +            super(nil, req)              @data = {}              @loaded = true            end diff --git a/actionpack/lib/action_controller/test_case.rb b/actionpack/lib/action_controller/test_case.rb index e012fa617e..d14483dc72 100644 --- a/actionpack/lib/action_controller/test_case.rb +++ b/actionpack/lib/action_controller/test_case.rb @@ -36,11 +36,15 @@ module ActionController      end      def query_string=(string) -      @env[Rack::QUERY_STRING] = string +      set_header Rack::QUERY_STRING, string      end      def request_parameters=(params) -      @env["action_dispatch.request.request_parameters"] = params +      set_header "action_dispatch.request.request_parameters", params +    end + +    def content_type=(type) +      set_header 'CONTENT_TYPE', type      end      def assign_parameters(routes, controller_path, action, parameters, generated_path, query_string_keys) @@ -67,10 +71,12 @@ module ActionController          end        else          if ENCODER.should_multipart?(non_path_parameters) -          @env['CONTENT_TYPE'] = ENCODER.content_type +          self.content_type = ENCODER.content_type            data = ENCODER.build_multipart non_path_parameters          else -          @env['CONTENT_TYPE'] ||= 'application/x-www-form-urlencoded' +          get_header('CONTENT_TYPE') do |k| +            set_header k, 'application/x-www-form-urlencoded' +          end            # FIXME: setting `request_parametes` is normally handled by the            # params parser middleware, and we should remove this roundtripping @@ -92,8 +98,8 @@ module ActionController            end          end -        @env['CONTENT_LENGTH'] = data.length.to_s -        @env['rack.input'] = StringIO.new(data) +        set_header 'CONTENT_LENGTH', data.length.to_s +        set_header 'rack.input', StringIO.new(data)        end        @env["PATH_INFO"] ||= generated_path @@ -450,7 +456,7 @@ module ActionController          end          if body.present? -          @request.env['RAW_POST_DATA'] = body +          @request.set_header 'RAW_POST_DATA', body          end          if http_method.present? @@ -472,15 +478,15 @@ module ActionController          end          self.cookies.update @request.cookies -        @request.env['HTTP_COOKIE'] = cookies.to_header -        @request.env['action_dispatch.cookies'] = nil +        @request.set_header 'HTTP_COOKIE', cookies.to_header +        @request.delete_header 'action_dispatch.cookies'          @request          = TestRequest.new scrub_env!(@request.env), @request.session          @response         = build_response @response_klass          @response.request = @request          @controller.recycle! -        @request.env['REQUEST_METHOD'] = http_method +        @request.set_header 'REQUEST_METHOD', http_method          parameters = parameters.symbolize_keys @@ -494,24 +500,28 @@ module ActionController          @request.flash.update(flash || {})          if xhr -          @request.env['HTTP_X_REQUESTED_WITH'] = 'XMLHttpRequest' -          @request.env['HTTP_ACCEPT'] ||= [Mime::JS, Mime::HTML, Mime::XML, 'text/xml', Mime::ALL].join(', ') +          @request.set_header 'HTTP_X_REQUESTED_WITH', 'XMLHttpRequest' +          @request.get_header('HTTP_ACCEPT') do |k| +            @request.set_header k, [Mime::JS, Mime::HTML, Mime::XML, 'text/xml', Mime::ALL].join(', ') +          end          end          @controller.request  = @request          @controller.response = @response -        @request.env["SCRIPT_NAME"] ||= @controller.config.relative_url_root +        @request.get_header("SCRIPT_NAME") do |k| +          @request.set_header k, @controller.config.relative_url_root +        end          @controller.recycle!          @controller.process(action) -        @request.env.delete 'HTTP_COOKIE' +        @request.delete_header 'HTTP_COOKIE' -        if cookies = @request.env['action_dispatch.cookies'] +        if @request.have_cookie_jar?            unless @response.committed? -            cookies.write(@response) -            self.cookies.update(cookies.instance_variable_get(:@cookies)) +            @request.cookie_jar.write(@response) +            self.cookies.update(@request.cookie_jar.instance_variable_get(:@cookies))            end          end          @response.prepare! @@ -523,8 +533,8 @@ module ActionController          end          if xhr -          @request.env.delete 'HTTP_X_REQUESTED_WITH' -          @request.env.delete 'HTTP_ACCEPT' +          @request.delete_header 'HTTP_X_REQUESTED_WITH' +          @request.delete_header 'HTTP_ACCEPT'          end          @request.query_string = '' diff --git a/actionpack/lib/action_dispatch/http/filter_parameters.rb b/actionpack/lib/action_dispatch/http/filter_parameters.rb index 3170389b36..e70e90018c 100644 --- a/actionpack/lib/action_dispatch/http/filter_parameters.rb +++ b/actionpack/lib/action_dispatch/http/filter_parameters.rb @@ -50,13 +50,13 @@ module ActionDispatch      protected        def parameter_filter -        parameter_filter_for @env.fetch("action_dispatch.parameter_filter") { +        parameter_filter_for get_header("action_dispatch.parameter_filter") {            return NULL_PARAM_FILTER          }        end        def env_filter -        user_key = @env.fetch("action_dispatch.parameter_filter") { +        user_key = get_header("action_dispatch.parameter_filter") {            return NULL_ENV_FILTER          }          parameter_filter_for(Array(user_key) + ENV_MATCH) diff --git a/actionpack/lib/action_dispatch/http/headers.rb b/actionpack/lib/action_dispatch/http/headers.rb index bc5410dc38..fbdec6c132 100644 --- a/actionpack/lib/action_dispatch/http/headers.rb +++ b/actionpack/lib/action_dispatch/http/headers.rb @@ -30,27 +30,32 @@ module ActionDispatch        HTTP_HEADER = /\A[A-Za-z0-9-]+\z/        include Enumerable -      attr_reader :env -      def initialize(env = {}) # :nodoc: -        @env = env +      def self.from_hash(hash) +        new ActionDispatch::Request.new hash +      end + +      def initialize(request) # :nodoc: +        @req = request        end        # Returns the value for the given key mapped to @env.        def [](key) -        @env[env_name(key)] +        @req.get_header env_name(key)        end        # Sets the given value for the key mapped to @env.        def []=(key, value) -        @env[env_name(key)] = value +        @req.set_header env_name(key), value        end        def key?(key) -        @env.key? env_name(key) +        @req.has_header? env_name(key)        end        alias :include? :key? +      DEFAULT = Object.new # :nodoc: +        # Returns the value for the given key mapped to @env.        #        # If the key is not found and an optional code block is not provided, @@ -58,18 +63,22 @@ module ActionDispatch        #        # If the code block is provided, then it will be run and        # its result returned. -      def fetch(key, *args, &block) -        @env.fetch env_name(key), *args, &block +      def fetch(key, default = DEFAULT) +        @req.get_header(env_name(key)) do +          return default unless default == DEFAULT +          return yield if block_given? +          raise NameError, key +        end        end        def each(&block) -        @env.each(&block) +        @req.each_header(&block)        end        # Returns a new Http::Headers instance containing the contents of        # <tt>headers_or_env</tt> and the original instance.        def merge(headers_or_env) -        headers = Http::Headers.new(env.dup) +        headers = @req.dup.headers          headers.merge!(headers_or_env)          headers        end @@ -79,11 +88,14 @@ module ActionDispatch        # <tt>headers_or_env</tt>.        def merge!(headers_or_env)          headers_or_env.each do |key, value| -          self[env_name(key)] = value +          @req.set_header env_name(key), value          end        end +      def env; @req.env.dup; end +        private +        # Converts a HTTP header name to an environment variable name if it is        # not contained within the headers hash.        def env_name(key) diff --git a/actionpack/lib/action_dispatch/http/mime_negotiation.rb b/actionpack/lib/action_dispatch/http/mime_negotiation.rb index ff336b7354..e01d5ecc8f 100644 --- a/actionpack/lib/action_dispatch/http/mime_negotiation.rb +++ b/actionpack/lib/action_dispatch/http/mime_negotiation.rb @@ -15,12 +15,13 @@ module ActionDispatch        # For backward compatibility, the post \format is extracted from the        # X-Post-Data-Format HTTP header if present.        def content_mime_type -        @env["action_dispatch.request.content_type"] ||= begin -          if @env['CONTENT_TYPE'] =~ /^([^,\;]*)/ +        get_header("action_dispatch.request.content_type") do |k| +          v = if get_header('CONTENT_TYPE') =~ /^([^,\;]*)/              Mime::Type.lookup($1.strip.downcase)            else              nil            end +          set_header k, v          end        end @@ -30,14 +31,15 @@ module ActionDispatch        # Returns the accepted MIME type for the request.        def accepts -        @env["action_dispatch.request.accepts"] ||= begin -          header = @env['HTTP_ACCEPT'].to_s.strip +        get_header("action_dispatch.request.accepts") do |k| +          header = get_header('HTTP_ACCEPT').to_s.strip -          if header.empty? +          v = if header.empty?              [content_mime_type]            else              Mime::Type.parse(header)            end +          set_header k, v          end        end @@ -52,14 +54,14 @@ module ActionDispatch        end        def formats -        @env["action_dispatch.request.formats"] ||= begin +        get_header("action_dispatch.request.formats") do |k|            params_readable = begin                                parameters[:format]                              rescue ActionController::BadRequest                                false                              end -          if params_readable +          v = if params_readable              Array(Mime[parameters[:format]])            elsif use_accept_header && valid_accept_header              accepts @@ -68,6 +70,7 @@ module ActionDispatch            else              [Mime::HTML]            end +          set_header k, v          end        end @@ -102,7 +105,7 @@ module ActionDispatch        #   end        def format=(extension)          parameters[:format] = extension.to_s -        @env["action_dispatch.request.formats"] = [Mime::Type.lookup_by_extension(parameters[:format])] +        set_header "action_dispatch.request.formats", [Mime::Type.lookup_by_extension(parameters[:format])]        end        # Sets the \formats by string extensions. This differs from #format= by allowing you @@ -121,9 +124,9 @@ module ActionDispatch        #   end        def formats=(extensions)          parameters[:format] = extensions.first.to_s -        @env["action_dispatch.request.formats"] = extensions.collect do |extension| +        set_header "action_dispatch.request.formats", extensions.collect { |extension|            Mime::Type.lookup_by_extension(extension) -        end +        }        end        # Receives an array of mimes and return the first user sent mime that diff --git a/actionpack/lib/action_dispatch/http/parameters.rb b/actionpack/lib/action_dispatch/http/parameters.rb index 4defb7f858..277207ae6e 100644 --- a/actionpack/lib/action_dispatch/http/parameters.rb +++ b/actionpack/lib/action_dispatch/http/parameters.rb @@ -8,20 +8,23 @@ module ActionDispatch        # Returns both GET and POST \parameters in a single hash.        def parameters -        @env["action_dispatch.request.parameters"] ||= begin -          params = begin -            request_parameters.merge(query_parameters) -          rescue EOFError -            query_parameters.dup -          end -          params.merge!(path_parameters) -        end +        params = get_header("action_dispatch.request.parameters") +        return params if params + +        params = begin +                   request_parameters.merge(query_parameters) +                 rescue EOFError +                   query_parameters.dup +                 end +        params.merge!(path_parameters) +        set_header("action_dispatch.request.parameters", params) +        params        end        alias :params :parameters        def path_parameters=(parameters) #:nodoc: -        @env.delete('action_dispatch.request.parameters') -        @env[PARAMETERS_KEY] = parameters +        delete_header('action_dispatch.request.parameters') +        set_header PARAMETERS_KEY, parameters        end        # Returns a hash with the \parameters used to form the \path of the request. @@ -29,7 +32,7 @@ module ActionDispatch        #        #   {'action' => 'my_action', 'controller' => 'my_controller'}        def path_parameters -        @env[PARAMETERS_KEY] ||= {} +        get_header(PARAMETERS_KEY) || {}        end      private diff --git a/actionpack/lib/action_dispatch/http/request.rb b/actionpack/lib/action_dispatch/http/request.rb index de28cd0998..4748a54550 100644 --- a/actionpack/lib/action_dispatch/http/request.rb +++ b/actionpack/lib/action_dispatch/http/request.rb @@ -20,8 +20,6 @@ module ActionDispatch      include ActionDispatch::Http::FilterParameters      include ActionDispatch::Http::URL -    HTTP_X_REQUEST_ID = "HTTP_X_REQUEST_ID".freeze # :nodoc: -      autoload :Session, 'action_dispatch/request/session'      autoload :Utils,   'action_dispatch/request/utils' @@ -31,17 +29,19 @@ module ActionDispatch          PATH_TRANSLATED REMOTE_HOST          REMOTE_IDENT REMOTE_USER REMOTE_ADDR          SERVER_NAME SERVER_PROTOCOL +        ORIGINAL_SCRIPT_NAME          HTTP_ACCEPT HTTP_ACCEPT_CHARSET HTTP_ACCEPT_ENCODING          HTTP_ACCEPT_LANGUAGE HTTP_CACHE_CONTROL HTTP_FROM          HTTP_NEGOTIATE HTTP_PRAGMA HTTP_CLIENT_IP          HTTP_X_FORWARDED_FOR HTTP_VERSION +        HTTP_X_REQUEST_ID          ].freeze      ENV_METHODS.each do |env|        class_eval <<-METHOD, __FILE__, __LINE__ + 1          def #{env.sub(/^HTTP_/n, '').downcase}  # def accept_charset -          @env["#{env}".freeze]                 #   @env["HTTP_ACCEPT_CHARSET".freeze] +          get_header "#{env}".freeze            #   get_header "HTTP_ACCEPT_CHARSET".freeze          end                                     # end        METHOD      end @@ -67,8 +67,20 @@ module ActionDispatch        end      end +    def controller_class +      check_path_parameters! +      params = path_parameters +      controller_param = params[:controller].underscore if params.key?(:controller) +      params[:action] ||= 'index' + +      yield unless controller_param + +      const_name = "#{controller_param.camelize}Controller" +      ActiveSupport::Dependencies.constantize(const_name) +    end +      def key?(key) -      @env.key?(key) +      has_header? key      end      # List of HTTP request methods from the following RFCs: @@ -109,44 +121,40 @@ module ActionDispatch      end      def routes # :nodoc: -      env["action_dispatch.routes".freeze] +      get_header("action_dispatch.routes".freeze)      end      def routes=(routes) # :nodoc: -      env["action_dispatch.routes".freeze] = routes -    end - -    def original_script_name # :nodoc: -      env['ORIGINAL_SCRIPT_NAME'.freeze] +      set_header("action_dispatch.routes".freeze, routes)      end      def engine_script_name(_routes) # :nodoc: -      env[_routes.env_key] +      get_header(_routes.env_key)      end      def engine_script_name=(name) # :nodoc: -      env[routes.env_key] = name.dup +      set_header(routes.env_key, name.dup)      end      def request_method=(request_method) #:nodoc:        if check_method(request_method) -        @request_method = env["REQUEST_METHOD"] = request_method +        @request_method = set_header("REQUEST_METHOD", request_method)        end      end      def controller_instance # :nodoc: -      env['action_controller.instance'.freeze] +      get_header('action_controller.instance'.freeze)      end      def controller_instance=(controller) # :nodoc: -      env['action_controller.instance'.freeze] = controller +      set_header('action_controller.instance'.freeze, controller)      end      def show_exceptions? # :nodoc:        # We're treating `nil` as "unset", and we want the default setting to be        # `true`.  This logic should be extracted to `env_config` and calculated        # once. -      !(env['action_dispatch.show_exceptions'.freeze] == false) +      !(get_header('action_dispatch.show_exceptions'.freeze) == false)      end      # Returns a symbol form of the #request_method @@ -158,7 +166,7 @@ module ActionDispatch      # even if it was overridden by middleware. See #request_method for      # more information.      def method -      @method ||= check_method(env["rack.methodoverride.original_method"] || env['REQUEST_METHOD']) +      @method ||= check_method(get_header("rack.methodoverride.original_method") || get_header('REQUEST_METHOD'))      end      # Returns a symbol form of the #method @@ -170,7 +178,7 @@ module ActionDispatch      #      #   request.headers["Content-Type"] # => "text/plain"      def headers -      @headers ||= Http::Headers.new(@env) +      @headers ||= Http::Headers.new(self)      end      # Returns a +String+ with the last requested path including their params. @@ -181,7 +189,7 @@ module ActionDispatch      #    # get '/foo?bar'      #    request.original_fullpath # => '/foo?bar'      def original_fullpath -      @original_fullpath ||= (env["ORIGINAL_FULLPATH"] || fullpath) +      @original_fullpath ||= (get_header("ORIGINAL_FULLPATH") || fullpath)      end      # Returns the +String+ full path including params of the last URL requested. @@ -220,7 +228,7 @@ module ActionDispatch      # (case-insensitive), which may need to be manually added depending on the      # choice of JavaScript libraries and frameworks.      def xml_http_request? -      @env['HTTP_X_REQUESTED_WITH'] =~ /XMLHttpRequest/i +      get_header('HTTP_X_REQUESTED_WITH') =~ /XMLHttpRequest/i      end      alias :xhr? :xml_http_request? @@ -232,11 +240,11 @@ module ActionDispatch      # Returns the IP address of client as a +String+,      # usually set by the RemoteIp middleware.      def remote_ip -      @remote_ip ||= (@env["action_dispatch.remote_ip"] || ip).to_s +      @remote_ip ||= (get_header("action_dispatch.remote_ip") || ip).to_s      end      def remote_ip=(remote_ip) -      @env["action_dispatch.remote_ip".freeze] = remote_ip +      set_header "action_dispatch.remote_ip".freeze, remote_ip      end      ACTION_DISPATCH_REQUEST_ID = "action_dispatch.request_id".freeze # :nodoc: @@ -248,43 +256,39 @@ module ActionDispatch      # This unique ID is useful for tracing a request from end-to-end as part of logging or debugging.      # This relies on the rack variable set by the ActionDispatch::RequestId middleware.      def request_id -      env[ACTION_DISPATCH_REQUEST_ID] +      get_header ACTION_DISPATCH_REQUEST_ID      end      def request_id=(id) # :nodoc: -      env[ACTION_DISPATCH_REQUEST_ID] = id +      set_header ACTION_DISPATCH_REQUEST_ID, id      end      alias_method :uuid, :request_id -    def x_request_id # :nodoc: -      @env[HTTP_X_REQUEST_ID] -    end -      # Returns the lowercase name of the HTTP server software.      def server_software -      (@env['SERVER_SOFTWARE'] && /^([a-zA-Z]+)/ =~ @env['SERVER_SOFTWARE']) ? $1.downcase : nil +      (get_header('SERVER_SOFTWARE') && /^([a-zA-Z]+)/ =~ get_header('SERVER_SOFTWARE')) ? $1.downcase : nil      end      # Read the request \body. This is useful for web services that need to      # work with raw requests directly.      def raw_post -      unless @env.include? 'RAW_POST_DATA' +      unless has_header? 'RAW_POST_DATA'          raw_post_body = body -        @env['RAW_POST_DATA'] = raw_post_body.read(content_length) +        set_header('RAW_POST_DATA', raw_post_body.read(content_length))          raw_post_body.rewind if raw_post_body.respond_to?(:rewind)        end -      @env['RAW_POST_DATA'] +      get_header 'RAW_POST_DATA'      end      # The request body is an IO input stream. If the RAW_POST_DATA environment      # variable is already set, wrap it in a StringIO.      def body -      if raw_post = @env['RAW_POST_DATA'] +      if raw_post = get_header('RAW_POST_DATA')          raw_post.force_encoding(Encoding::BINARY)          StringIO.new(raw_post)        else -        @env['rack.input'] +        body_stream        end      end @@ -295,7 +299,7 @@ module ActionDispatch      end      def body_stream #:nodoc: -      @env['rack.input'] +      get_header('rack.input')      end      # TODO This should be broken apart into AD::Request::Session and probably @@ -306,15 +310,15 @@ module ActionDispatch        else          self.session = {}        end -      @env['action_dispatch.request.flash_hash'] = nil +      set_header('action_dispatch.request.flash_hash', nil)      end      def session=(session) #:nodoc: -      Session.set @env, session +      Session.set self, session      end      def session_options=(options) -      Session::Options.set @env, options +      Session::Options.set self, options      end      # Override Rack's GET method to support indifferent access @@ -336,10 +340,10 @@ module ActionDispatch      # Returns the authorization header regardless of whether it was specified directly or through one of the      # proxy alternatives.      def authorization -      @env['HTTP_AUTHORIZATION']   || -      @env['X-HTTP_AUTHORIZATION'] || -      @env['X_HTTP_AUTHORIZATION'] || -      @env['REDIRECT_X_HTTP_AUTHORIZATION'] +      get_header('HTTP_AUTHORIZATION')   || +      get_header('X-HTTP_AUTHORIZATION') || +      get_header('X_HTTP_AUTHORIZATION') || +      get_header('REDIRECT_X_HTTP_AUTHORIZATION')      end      # True if the request came from localhost, 127.0.0.1. @@ -348,11 +352,11 @@ module ActionDispatch      end      def request_parameters=(params) -      env["action_dispatch.request.request_parameters".freeze] = params +      set_header("action_dispatch.request.request_parameters".freeze, params)      end      def logger -      env["action_dispatch.logger".freeze] +      get_header("action_dispatch.logger".freeze)      end      private diff --git a/actionpack/lib/action_dispatch/middleware/cookies.rb b/actionpack/lib/action_dispatch/middleware/cookies.rb index cf4f654ed6..e4c5b0bd29 100644 --- a/actionpack/lib/action_dispatch/middleware/cookies.rb +++ b/actionpack/lib/action_dispatch/middleware/cookies.rb @@ -8,48 +8,50 @@ require 'active_support/json'  module ActionDispatch    class Request < Rack::Request      def cookie_jar -      env['action_dispatch.cookies'.freeze] ||= Cookies::CookieJar.build(self, cookies) +      get_header('action_dispatch.cookies'.freeze) do |k| +        self.cookie_jar = Cookies::CookieJar.build(self, cookies) +      end      end      # :stopdoc:      def have_cookie_jar? -      env.key? 'action_dispatch.cookies'.freeze +      has_header? 'action_dispatch.cookies'.freeze      end      def cookie_jar=(jar) -      env['action_dispatch.cookies'.freeze] = jar +      set_header 'action_dispatch.cookies'.freeze, jar      end      def key_generator -      env[Cookies::GENERATOR_KEY] +      get_header Cookies::GENERATOR_KEY      end      def signed_cookie_salt -      env[Cookies::SIGNED_COOKIE_SALT] +      get_header Cookies::SIGNED_COOKIE_SALT      end      def encrypted_cookie_salt -      env[Cookies::ENCRYPTED_COOKIE_SALT] +      get_header Cookies::ENCRYPTED_COOKIE_SALT      end      def encrypted_signed_cookie_salt -      env[Cookies::ENCRYPTED_SIGNED_COOKIE_SALT] +      get_header Cookies::ENCRYPTED_SIGNED_COOKIE_SALT      end      def secret_token -      env[Cookies::SECRET_TOKEN] +      get_header Cookies::SECRET_TOKEN      end      def secret_key_base -      env[Cookies::SECRET_KEY_BASE] +      get_header Cookies::SECRET_KEY_BASE      end      def cookies_serializer -      env[Cookies::COOKIES_SERIALIZER] +      get_header Cookies::COOKIES_SERIALIZER      end      def cookies_digest -      env[Cookies::COOKIES_DIGEST] +      get_header Cookies::COOKIES_DIGEST      end      # :startdoc:    end diff --git a/actionpack/lib/action_dispatch/middleware/debug_exceptions.rb b/actionpack/lib/action_dispatch/middleware/debug_exceptions.rb index 226a688fe2..66bb74b9c5 100644 --- a/actionpack/lib/action_dispatch/middleware/debug_exceptions.rb +++ b/actionpack/lib/action_dispatch/middleware/debug_exceptions.rb @@ -55,18 +55,17 @@ module ActionDispatch        response      rescue Exception => exception        raise exception unless request.show_exceptions? -      render_exception(env, exception) +      render_exception(request, exception)      end      private -    def render_exception(env, exception) -      backtrace_cleaner = env['action_dispatch.backtrace_cleaner'] +    def render_exception(request, exception) +      backtrace_cleaner = request.get_header('action_dispatch.backtrace_cleaner')        wrapper = ExceptionWrapper.new(backtrace_cleaner, exception) -      log_error(env, wrapper) +      log_error(request, wrapper) -      if env['action_dispatch.show_detailed_exceptions'] -        request = Request.new(env) +      if request.get_header('action_dispatch.show_detailed_exceptions')          traces = wrapper.traces          trace_to_show = 'Application Trace' @@ -108,8 +107,8 @@ module ActionDispatch        [status, {'Content-Type' => "#{format}; charset=#{Response.default_charset}", 'Content-Length' => body.bytesize.to_s}, [body]]      end -    def log_error(env, wrapper) -      logger = logger(env) +    def log_error(request, wrapper) +      logger = logger(request)        return unless logger        exception = wrapper.exception @@ -125,8 +124,8 @@ module ActionDispatch        end      end -    def logger(env) -      env['action_dispatch.logger'] || stderr_logger +    def logger(request) +      request.logger || stderr_logger      end      def stderr_logger diff --git a/actionpack/lib/action_dispatch/middleware/flash.rb b/actionpack/lib/action_dispatch/middleware/flash.rb index 23da169b22..6041f84834 100644 --- a/actionpack/lib/action_dispatch/middleware/flash.rb +++ b/actionpack/lib/action_dispatch/middleware/flash.rb @@ -6,15 +6,17 @@ module ActionDispatch      # read a notice you put there or <tt>flash["notice"] = "hello"</tt>      # to put a new one.      def flash -      @env[Flash::KEY] ||= Flash::FlashHash.from_session_value(session["flash"]) +      flash = flash_hash +      return flash if flash +      self.flash = Flash::FlashHash.from_session_value(session["flash"])      end      def flash=(flash) -      @env[Flash::KEY] = flash +      set_header Flash::KEY, flash      end      def flash_hash # :nodoc: -      @env[Flash::KEY] +      get_header Flash::KEY      end    end @@ -274,7 +276,7 @@ module ActionDispatch        req = ActionDispatch::Request.new env        @app.call(env)      ensure -      session    = Request::Session.find(env) || {} +      session    = Request::Session.find(req) || {}        flash_hash = req.flash_hash        if flash_hash && (flash_hash.present? || session.key?('flash')) diff --git a/actionpack/lib/action_dispatch/middleware/public_exceptions.rb b/actionpack/lib/action_dispatch/middleware/public_exceptions.rb index 7cde76b30e..0f27984550 100644 --- a/actionpack/lib/action_dispatch/middleware/public_exceptions.rb +++ b/actionpack/lib/action_dispatch/middleware/public_exceptions.rb @@ -17,8 +17,8 @@ module ActionDispatch      end      def call(env) -      status       = env["PATH_INFO"][1..-1].to_i        request      = ActionDispatch::Request.new(env) +      status       = request.path_info[1..-1].to_i        content_type = request.formats.first        body         = { :status => status, :error => Rack::Utils::HTTP_STATUS_CODES.fetch(status, Rack::Utils::HTTP_STATUS_CODES[500]) } diff --git a/actionpack/lib/action_dispatch/middleware/session/abstract_store.rb b/actionpack/lib/action_dispatch/middleware/session/abstract_store.rb index 84df55fd5a..b924df789f 100644 --- a/actionpack/lib/action_dispatch/middleware/session/abstract_store.rb +++ b/actionpack/lib/action_dispatch/middleware/session/abstract_store.rb @@ -36,6 +36,11 @@ module ActionDispatch          @default_options.delete(:sidbits)          @default_options.delete(:secure_random)        end + +      private +      def make_request(env) +        ActionDispatch::Request.new env +      end      end      module StaleSessionCheck @@ -65,8 +70,8 @@ module ActionDispatch      end      module SessionObject # :nodoc: -      def prepare_session(env) -        Request::Session.create(self, env, @default_options) +      def prepare_session(req) +        Request::Session.create(self, req, @default_options)        end        def loaded_session?(session) @@ -81,8 +86,7 @@ module ActionDispatch        private -      def set_cookie(env, session_id, cookie) -        request = ActionDispatch::Request.new(env) +      def set_cookie(request, session_id, cookie)          request.cookie_jar[key] = cookie        end      end diff --git a/actionpack/lib/action_dispatch/middleware/session/cookie_store.rb b/actionpack/lib/action_dispatch/middleware/session/cookie_store.rb index d8f9614904..e225f356df 100644 --- a/actionpack/lib/action_dispatch/middleware/session/cookie_store.rb +++ b/actionpack/lib/action_dispatch/middleware/session/cookie_store.rb @@ -71,16 +71,16 @@ module ActionDispatch          super(app, options.merge!(:cookie_only => true))        end -      def destroy_session(env, session_id, options) +      def destroy_session(req, session_id, options)          new_sid = generate_sid unless options[:drop]          # Reset hash and Assign the new session id -        env["action_dispatch.request.unsigned_session_cookie"] = new_sid ? { "session_id" => new_sid } : {} +        req.set_header("action_dispatch.request.unsigned_session_cookie", new_sid ? { "session_id" => new_sid } : {})          new_sid        end -      def load_session(env) +      def load_session(req)          stale_session_check! do -          data = unpacked_cookie_data(env) +          data = unpacked_cookie_data(req)            data = persistent_session_id!(data)            [data["session_id"], data]          end @@ -88,20 +88,21 @@ module ActionDispatch        private -      def extract_session_id(env) +      def extract_session_id(req)          stale_session_check! do -          unpacked_cookie_data(env)["session_id"] +          unpacked_cookie_data(req)["session_id"]          end        end -      def unpacked_cookie_data(env) -        env["action_dispatch.request.unsigned_session_cookie"] ||= begin -          stale_session_check! do -            if data = get_cookie(env) +      def unpacked_cookie_data(req) +        req.get_header("action_dispatch.request.unsigned_session_cookie") do |k| +          v = stale_session_check! do +            if data = get_cookie(req)                data.stringify_keys!              end              data || {}            end +          req.set_header k, v          end        end @@ -111,21 +112,20 @@ module ActionDispatch          data        end -      def set_session(env, sid, session_data, options) +      def set_session(req, sid, session_data, options)          session_data["session_id"] = sid          session_data        end -      def set_cookie(env, session_id, cookie) -        cookie_jar(env)[@key] = cookie +      def set_cookie(request, session_id, cookie) +        cookie_jar(request)[@key] = cookie        end -      def get_cookie(env) -        cookie_jar(env)[@key] +      def get_cookie(req) +        cookie_jar(req)[@key]        end -      def cookie_jar(env) -        request = ActionDispatch::Request.new(env) +      def cookie_jar(request)          request.cookie_jar.signed_or_encrypted        end      end diff --git a/actionpack/lib/action_dispatch/middleware/show_exceptions.rb b/actionpack/lib/action_dispatch/middleware/show_exceptions.rb index 12d8dab8eb..64695f9738 100644 --- a/actionpack/lib/action_dispatch/middleware/show_exceptions.rb +++ b/actionpack/lib/action_dispatch/middleware/show_exceptions.rb @@ -31,7 +31,7 @@ module ActionDispatch        @app.call(env)      rescue Exception => exception        if request.show_exceptions? -        render_exception(env, exception) +        render_exception(request, exception)        else          raise exception        end @@ -39,14 +39,14 @@ module ActionDispatch      private -    def render_exception(env, exception) -      backtrace_cleaner = env['action_dispatch.backtrace_cleaner'] +    def render_exception(request, exception) +      backtrace_cleaner = request.get_header 'action_dispatch.backtrace_cleaner'        wrapper = ExceptionWrapper.new(backtrace_cleaner, exception)        status  = wrapper.status_code -      env["action_dispatch.exception"] = wrapper.exception -      env["action_dispatch.original_path"] = env["PATH_INFO"] -      env["PATH_INFO"] = "/#{status}" -      response = @exceptions_app.call(env) +      request.set_header "action_dispatch.exception", wrapper.exception +      request.set_header "action_dispatch.original_path", request.path_info +      request.path_info = "/#{status}" +      response = @exceptions_app.call(request.env)        response[1]['X-Cascade'] == 'pass' ? pass_response(status) : response      rescue Exception => failsafe_error        $stderr.puts "Error during failsafe response: #{failsafe_error}\n  #{failsafe_error.backtrace * "\n  "}" diff --git a/actionpack/lib/action_dispatch/request/session.rb b/actionpack/lib/action_dispatch/request/session.rb index bae3d909fa..b946ccb49f 100644 --- a/actionpack/lib/action_dispatch/request/session.rb +++ b/actionpack/lib/action_dispatch/request/session.rb @@ -11,31 +11,31 @@ module ActionDispatch        Unspecified = Object.new        # Creates a session hash, merging the properties of the previous session if any -      def self.create(store, env, default_options) -        session_was = find env -        session     = Request::Session.new(store, env) +      def self.create(store, req, default_options) +        session_was = find req +        session     = Request::Session.new(store, req)          session.merge! session_was if session_was -        set(env, session) -        Options.set(env, Request::Session::Options.new(store, default_options)) +        set(req, session) +        Options.set(req, Request::Session::Options.new(store, default_options))          session        end -      def self.find(env) -        env[ENV_SESSION_KEY] +      def self.find(req) +        req.get_header ENV_SESSION_KEY        end -      def self.set(env, session) -        env[ENV_SESSION_KEY] = session +      def self.set(req, session) +        req.set_header ENV_SESSION_KEY, session        end        class Options #:nodoc: -        def self.set(env, options) -          env[ENV_SESSION_OPTIONS_KEY] = options +        def self.set(req, options) +          req.set_header ENV_SESSION_OPTIONS_KEY, options          end -        def self.find(env) -          env[ENV_SESSION_OPTIONS_KEY] +        def self.find(req) +          req.get_header ENV_SESSION_OPTIONS_KEY          end          def initialize(by, default_options) @@ -47,9 +47,9 @@ module ActionDispatch            @delegate[key]          end -        def id(env) +        def id(req)            @delegate.fetch(:id) { -            @by.send(:extract_session_id, env) +            @by.send(:extract_session_id, req)            }          end @@ -58,26 +58,26 @@ module ActionDispatch          def values_at(*args); @delegate.values_at(*args); end        end -      def initialize(by, env) +      def initialize(by, req)          @by       = by -        @env      = env +        @req      = req          @delegate = {}          @loaded   = false          @exists   = nil # we haven't checked yet        end        def id -        options.id(@env) +        options.id(@req)        end        def options -        Options.find @env +        Options.find @req        end        def destroy          clear          options = self.options || {} -        @by.send(:destroy_session, @env, options.id(@env), options) +        @by.send(:destroy_session, @req, options.id(@req), options)          # Load the new sid to be written with the response          @loaded = false @@ -181,7 +181,7 @@ module ActionDispatch        def exists?          return @exists unless @exists.nil? -        @exists = @by.send(:session_exists?, @env) +        @exists = @by.send(:session_exists?, @req)        end        def loaded? @@ -209,7 +209,7 @@ module ActionDispatch        end        def load! -        id, session = @by.load_session @env +        id, session = @by.load_session @req          options[:id] = id          @delegate.replace(stringify_keys(session))          @loaded = true diff --git a/actionpack/lib/action_dispatch/routing/route_set.rb b/actionpack/lib/action_dispatch/routing/route_set.rb index 20926012b4..af6f0de556 100644 --- a/actionpack/lib/action_dispatch/routing/route_set.rb +++ b/actionpack/lib/action_dispatch/routing/route_set.rb @@ -1,6 +1,5 @@  require 'action_dispatch/journey'  require 'forwardable' -require 'thread_safe'  require 'active_support/concern'  require 'active_support/core_ext/object/to_query'  require 'active_support/core_ext/hash/slice' @@ -23,49 +22,27 @@ module ActionDispatch        class Dispatcher < Routing::Endpoint          def initialize(raise_on_name_error)            @raise_on_name_error = raise_on_name_error -          @controller_class_names = ThreadSafe::Cache.new          end          def dispatcher?; true; end          def serve(req) -          req.check_path_parameters!            params = req.path_parameters - -          prepare_params!(params) - -          controller = controller(params, @raise_on_name_error) do +          controller = controller_reference(req) do              return [404, {'X-Cascade' => 'pass'}, []]            end -            dispatch(controller, params[:action], req) -        end - -        def prepare_params!(params) -          normalize_controller!(params) -          merge_default_action!(params) -        end - -        # If this is a default_controller (i.e. a controller specified by the user) -        # we should raise an error in case it's not found, because it usually means -        # a user error. However, if the controller was retrieved through a dynamic -        # segment, as in :controller(/:action), we should simply return nil and -        # delegate the control back to Rack cascade. Besides, if this is not a default -        # controller, it means we should respect the @scope[:module] parameter. -        def controller(params, raise_on_name_error=true) -          controller_reference params.fetch(:controller) { yield }          rescue NameError => e -          raise ActionController::RoutingError, e.message, e.backtrace if raise_on_name_error -          yield +          if @raise_on_name_error +            raise ActionController::RoutingError, e.message, e.backtrace +          else +            return [404, {'X-Cascade' => 'pass'}, []] +          end          end        protected - -        attr_reader :controller_class_names - -        def controller_reference(controller_param) -          const_name = controller_class_names[controller_param] ||= "#{controller_param.camelize}Controller" -          ActiveSupport::Dependencies.constantize(const_name) +        def controller_reference(req, &block) +          req.controller_class(&block)          end        private @@ -73,14 +50,6 @@ module ActionDispatch          def dispatch(controller, action, req)            controller.action(action).call(req.env)          end - -        def normalize_controller!(params) -          params[:controller] = params[:controller].underscore if params.key?(:controller) -        end - -        def merge_default_action!(params) -          params[:action] ||= 'index' -        end        end        # A NamedRouteCollection instance is a collection of named routes, and also @@ -760,14 +729,13 @@ module ActionDispatch            req.path_parameters = old_params.merge params            app = route.app            if app.matches?(req) && app.dispatcher? -            dispatcher = app.app - -            dispatcher.controller(params, false) do +            begin +              req.controller_class +            rescue NameError                raise ActionController::RoutingError, "A route matches #{path.inspect}, but references missing controller: #{params[:controller].camelize}Controller"              end -            dispatcher.prepare_params!(params) -            return params +            return req.path_parameters            end          end diff --git a/actionpack/lib/action_dispatch/testing/integration.rb b/actionpack/lib/action_dispatch/testing/integration.rb index 0cdc6d4e77..4dfd4f3f71 100644 --- a/actionpack/lib/action_dispatch/testing/integration.rb +++ b/actionpack/lib/action_dispatch/testing/integration.rb @@ -359,10 +359,10 @@ module ActionDispatch            # this modifies the passed request_env directly            if headers.present? -            Http::Headers.new(request_env).merge!(headers) +            Http::Headers.from_hash(request_env).merge!(headers)            end            if env.present? -            Http::Headers.new(request_env).merge!(env) +            Http::Headers.from_hash(request_env).merge!(env)            end            session = Rack::Test::Session.new(_mock_session) diff --git a/actionpack/test/abstract_unit.rb b/actionpack/test/abstract_unit.rb index 60e2cea8a2..fa0b6087ba 100644 --- a/actionpack/test/abstract_unit.rb +++ b/actionpack/test/abstract_unit.rb @@ -122,7 +122,7 @@ class ActionDispatch::IntegrationTest < ActiveSupport::TestCase    class StubDispatcher < ::ActionDispatch::Routing::RouteSet::Dispatcher      protected      def controller_reference(controller_param) -      controller_param +      controller_param.params[:controller]      end      def dispatch(controller, action, env) diff --git a/actionpack/test/dispatch/header_test.rb b/actionpack/test/dispatch/header_test.rb index e2b38c23bc..79600b654b 100644 --- a/actionpack/test/dispatch/header_test.rb +++ b/actionpack/test/dispatch/header_test.rb @@ -1,15 +1,19 @@  require "abstract_unit"  class HeaderTest < ActiveSupport::TestCase +  def make_headers(hash) +    ActionDispatch::Http::Headers.new ActionDispatch::Request.new hash +  end +    setup do -    @headers = ActionDispatch::Http::Headers.new( +    @headers = make_headers(        "CONTENT_TYPE" => "text/plain",        "HTTP_REFERER" => "/some/page"      )    end    test "#new does not normalize the data" do -    headers = ActionDispatch::Http::Headers.new( +    headers = make_headers(        "Content-Type" => "application/json",        "HTTP_REFERER" => "/some/page",        "Host" => "http://test.com") @@ -108,7 +112,7 @@ class HeaderTest < ActiveSupport::TestCase    end    test "env variables with . are not modified" do -    headers = ActionDispatch::Http::Headers.new +    headers = make_headers({})      headers.merge! "rack.input" => "",       "rack.request.cookie_hash" => "",       "action_dispatch.logger" => "" @@ -119,7 +123,7 @@ class HeaderTest < ActiveSupport::TestCase    end    test "symbols are treated as strings" do -    headers = ActionDispatch::Http::Headers.new +    headers = make_headers({})      headers.merge!(:SERVER_NAME => "example.com",                     "HTTP_REFERER" => "/",                     :Host => "test.com") @@ -130,7 +134,7 @@ class HeaderTest < ActiveSupport::TestCase    test "headers directly modifies the passed environment" do      env = {"HTTP_REFERER" => "/"} -    headers = ActionDispatch::Http::Headers.new(env) +    headers = make_headers(env)      headers['Referer'] = "http://example.com/"      headers.merge! "CONTENT_TYPE" => "text/plain"      assert_equal({"HTTP_REFERER"=>"http://example.com/", diff --git a/actionpack/test/dispatch/request/session_test.rb b/actionpack/test/dispatch/request/session_test.rb index 8bcc07dd04..410e3194e2 100644 --- a/actionpack/test/dispatch/request/session_test.rb +++ b/actionpack/test/dispatch/request/session_test.rb @@ -4,40 +4,42 @@ require 'action_dispatch/middleware/session/abstract_store'  module ActionDispatch    class Request      class SessionTest < ActiveSupport::TestCase +      attr_reader :req + +      def setup +        @req = ActionDispatch::Request.new({}) +      end +        def test_create_adds_itself_to_env -        env = {} -        s = Session.create(store, env, {}) -        assert_equal s, env[Rack::RACK_SESSION] +        s = Session.create(store, req, {}) +        assert_equal s, req.env[Rack::RACK_SESSION]        end        def test_to_hash -        env = {} -        s = Session.create(store, env, {}) +        s = Session.create(store, req, {})          s['foo'] = 'bar'          assert_equal 'bar', s['foo']          assert_equal({'foo' => 'bar'}, s.to_hash)        end        def test_create_merges_old -        env = {} -        s = Session.create(store, env, {}) +        s = Session.create(store, req, {})          s['foo'] = 'bar' -        s1 = Session.create(store, env, {}) +        s1 = Session.create(store, req, {})          assert_not_equal s, s1          assert_equal 'bar', s1['foo']        end        def test_find -        env = {} -        assert_nil Session.find(env) +        assert_nil Session.find(req) -        s = Session.create(store, env, {}) -        assert_equal s, Session.find(env) +        s = Session.create(store, req, {}) +        assert_equal s, Session.find(req)        end        def test_destroy -        s = Session.create(store, {}, {}) +        s = Session.create(store, req, {})          s['rails'] = 'ftw'          s.destroy @@ -46,21 +48,21 @@ module ActionDispatch        end        def test_keys -        s = Session.create(store, {}, {}) +        s = Session.create(store, req, {})          s['rails'] = 'ftw'          s['adequate'] = 'awesome'          assert_equal %w[rails adequate], s.keys        end        def test_values -        s = Session.create(store, {}, {}) +        s = Session.create(store, req, {})          s['rails'] = 'ftw'          s['adequate'] = 'awesome'          assert_equal %w[ftw awesome], s.values        end        def test_clear -        s = Session.create(store, {}, {}) +        s = Session.create(store, req, {})          s['rails'] = 'ftw'          s['adequate'] = 'awesome' @@ -69,7 +71,7 @@ module ActionDispatch        end        def test_update -        s = Session.create(store, {}, {}) +        s = Session.create(store, req, {})          s['rails'] = 'ftw'          s.update(:rails => 'awesome') @@ -79,7 +81,7 @@ module ActionDispatch        end        def test_delete -        s = Session.create(store, {}, {}) +        s = Session.create(store, req, {})          s['rails'] = 'ftw'          s.delete('rails') @@ -88,7 +90,7 @@ module ActionDispatch        end        def test_fetch -        session = Session.create(store, {}, {}) +        session = Session.create(store, req, {})          session['one'] = '1'          assert_equal '1', session.fetch(:one) diff --git a/actionpack/test/dispatch/session/abstract_store_test.rb b/actionpack/test/dispatch/session/abstract_store_test.rb index fe1a7b4f86..1c35144e6f 100644 --- a/actionpack/test/dispatch/session/abstract_store_test.rb +++ b/actionpack/test/dispatch/session/abstract_store_test.rb @@ -27,7 +27,7 @@ module ActionDispatch          as.call(env)          assert @env -        assert Request::Session.find @env +        assert Request::Session.find ActionDispatch::Request.new @env        end        def test_new_session_object_is_merged_with_old @@ -36,11 +36,11 @@ module ActionDispatch          as.call(env)          assert @env -        session = Request::Session.find @env +        session = Request::Session.find ActionDispatch::Request.new @env          session['foo'] = 'bar'          as.call(@env) -        session1 = Request::Session.find @env +        session1 = Request::Session.find ActionDispatch::Request.new @env          assert_not_equal session, session1          assert_equal session.to_hash, session1.to_hash diff --git a/actionview/test/abstract_unit.rb b/actionview/test/abstract_unit.rb index 35f520ed1c..12870acaa6 100644 --- a/actionview/test/abstract_unit.rb +++ b/actionview/test/abstract_unit.rb @@ -16,6 +16,7 @@ silence_warnings do  end  require 'active_support/testing/autorun' +require 'active_support/testing/method_call_assertions'  require 'action_controller'  require 'action_view'  require 'action_view/testing/resolvers' @@ -281,3 +282,6 @@ def jruby_skip(message = '')  end  require 'mocha/setup' # FIXME: stop using mocha +class ActiveSupport::TestCase +  include ActiveSupport::Testing::MethodCallAssertions +end diff --git a/actionview/test/template/asset_tag_helper_test.rb b/actionview/test/template/asset_tag_helper_test.rb index 01fc66bed6..496b33b35e 100644 --- a/actionview/test/template/asset_tag_helper_test.rb +++ b/actionview/test/template/asset_tag_helper_test.rb @@ -588,11 +588,13 @@ class AssetTagHelperTest < ActionView::TestCase        end      end -    @controller.request.stubs(:ssl?).returns(false) -    assert_equal "http://assets15.example.com/images/xml.png", image_path("xml.png") +    @controller.request.stub(:ssl?, false) do +      assert_equal "http://assets15.example.com/images/xml.png", image_path("xml.png") +    end -    @controller.request.stubs(:ssl?).returns(true) -    assert_equal "http://localhost/images/xml.png", image_path("xml.png") +    @controller.request.stub(:ssl?, true) do +      assert_equal "http://localhost/images/xml.png", image_path("xml.png") +    end    end  end diff --git a/actionview/test/template/form_helper_test.rb b/actionview/test/template/form_helper_test.rb index 37f2d7cad6..aef137935a 100644 --- a/actionview/test/template/form_helper_test.rb +++ b/actionview/test/template/form_helper_test.rb @@ -2003,21 +2003,22 @@ class FormHelperTest < ActionView::TestCase    def test_form_for_with_remote_without_html      @post.persisted = false -    @post.stubs(:to_key).returns(nil) -    form_for(@post, remote: true) do |f| -      concat f.text_field(:title) -      concat f.text_area(:body) -      concat f.check_box(:secret) -    end +    @post.stub(:to_key, nil) do +      form_for(@post, remote: true) do |f| +        concat f.text_field(:title) +        concat f.text_area(:body) +        concat f.check_box(:secret) +      end -    expected =  whole_form("/posts", "new_post", "new_post", remote: true) do -      "<input name='post[title]' type='text' id='post_title' value='Hello World' />" + -      "<textarea name='post[body]' id='post_body'>\nBack to the hill and over it again!</textarea>" + -      "<input name='post[secret]' type='hidden' value='0' />" + -      "<input name='post[secret]' checked='checked' type='checkbox' id='post_secret' value='1' />" -    end +      expected =  whole_form("/posts", "new_post", "new_post", remote: true) do +        "<input name='post[title]' type='text' id='post_title' value='Hello World' />" + +        "<textarea name='post[body]' id='post_body'>\nBack to the hill and over it again!</textarea>" + +        "<input name='post[secret]' type='hidden' value='0' />" + +        "<input name='post[secret]' checked='checked' type='checkbox' id='post_secret' value='1' />" +      end -    assert_dom_equal expected, output_buffer +      assert_dom_equal expected, output_buffer +    end    end    def test_form_for_without_object @@ -2220,16 +2221,17 @@ class FormHelperTest < ActionView::TestCase    def test_submit_with_object_as_new_record_and_locale_strings      with_locale :submit do        @post.persisted = false -      @post.stubs(:to_key).returns(nil) -      form_for(@post) do |f| -        concat f.submit -      end +      @post.stub(:to_key, nil) do +        form_for(@post) do |f| +          concat f.submit +        end -      expected = whole_form('/posts', 'new_post', 'new_post') do -        "<input name='commit' data-disable-with='Create Post' type='submit' value='Create Post' />" -      end +        expected = whole_form('/posts', 'new_post', 'new_post') do +          "<input name='commit' data-disable-with='Create Post' type='submit' value='Create Post' />" +        end -      assert_dom_equal expected, output_buffer +        assert_dom_equal expected, output_buffer +      end      end    end @@ -2807,11 +2809,12 @@ class FormHelperTest < ActionView::TestCase    def test_nested_fields_label_translation_with_more_than_10_records      @post.comments = Array.new(11) { |id| Comment.new(id + 1) } -    I18n.expects(:t).with('post.comments.body', default: [:"comment.body", ''], scope: "helpers.label").times(11).returns "Write body here" - -    form_for(@post) do |f| -      f.fields_for(:comments) do |cf| -        concat cf.label(:body) +    params = 11.times.map { ['post.comments.body', default: [:"comment.body", ''], scope: "helpers.label"] } +    assert_called_with(I18n, :t, params, returns: "Write body here") do +      form_for(@post) do |f| +        f.fields_for(:comments) do |cf| +          concat cf.label(:body) +        end        end      end    end diff --git a/actionview/test/template/form_options_helper_i18n_test.rb b/actionview/test/template/form_options_helper_i18n_test.rb index 4972ea6511..26ede09a5f 100644 --- a/actionview/test/template/form_options_helper_i18n_test.rb +++ b/actionview/test/template/form_options_helper_i18n_test.rb @@ -14,8 +14,9 @@ class FormOptionsHelperI18nTests < ActionView::TestCase    end    def test_select_with_prompt_true_translates_prompt_message -    I18n.expects(:translate).with('helpers.select.prompt', { :default => 'Please select' }) -    select('post', 'category', [], :prompt => true) +    assert_called_with(I18n, :translate, ['helpers.select.prompt', { :default => 'Please select' }]) do +      select('post', 'category', [], :prompt => true) +    end    end    def test_select_with_translated_prompt diff --git a/actionview/test/template/lookup_context_test.rb b/actionview/test/template/lookup_context_test.rb index 4f7823045e..14dafd7d63 100644 --- a/actionview/test/template/lookup_context_test.rb +++ b/actionview/test/template/lookup_context_test.rb @@ -108,10 +108,11 @@ class LookupContextTest < ActiveSupport::TestCase    end    test "found templates respects given formats if one cannot be found from template or handler" do -    ActionView::Template::Handlers::Builder.expects(:default_format).returns(nil) -    @lookup_context.formats = [:text] -    template = @lookup_context.find("hello", %w(test)) -    assert_equal [:text], template.formats +    assert_called(ActionView::Template::Handlers::Builder, :default_format, returns: nil) do +      @lookup_context.formats = [:text] +      template = @lookup_context.find("hello", %w(test)) +      assert_equal [:text], template.formats +    end    end    test "adds fallbacks to view paths when required" do @@ -210,45 +211,50 @@ end  class LookupContextWithFalseCaching < ActiveSupport::TestCase    def setup      @resolver = ActionView::FixtureResolver.new("test/_foo.erb" => ["Foo", Time.utc(2000)]) -    ActionView::Resolver.stubs(:caching?).returns(false)      @lookup_context = ActionView::LookupContext.new(@resolver, {})    end    test "templates are always found in the resolver but timestamp is checked before being compiled" do -    template = @lookup_context.find("foo", %w(test), true) -    assert_equal "Foo", template.source - -    # Now we are going to change the template, but it won't change the returned template -    # since the timestamp is the same. -    @resolver.hash["test/_foo.erb"][0] = "Bar" -    template = @lookup_context.find("foo", %w(test), true) -    assert_equal "Foo", template.source - -    # Now update the timestamp. -    @resolver.hash["test/_foo.erb"][1] = Time.now.utc -    template = @lookup_context.find("foo", %w(test), true) -    assert_equal "Bar", template.source +    ActionView::Resolver.stub(:caching?, false) do +      template = @lookup_context.find("foo", %w(test), true) +      assert_equal "Foo", template.source + +      # Now we are going to change the template, but it won't change the returned template +      # since the timestamp is the same. +      @resolver.hash["test/_foo.erb"][0] = "Bar" +      template = @lookup_context.find("foo", %w(test), true) +      assert_equal "Foo", template.source + +      # Now update the timestamp. +      @resolver.hash["test/_foo.erb"][1] = Time.now.utc +      template = @lookup_context.find("foo", %w(test), true) +      assert_equal "Bar", template.source +    end    end    test "if no template was found in the second lookup, with no cache, raise error" do -    template = @lookup_context.find("foo", %w(test), true) -    assert_equal "Foo", template.source +    ActionView::Resolver.stub(:caching?, false) do +      template = @lookup_context.find("foo", %w(test), true) +      assert_equal "Foo", template.source -    @resolver.hash.clear -    assert_raise ActionView::MissingTemplate do -      @lookup_context.find("foo", %w(test), true) +      @resolver.hash.clear +      assert_raise ActionView::MissingTemplate do +        @lookup_context.find("foo", %w(test), true) +      end      end    end    test "if no template was cached in the first lookup, retrieval should work in the second call" do -    @resolver.hash.clear -    assert_raise ActionView::MissingTemplate do -      @lookup_context.find("foo", %w(test), true) -    end +    ActionView::Resolver.stub(:caching?, false) do +      @resolver.hash.clear +      assert_raise ActionView::MissingTemplate do +        @lookup_context.find("foo", %w(test), true) +      end -    @resolver.hash["test/_foo.erb"] = ["Foo", Time.utc(2000)] -    template = @lookup_context.find("foo", %w(test), true) -    assert_equal "Foo", template.source +      @resolver.hash["test/_foo.erb"] = ["Foo", Time.utc(2000)] +      template = @lookup_context.find("foo", %w(test), true) +      assert_equal "Foo", template.source +    end    end  end diff --git a/actionview/test/template/template_test.rb b/actionview/test/template/template_test.rb index d3b51cd629..921011b073 100644 --- a/actionview/test/template/template_test.rb +++ b/actionview/test/template/template_test.rb @@ -118,15 +118,17 @@ class TestERBTemplate < ActiveSupport::TestCase    def test_refresh_with_templates      @template = new_template("Hello", :virtual_path => "test/foo/bar")      @template.locals = [:key] -    @context.lookup_context.expects(:find_template).with("bar", %w(test/foo), false, [:key]).returns("template") -    assert_equal "template", @template.refresh(@context) +    assert_called_with(@context.lookup_context, :find_template,["bar", %w(test/foo), false, [:key]], returns: "template") do +      assert_equal "template", @template.refresh(@context) +    end    end    def test_refresh_with_partials      @template = new_template("Hello", :virtual_path => "test/_foo")      @template.locals = [:key] -    @context.lookup_context.expects(:find_template).with("foo", %w(test), true, [:key]).returns("partial") -    assert_equal "partial", @template.refresh(@context) +    assert_called_with(@context.lookup_context, :find_template,[ "foo", %w(test), true, [:key]], returns: "partial") do +      assert_equal "partial", @template.refresh(@context) +    end    end    def test_refresh_raises_an_error_without_virtual_path diff --git a/actionview/test/template/test_case_test.rb b/actionview/test/template/test_case_test.rb index 80fbaa8392..b057d43ee0 100644 --- a/actionview/test/template/test_case_test.rb +++ b/actionview/test/template/test_case_test.rb @@ -20,6 +20,7 @@ module ActionView    class TestCase      helper ASharedTestHelper +    DeveloperStruct = Struct.new(:name)      module SharedTests        def self.included(test_case) @@ -50,7 +51,7 @@ module ActionView      end      test "works without testing a helper module" do -      assert_equal 'Eloy', render('developers/developer', :developer => stub(:name => 'Eloy')) +      assert_equal 'Eloy', render('developers/developer', :developer => DeveloperStruct.new('Eloy'))      end      test "can render a layout with block" do @@ -69,13 +70,15 @@ module ActionView      end      test "delegates notice to request.flash[:notice]" do -      view.request.flash.expects(:[]).with(:notice) -      view.notice +      assert_called_with(view.request.flash, :[], [:notice]) do +        view.notice +      end      end      test "delegates alert to request.flash[:alert]" do -      view.request.flash.expects(:[]).with(:alert) -      view.alert +      assert_called_with(view.request.flash, :[], [:alert]) do +        view.alert +      end      end      test "uses controller lookup context" do @@ -119,7 +122,7 @@ module ActionView      test "helper class that is being tested is always included in view instance" do        @controller.controller_path = 'test' -      @customers = [stub(:name => 'Eloy'), stub(:name => 'Manfred')] +      @customers = [DeveloperStruct.new('Eloy'), DeveloperStruct.new('Manfred')]        assert_match(/Hello: EloyHello: Manfred/, render(:partial => 'test/from_helper'))      end    end @@ -255,15 +258,15 @@ module ActionView      end      test "is able to render partials with local variables" do -      assert_equal 'Eloy', render('developers/developer', :developer => stub(:name => 'Eloy')) +      assert_equal 'Eloy', render('developers/developer', :developer => DeveloperStruct.new('Eloy'))        assert_equal 'Eloy', render(:partial => 'developers/developer', -                                  :locals => { :developer => stub(:name => 'Eloy') }) +                                  :locals => { :developer => DeveloperStruct.new('Eloy') })      end      test "is able to render partials from templates and also use instance variables" do        @controller.controller_path = "test" -      @customers = [stub(:name => 'Eloy'), stub(:name => 'Manfred')] +      @customers = [DeveloperStruct.new('Eloy'), DeveloperStruct.new('Manfred')]        assert_match(/Hello: EloyHello: Manfred/, render(:file => 'test/list'))      end @@ -272,7 +275,7 @@ module ActionView        view -      @customers = [stub(:name => 'Eloy'), stub(:name => 'Manfred')] +      @customers = [DeveloperStruct.new('Eloy'), DeveloperStruct.new('Manfred')]        assert_match(/Hello: EloyHello: Manfred/, render(:file => 'test/list'))      end diff --git a/actionview/test/template/translation_helper_test.rb b/actionview/test/template/translation_helper_test.rb index 5dc281adb2..749d0dd7fd 100644 --- a/actionview/test/template/translation_helper_test.rb +++ b/actionview/test/template/translation_helper_test.rb @@ -42,14 +42,16 @@ class TranslationHelperTest < ActiveSupport::TestCase    end    def test_delegates_setting_to_i18n -    I18n.expects(:translate).with(:foo, :locale => 'en', :raise => true).returns("") -    translate :foo, :locale => 'en' +    assert_called_with(I18n, :translate, [:foo, :locale => 'en', :raise => true], returns: "") do +      translate :foo, :locale => 'en' +    end    end    def test_delegates_localize_to_i18n      @time = Time.utc(2008, 7, 8, 12, 18, 38) -    I18n.expects(:localize).with(@time) -    localize @time +    assert_called_with(I18n, :localize, [@time]) do +      localize @time +    end    end    def test_returns_missing_translation_message_wrapped_into_span @@ -125,8 +127,9 @@ class TranslationHelperTest < ActiveSupport::TestCase    end    def test_translate_escapes_interpolations_in_translations_with_a_html_suffix +    word_struct = Struct.new(:to_s)      assert_equal '<a>Hello <World></a>', translate(:'translations.interpolated_html', :word => '<World>') -    assert_equal '<a>Hello <World></a>', translate(:'translations.interpolated_html', :word => stub(:to_s => "<World>")) +    assert_equal '<a>Hello <World></a>', translate(:'translations.interpolated_html', :word => word_struct.new("<World>"))    end    def test_translate_with_html_count diff --git a/activerecord/lib/active_record/associations/collection_proxy.rb b/activerecord/lib/active_record/associations/collection_proxy.rb index b5a8c81fe4..29c711082a 100644 --- a/activerecord/lib/active_record/associations/collection_proxy.rb +++ b/activerecord/lib/active_record/associations/collection_proxy.rb @@ -227,6 +227,31 @@ module ActiveRecord          @association.last(*args)        end +      # Gives a record (or N records if a parameter is supplied) from the collection +      # using the same rules as <tt>ActiveRecord::Base.take</tt>. +      # +      #   class Person < ActiveRecord::Base +      #     has_many :pets +      #   end +      # +      #   person.pets +      #   # => [ +      #   #       #<Pet id: 1, name: "Fancy-Fancy", person_id: 1>, +      #   #       #<Pet id: 2, name: "Spook", person_id: 1>, +      #   #       #<Pet id: 3, name: "Choo-Choo", person_id: 1> +      #   #    ] +      # +      #   person.pets.take # => #<Pet id: 1, name: "Fancy-Fancy", person_id: 1> +      # +      #   person.pets.take(2) +      #   # => [ +      #   #      #<Pet id: 1, name: "Fancy-Fancy", person_id: 1>, +      #   #      #<Pet id: 2, name: "Spook", person_id: 1> +      #   #    ] +      # +      #   another_person_without.pets         # => [] +      #   another_person_without.pets.take    # => nil +      #   another_person_without.pets.take(2) # => []        def take(n = nil)          @association.take(n)        end diff --git a/activerecord/lib/active_record/connection_adapters/abstract_mysql_adapter.rb b/activerecord/lib/active_record/connection_adapters/abstract_mysql_adapter.rb index 96ea866580..b3e55a0b90 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract_mysql_adapter.rb @@ -1050,7 +1050,7 @@ module ActiveRecord          end        end -      class MysqlJson < Type::Json # :nodoc: +      class MysqlJson < Type::Internal::AbstractJson # :nodoc:          def changed_in_place?(raw_old_value, new_value)            # Normalization is required because MySQL JSON data format includes            # the space between the elements. diff --git a/activerecord/lib/active_record/connection_adapters/postgresql/oid.rb b/activerecord/lib/active_record/connection_adapters/postgresql/oid.rb index 37226831dc..68752cdd80 100644 --- a/activerecord/lib/active_record/connection_adapters/postgresql/oid.rb +++ b/activerecord/lib/active_record/connection_adapters/postgresql/oid.rb @@ -8,6 +8,7 @@ require 'active_record/connection_adapters/postgresql/oid/decimal'  require 'active_record/connection_adapters/postgresql/oid/enum'  require 'active_record/connection_adapters/postgresql/oid/hstore'  require 'active_record/connection_adapters/postgresql/oid/inet' +require 'active_record/connection_adapters/postgresql/oid/json'  require 'active_record/connection_adapters/postgresql/oid/jsonb'  require 'active_record/connection_adapters/postgresql/oid/money'  require 'active_record/connection_adapters/postgresql/oid/point' diff --git a/activerecord/lib/active_record/connection_adapters/postgresql/oid/json.rb b/activerecord/lib/active_record/connection_adapters/postgresql/oid/json.rb new file mode 100644 index 0000000000..dbc879ffd4 --- /dev/null +++ b/activerecord/lib/active_record/connection_adapters/postgresql/oid/json.rb @@ -0,0 +1,10 @@ +module ActiveRecord +  module ConnectionAdapters +    module PostgreSQL +      module OID # :nodoc: +        class Json < Type::Internal::AbstractJson +        end +      end +    end +  end +end diff --git a/activerecord/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb b/activerecord/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb index 1f6d63582c..87391b5dc7 100644 --- a/activerecord/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +++ b/activerecord/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb @@ -2,7 +2,7 @@ module ActiveRecord    module ConnectionAdapters      module PostgreSQL        module OID # :nodoc: -        class Jsonb < Type::Json # :nodoc: +        class Jsonb < Json # :nodoc:            def type              :jsonb            end diff --git a/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb b/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb index 861edbf3a2..27291bd2ea 100644 --- a/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb @@ -482,7 +482,7 @@ module ActiveRecord            m.register_type 'bytea', OID::Bytea.new            m.register_type 'point', OID::Point.new            m.register_type 'hstore', OID::Hstore.new -          m.register_type 'json', Type::Json.new +          m.register_type 'json', OID::Json.new            m.register_type 'jsonb', OID::Jsonb.new            m.register_type 'cidr', OID::Cidr.new            m.register_type 'inet', OID::Inet.new @@ -838,6 +838,7 @@ module ActiveRecord          ActiveRecord::Type.register(:enum, OID::Enum, adapter: :postgresql)          ActiveRecord::Type.register(:hstore, OID::Hstore, adapter: :postgresql)          ActiveRecord::Type.register(:inet, OID::Inet, adapter: :postgresql) +        ActiveRecord::Type.register(:json, OID::Json, adapter: :postgresql)          ActiveRecord::Type.register(:jsonb, OID::Jsonb, adapter: :postgresql)          ActiveRecord::Type.register(:money, OID::Money, adapter: :postgresql)          ActiveRecord::Type.register(:point, OID::Point, adapter: :postgresql) diff --git a/activerecord/lib/active_record/migration.rb b/activerecord/lib/active_record/migration.rb index c35efbdee8..04c6124eef 100644 --- a/activerecord/lib/active_record/migration.rb +++ b/activerecord/lib/active_record/migration.rb @@ -814,7 +814,7 @@ module ActiveRecord          new(:up, migrations, target_version).migrate        end -      def down(migrations_paths, target_version = nil, &block) +      def down(migrations_paths, target_version = nil)          migrations = migrations(migrations_paths)          migrations.select! { |m| yield m } if block_given? diff --git a/activerecord/lib/active_record/type.rb b/activerecord/lib/active_record/type.rb index 8f56a37f3c..53f3b53bec 100644 --- a/activerecord/lib/active_record/type.rb +++ b/activerecord/lib/active_record/type.rb @@ -10,7 +10,6 @@ require 'active_record/type/decimal'  require 'active_record/type/decimal_without_scale'  require 'active_record/type/float'  require 'active_record/type/integer' -require 'active_record/type/json'  require 'active_record/type/serialized'  require 'active_record/type/string'  require 'active_record/type/text' @@ -21,6 +20,8 @@ require 'active_record/type/adapter_specific_registry'  require 'active_record/type/type_map'  require 'active_record/type/hash_lookup_type_map' +require 'active_record/type/internal/abstract_json' +  module ActiveRecord    module Type      @registry = AdapterSpecificRegistry.new @@ -60,7 +61,6 @@ module ActiveRecord      register(:decimal, Type::Decimal, override: false)      register(:float, Type::Float, override: false)      register(:integer, Type::Integer, override: false) -    register(:json, Type::Json, override: false)      register(:string, Type::String, override: false)      register(:text, Type::Text, override: false)      register(:time, Type::Time, override: false) diff --git a/activerecord/lib/active_record/type/internal/abstract_json.rb b/activerecord/lib/active_record/type/internal/abstract_json.rb new file mode 100644 index 0000000000..963a8245d0 --- /dev/null +++ b/activerecord/lib/active_record/type/internal/abstract_json.rb @@ -0,0 +1,33 @@ +module ActiveRecord +  module Type +    module Internal # :nodoc: +      class AbstractJson < Type::Value # :nodoc: +        include Type::Helpers::Mutable + +        def type +          :json +        end + +        def deserialize(value) +          if value.is_a?(::String) +            ::ActiveSupport::JSON.decode(value) rescue nil +          else +            value +          end +        end + +        def serialize(value) +          if value.is_a?(::Array) || value.is_a?(::Hash) +            ::ActiveSupport::JSON.encode(value) +          else +            value +          end +        end + +        def accessor +          ActiveRecord::Store::StringKeyedHashAccessor +        end +      end +    end +  end +end diff --git a/activerecord/lib/active_record/type/json.rb b/activerecord/lib/active_record/type/json.rb deleted file mode 100644 index 1728bd3a8e..0000000000 --- a/activerecord/lib/active_record/type/json.rb +++ /dev/null @@ -1,31 +0,0 @@ -module ActiveRecord -  module Type -    class Json < Type::Value # :nodoc: -      include Type::Helpers::Mutable - -      def type -        :json -      end - -      def deserialize(value) -        if value.is_a?(::String) -          ::ActiveSupport::JSON.decode(value) rescue nil -        else -          value -        end -      end - -      def serialize(value) -        if value.is_a?(::Array) || value.is_a?(::Hash) -          ::ActiveSupport::JSON.encode(value) -        else -          value -        end -      end - -      def accessor -        ActiveRecord::Store::StringKeyedHashAccessor -      end -    end -  end -end diff --git a/activerecord/test/cases/migration_test.rb b/activerecord/test/cases/migration_test.rb index b2f209fe97..11e42f28dc 100644 --- a/activerecord/test/cases/migration_test.rb +++ b/activerecord/test/cases/migration_test.rb @@ -132,13 +132,9 @@ class MigrationTest < ActiveRecord::TestCase      Person.connection.drop_table :testings2, if_exists: true    end -  def connection -    ActiveRecord::Base.connection -  end -    def test_migration_instance_has_connection      migration = Class.new(ActiveRecord::Migration).new -    assert_equal connection, migration.connection +    assert_equal ActiveRecord::Base.connection, migration.connection    end    def test_method_missing_delegates_to_connection diff --git a/activesupport/lib/active_support/log_subscriber/test_helper.rb b/activesupport/lib/active_support/log_subscriber/test_helper.rb index 70ac4a4d5c..cbc20c103d 100644 --- a/activesupport/lib/active_support/log_subscriber/test_helper.rb +++ b/activesupport/lib/active_support/log_subscriber/test_helper.rb @@ -11,6 +11,7 @@ module ActiveSupport      #     include ActiveSupport::LogSubscriber::TestHelper      #      #     def setup +    #       super      #       ActiveRecord::LogSubscriber.attach_to(:active_record)      #     end      # diff --git a/guides/source/4_2_release_notes.md b/guides/source/4_2_release_notes.md index 684bd286bc..8a59007420 100644 --- a/guides/source/4_2_release_notes.md +++ b/guides/source/4_2_release_notes.md @@ -227,6 +227,17 @@ restore the old behavior.  If you do this, be sure to configure your firewall properly such that only  trusted machines on your network can access your development server. +### Changed status option symbols for `render` + +Due to a [change in Rack](https://github.com/rack/rack/commit/be28c6a2ac152fe4adfbef71f3db9f4200df89e8), the symbols that the `render` method accepts for the `:status` option have changed: + +- 306: `:reserved` has been removed. +- 413: `:request_entity_too_large` has been renamed to `:payload_too_large`. +- 414: `:request_uri_too_long` has been renamed to `:uri_too_long`. +- 416: `:requested_range_not_satisfiable` has been renamed to `:range_not_satisfiable`. + +Keep in mind that if calling `render` with an unknown symbol, the response status will default to 500. +  ### HTML Sanitizer  The HTML sanitizer has been replaced with a new, more robust, implementation diff --git a/guides/source/action_controller_overview.md b/guides/source/action_controller_overview.md index 09fbdc0d32..d2173c39f6 100644 --- a/guides/source/action_controller_overview.md +++ b/guides/source/action_controller_overview.md @@ -1174,7 +1174,7 @@ end  WARNING: You shouldn't do `rescue_from Exception` or `rescue_from StandardError` unless you have a particular reason as it will cause serious side-effects (e.g. you won't be able to see exception details and tracebacks during development). -NOTE: Certain exceptions are only rescuable from the `ApplicationController` class, as they are raised before the controller gets initialized and the action gets executed. See Pratik Naik's [article](http://m.onkey.org/2008/7/20/rescue-from-dispatching) on the subject for more information. +NOTE: Certain exceptions are only rescuable from the `ApplicationController` class, as they are raised before the controller gets initialized and the action gets executed.   Force HTTPS protocol  -------------------- diff --git a/guides/source/action_view_overview.md b/guides/source/action_view_overview.md index db7eeed19a..00c41a480e 100644 --- a/guides/source/action_view_overview.md +++ b/guides/source/action_view_overview.md @@ -1443,12 +1443,12 @@ Sanitizes a block of CSS code.  Strips all link tags from text leaving just the link text.  ```ruby -strip_links("<a href="http://rubyonrails.org">Ruby on Rails</a>") +strip_links('<a href="http://rubyonrails.org">Ruby on Rails</a>')  # => Ruby on Rails  ```  ```ruby -strip_links("emails to <a href="mailto:me@email.com">me@email.com</a>.") +strip_links('emails to <a href="mailto:me@email.com">me@email.com</a>.')  # => emails to me@email.com.  ``` diff --git a/guides/source/active_record_querying.md b/guides/source/active_record_querying.md index 4b4d70d3ce..c0a4a0ba39 100644 --- a/guides/source/active_record_querying.md +++ b/guides/source/active_record_querying.md @@ -341,8 +341,6 @@ User.find_each(begin_at: 2000, batch_size: 5000) do |user|  end  ``` -Another example would be if you wanted multiple workers handling the same processing queue. You could have each worker handle 10000 records by setting the appropriate `:begin_at` option on each worker. -  **`:end_at`**  Similar to the `:begin_at` option, `:end_at` allows you to configure the last ID of the sequence whenever the highest ID is not the one you need. @@ -356,6 +354,10 @@ User.find_each(begin_at: 2000, end_at: 10000, batch_size: 5000) do |user|  end  ``` +Another example would be if you wanted multiple workers handling the same +processing queue. You could have each worker handle 10000 records by setting the +appropriate `:begin_at` and `:end_at` options on each worker. +  #### `find_in_batches`  The `find_in_batches` method is similar to `find_each`, since both retrieve batches of records. The difference is that `find_in_batches` yields _batches_ to the block as an array of models, instead of individually. The following example will yield to the supplied block an array of up to 1000 invoices at a time, with the final block containing any remaining invoices: diff --git a/guides/source/api_app.md b/guides/source/api_app.md index 29ca872254..1652d91086 100644 --- a/guides/source/api_app.md +++ b/guides/source/api_app.md @@ -367,7 +367,7 @@ controller modules by default:    methods returning `ActionDispatch::Request` and `ActionDispatch::Response`    objects.  - `ActionController::DataStreaming`: Support for `send_file` and `send_data`. -- `AbstractController::Callbacks`: Support for `before_filter` and friends. +- `AbstractController::Callbacks`: Support for `before_action` and friends.  - `ActionController::Instrumentation`: Support for the instrumentation    hooks defined by Action Controller (see [the instrumentation    guide](active_support_instrumentation.html#action-controller)). diff --git a/guides/source/api_documentation_guidelines.md b/guides/source/api_documentation_guidelines.md index 46c9013087..a4feff798d 100644 --- a/guides/source/api_documentation_guidelines.md +++ b/guides/source/api_documentation_guidelines.md @@ -84,6 +84,11 @@ English  Please use American English (*color*, *center*, *modularize*, etc). See [a list of American and British English spelling differences here](http://en.wikipedia.org/wiki/American_and_British_English_spelling_differences). +Comma +------- + +Please use the Oxford comma (*red, white, and blue* style). See [the detail of Oxford comma](http://en.wikipedia.org/wiki/Serial_comma). +  Example Code  ------------ diff --git a/guides/source/debugging_rails_applications.md b/guides/source/debugging_rails_applications.md index 76fdbb2f72..fe8757fbd5 100644 --- a/guides/source/debugging_rails_applications.md +++ b/guides/source/debugging_rails_applications.md @@ -620,13 +620,15 @@ Processing by ArticlesController#index as HTML  (byebug)  ``` -If we use `next`, we want go deep inside method calls. Instead, byebug will go -to the next line within the same context. In this case, this is the last line of -the method, so `byebug` will jump to next next line of the previous frame. - +If we use `next`, we won't go deep inside method calls. Instead, `byebug` will +go to the next line within the same context. In this case, it is the last line +of the current method, so `byebug` will return to the next line of the caller ++method.  ```  (byebug) next -Next went up a frame because previous frame finished + +Next advances to the next line (line 6: `end`), which returns to the next line +of the caller method:  [4, 13] in /PathTo/project/test_app/app/controllers/articles_controller.rb      4:   # GET /articles @@ -643,8 +645,8 @@ Next went up a frame because previous frame finished  (byebug)  ``` -If we use `step` in the same situation, we will literally go to the next Ruby -instruction to be executed. In this case, Active Support's `week` method. +If we use `step` in the same situation, `byebug` will literally go to the next +Ruby instruction to be executed -- in this case, Active Support's `week` method.  ```  (byebug) step diff --git a/guides/source/layouts_and_rendering.md b/guides/source/layouts_and_rendering.md index 94cd7297e2..b425eb126a 100644 --- a/guides/source/layouts_and_rendering.md +++ b/guides/source/layouts_and_rendering.md @@ -360,7 +360,6 @@ Rails understands both numeric status codes and the corresponding symbols shown  |                     | 303              | :see_other                       |  |                     | 304              | :not_modified                    |  |                     | 305              | :use_proxy                       | -|                     | 306              | :reserved                        |  |                     | 307              | :temporary_redirect              |  |                     | 308              | :permanent_redirect              |  | **Client Error**    | 400              | :bad_request                     | @@ -376,10 +375,10 @@ Rails understands both numeric status codes and the corresponding symbols shown  |                     | 410              | :gone                            |  |                     | 411              | :length_required                 |  |                     | 412              | :precondition_failed             | -|                     | 413              | :request_entity_too_large        | -|                     | 414              | :request_uri_too_long            | +|                     | 413              | :payload_too_large               | +|                     | 414              | :uri_too_long                    |  |                     | 415              | :unsupported_media_type          | -|                     | 416              | :requested_range_not_satisfiable | +|                     | 416              | :range_not_satisfiable           |  |                     | 417              | :expectation_failed              |  |                     | 422              | :unprocessable_entity            |  |                     | 423              | :locked                          | diff --git a/guides/source/security.md b/guides/source/security.md index 850d111bd7..93c270064a 100644 --- a/guides/source/security.md +++ b/guides/source/security.md @@ -760,7 +760,7 @@ s = sanitize(user_input, tags: tags, attributes: %w(href title))  This allows only the given tags and does a good job, even against all kinds of tricks and malformed tags. -As a second step, _it is good practice to escape all output of the application_, especially when re-displaying user input, which hasn't been input-filtered (as in the search form example earlier on). _Use `escapeHTML()` (or its alias `h()`) method_ to replace the HTML input characters &, ", <, > by their uninterpreted representations in HTML (`&`, `"`, `<`;, and `>`). However, it can easily happen that the programmer forgets to use it, so _it is recommended to use the SafeErb gem. SafeErb reminds you to escape strings from external sources. +As a second step, _it is good practice to escape all output of the application_, especially when re-displaying user input, which hasn't been input-filtered (as in the search form example earlier on). _Use `escapeHTML()` (or its alias `h()`) method_ to replace the HTML input characters &, ", <, and > by their uninterpreted representations in HTML (`&`, `"`, `<`, and `>`). However, it can easily happen that the programmer forgets to use it, so _it is recommended to use the SafeErb gem. SafeErb reminds you to escape strings from external sources.  ##### Obfuscation and Encoding Injection | 
