request.rb 24.9 KB
Newer Older
1 2 3 4
require 'tempfile'
require 'stringio'
require 'strscan'

D
Initial  
David Heinemeier Hansson 已提交
5
module ActionController
6
  # HTTP methods which are accepted by default. 
7
  ACCEPTED_HTTP_METHODS = Set.new(%w( get head put post delete options ))
8

9
  # CgiRequest and TestRequest provide concrete implementations.
D
Initial  
David Heinemeier Hansson 已提交
10
  class AbstractRequest
11
    cattr_accessor :relative_url_root
12
    remove_method :relative_url_root
13

14
    # The hash of environment variables for this request,
15 16 17
    # such as { 'RAILS_ENV' => 'production' }.
    attr_reader :env

18
    # The true HTTP request method as a lowercase symbol, such as <tt>:get</tt>.
19 20 21 22 23 24 25 26 27 28 29 30
    # UnknownHttpMethod is raised for invalid methods not listed in ACCEPTED_HTTP_METHODS.
    def request_method
      @request_method ||= begin
        method = ((@env['REQUEST_METHOD'] == 'POST' && !parameters[:_method].blank?) ? parameters[:_method].to_s : @env['REQUEST_METHOD']).downcase
        if ACCEPTED_HTTP_METHODS.include?(method)
          method.to_sym
        else
          raise UnknownHttpMethod, "#{method}, accepted HTTP methods are #{ACCEPTED_HTTP_METHODS.to_a.to_sentence}"
        end
      end
    end

31 32
    # The HTTP request method as a lowercase symbol, such as <tt>:get</tt>.
    # Note, HEAD is returned as <tt>:get</tt> since the two are functionally
33
    # equivalent from the application's perspective.
D
Initial  
David Heinemeier Hansson 已提交
34
    def method
35
      request_method == :head ? :get : request_method
D
Initial  
David Heinemeier Hansson 已提交
36 37
    end

38
    # Is this a GET (or HEAD) request?  Equivalent to <tt>request.method == :get</tt>.
D
Initial  
David Heinemeier Hansson 已提交
39 40 41 42
    def get?
      method == :get
    end

43
    # Is this a POST request?  Equivalent to <tt>request.method == :post</tt>.
D
Initial  
David Heinemeier Hansson 已提交
44
    def post?
45
      request_method == :post
D
Initial  
David Heinemeier Hansson 已提交
46 47
    end

48
    # Is this a PUT request?  Equivalent to <tt>request.method == :put</tt>.
D
Initial  
David Heinemeier Hansson 已提交
49
    def put?
50
      request_method == :put
D
Initial  
David Heinemeier Hansson 已提交
51 52
    end

53
    # Is this a DELETE request?  Equivalent to <tt>request.method == :delete</tt>.
D
Initial  
David Heinemeier Hansson 已提交
54
    def delete?
55
      request_method == :delete
D
Initial  
David Heinemeier Hansson 已提交
56 57
    end

58 59
    # Is this a HEAD request? <tt>request.method</tt> sees HEAD as <tt>:get</tt>,
    # so check the HTTP method directly.
60
    def head?
61
      request_method == :head
62
    end
63

64 65
    # Provides acccess to the request's HTTP headers, for example:
    #  request.headers["Content-Type"] # => "text/plain"
66
    def headers
67
      @headers ||= ActionController::Http::Headers.new(@env)
68 69
    end

70 71 72 73
    def content_length
      @content_length ||= env['CONTENT_LENGTH'].to_i
    end

74
    # The MIME type of the HTTP request, such as Mime::XML.
75 76 77
    #
    # For backward compatibility, the post format is extracted from the
    # X-Post-Data-Format HTTP header if present.
78
    def content_type
79
      @content_type ||= Mime::Type.lookup(content_type_without_parameters)
80 81
    end

82
    # Returns the accepted MIME type for the request
83
    def accepts
84 85
      @accepts ||=
        if @env['HTTP_ACCEPT'].to_s.strip.empty?
86
          [ content_type, Mime::ALL ].compact # make sure content_type being nil is not included
87 88 89
        else
          Mime::Type.parse(@env['HTTP_ACCEPT'])
        end
90
    end
91

92
    # Returns the Mime type for the format used in the request.
93 94 95
    #
    #   GET /posts/5.xml   | request.format => Mime::XML
    #   GET /posts/5.xhtml | request.format => Mime::HTML
96
    #   GET /posts/5       | request.format => Mime::HTML or MIME::JS, or request.accepts.first depending on the value of <tt>ActionController::Base.use_accept_header</tt>
97
    def format
98 99 100 101 102 103 104 105 106 107 108
      @format ||= begin
        if parameters[:format]
          Mime::Type.lookup_by_extension(parameters[:format])
        elsif ActionController::Base.use_accept_header
          accepts.first
        elsif xhr?
          Mime::Type.lookup_by_extension("js")
        else
          Mime::Type.lookup_by_extension("html")
        end
      end
109
    end
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
    
    
    # Sets the format by string extension, which can be used to force custom formats that are not controlled by the extension.
    # Example:
    #
    #   class ApplicationController < ActionController::Base
    #     before_filter :adjust_format_for_iphone
    #   
    #     private
    #       def adjust_format_for_iphone
    #         request.format = :iphone if request.env["HTTP_USER_AGENT"][/iPhone/]
    #       end
    #   end
    def format=(extension)
      parameters[:format] = extension.to_s
125
      @format = Mime::Type.lookup_by_extension(parameters[:format])
126
    end
127

128 129 130
    # Returns a symbolized version of the <tt>:format</tt> parameter of the request.
    # If no format is given it returns <tt>:js</tt>for AJAX requests and <tt>:html</tt>
    # otherwise.
131 132 133
    def template_format
      parameter_format = parameters[:format]

134 135 136
      if parameter_format
        parameter_format.to_sym
      elsif xhr?
137 138
        :js
      else
139
        :html
140 141 142
      end
    end

143 144 145 146 147
    def cache_format
      parameter_format = parameters[:format]
      parameter_format && parameter_format.to_sym
    end

148 149 150
    # Returns true if the request's "X-Requested-With" header contains
    # "XMLHttpRequest". (The Prototype Javascript library sends this header with
    # every Ajax request.)
151
    def xml_http_request?
152
      !(@env['HTTP_X_REQUESTED_WITH'] !~ /XMLHttpRequest/i)
153 154 155
    end
    alias xhr? :xml_http_request?

J
Jeremy Kemper 已提交
156 157 158 159
    # Which IP addresses are "trusted proxies" that can be stripped from
    # the right-hand-side of X-Forwarded-For
    TRUSTED_PROXIES = /^127\.0\.0\.1$|^(10|172\.(1[6-9]|2[0-9]|30|31)|192\.168)\./i

D
Initial  
David Heinemeier Hansson 已提交
160 161
    # Determine originating IP address.  REMOTE_ADDR is the standard
    # but will fail if the user is behind a proxy.  HTTP_CLIENT_IP and/or
J
Jeremy Kemper 已提交
162 163 164 165
    # HTTP_X_FORWARDED_FOR are set by proxies so check for these if
    # REMOTE_ADDR is a proxy.  HTTP_X_FORWARDED_FOR may be a comma-
    # delimited list in the case of multiple chained proxies; the last
    # address which is not trusted is the originating IP.
D
Initial  
David Heinemeier Hansson 已提交
166
    def remote_ip
J
Jeremy Kemper 已提交
167 168 169 170
      if TRUSTED_PROXIES !~ @env['REMOTE_ADDR']
        return @env['REMOTE_ADDR']
      end

171 172
      remote_ips = @env['HTTP_X_FORWARDED_FOR'] && @env['HTTP_X_FORWARDED_FOR'].split(',')

J
Jeremy Kemper 已提交
173
      if @env.include? 'HTTP_CLIENT_IP'
174
        if remote_ips && !remote_ips.include?(@env['HTTP_CLIENT_IP'])
J
Jeremy Kemper 已提交
175 176 177 178 179 180 181
          # We don't know which came from the proxy, and which from the user
          raise ActionControllerError.new(<<EOM)
IP spoofing attack?!
HTTP_CLIENT_IP=#{@env['HTTP_CLIENT_IP'].inspect}
HTTP_X_FORWARDED_FOR=#{@env['HTTP_X_FORWARDED_FOR'].inspect}
EOM
        end
182

J
Jeremy Kemper 已提交
183 184
        return @env['HTTP_CLIENT_IP']
      end
185

186
      if remote_ips
J
Jeremy Kemper 已提交
187 188
        while remote_ips.size > 1 && TRUSTED_PROXIES =~ remote_ips.last.strip
          remote_ips.pop
189 190
        end

J
Jeremy Kemper 已提交
191
        return remote_ips.last.strip
D
Initial  
David Heinemeier Hansson 已提交
192
      end
193

194
      @env['REMOTE_ADDR']
D
Initial  
David Heinemeier Hansson 已提交
195 196
    end

197 198 199 200 201 202 203 204
    # Returns the lowercase name of the HTTP server software.
    def server_software
      (@env['SERVER_SOFTWARE'] && /^([a-zA-Z]+)/ =~ @env['SERVER_SOFTWARE']) ? $1.downcase : nil
    end


    # Returns the complete URL used for this request
    def url
D
David Heinemeier Hansson 已提交
205
      protocol + host_with_port + request_uri
206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224
    end

    # Return 'https://' if this is an SSL request and 'http://' otherwise.
    def protocol
      ssl? ? 'https://' : 'http://'
    end

    # Is this an SSL request?
    def ssl?
      @env['HTTPS'] == 'on' || @env['HTTP_X_FORWARDED_PROTO'] == 'https'
    end

    # Returns the host for this request, such as example.com.
    def host
    end

    # Returns a host:port string for this request, such as example.com or
    # example.com:8080.
    def host_with_port
225
      @host_with_port ||= host + port_string
226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246
    end

    # Returns the port number of this request as an integer.
    def port
      @port_as_int ||= @env['SERVER_PORT'].to_i
    end

    # Returns the standard port number for this request's protocol
    def standard_port
      case protocol
        when 'https://' then 443
        else 80
      end
    end

    # Returns a port suffix like ":8080" if the port number of this request
    # is not the default HTTP port 80 or HTTPS port 443.
    def port_string
      (port == standard_port) ? '' : ":#{port}"
    end

247 248 249
    # Returns the domain part of a host, such as rubyonrails.org in "www.rubyonrails.org". You can specify
    # a different <tt>tld_length</tt>, such as 2 to catch rubyonrails.co.uk in "www.rubyonrails.co.uk".
    def domain(tld_length = 1)
250
      return nil unless named_host?(host)
251

252
      host.split('.').last(1 + tld_length).join('.')
253 254 255 256 257 258
    end

    # Returns all the subdomains as an array, so ["dev", "www"] would be returned for "dev.www.rubyonrails.org".
    # You can specify a different <tt>tld_length</tt>, such as 2 to catch ["www"] instead of ["www", "rubyonrails"]
    # in "www.rubyonrails.co.uk".
    def subdomains(tld_length = 1)
259
      return [] unless named_host?(host)
260
      parts = host.split('.')
261
      parts[0..-(tld_length+2)]
262 263
    end

264 265 266 267 268 269 270 271 272
    # Return the query string, accounting for server idiosyncracies.
    def query_string
      if uri = @env['REQUEST_URI']
        uri.split('?', 2)[1] || ''
      else
        @env['QUERY_STRING'] || ''
      end
    end

273 274
    # Return the request URI, accounting for server idiosyncracies.
    # WEBrick includes the full URL. IIS leaves REQUEST_URI blank.
D
Initial  
David Heinemeier Hansson 已提交
275
    def request_uri
276
      if uri = @env['REQUEST_URI']
277 278 279 280
        # Remove domain, which webrick puts into the request_uri.
        (%r{^\w+\://[^/]+(/.*|$)$} =~ uri) ? $1 : uri
      else
        # Construct IIS missing REQUEST_URI from SCRIPT_NAME and PATH_INFO.
281 282
        script_filename = @env['SCRIPT_NAME'].to_s.match(%r{[^/]+$})
        uri = @env['PATH_INFO']
283
        uri = uri.sub(/#{script_filename}\//, '') unless script_filename.nil?
284
        unless (env_qs = @env['QUERY_STRING']).nil? || env_qs.empty?
285 286
          uri << '?' << env_qs
        end
287 288 289 290 291 292 293

        if uri.nil?
          @env.delete('REQUEST_URI')
          uri
        else
          @env['REQUEST_URI'] = uri
        end
294
      end
295
    end
D
Initial  
David Heinemeier Hansson 已提交
296

297
    # Returns the interpreted path to requested resource after all the installation directory of this application was taken into account
D
Initial  
David Heinemeier Hansson 已提交
298
    def path
299
      path = (uri = request_uri) ? uri.split('?').first.to_s : ''
300

301
      # Cut off the path to the installation directory if given
302 303
      path.sub!(%r/^#{relative_url_root}/, '')
      path || ''      
304
    end
305
    
306
    # Returns the path minus the web server relative installation directory.
307 308 309
    # This can be set with the environment variable RAILS_RELATIVE_URL_ROOT.
    # It can be automatically extracted for Apache setups. If the server is not
    # Apache, this method returns an empty string.
310
    def relative_url_root
311 312 313 314 315 316 317 318
      @@relative_url_root ||= case
        when @env["RAILS_RELATIVE_URL_ROOT"]
          @env["RAILS_RELATIVE_URL_ROOT"]
        when server_software == 'apache'
          @env["SCRIPT_NAME"].to_s.sub(/\/dispatch\.(fcgi|rb|cgi)$/, '')
        else
          ''
      end
D
Initial  
David Heinemeier Hansson 已提交
319 320 321
    end


322 323
    # Read the request body. This is useful for web services that need to
    # work with raw requests directly.
324
    def raw_post
325 326 327 328 329
      unless env.include? 'RAW_POST_DATA'
        env['RAW_POST_DATA'] = body.read(content_length)
        body.rewind if body.respond_to?(:rewind)
      end
      env['RAW_POST_DATA']
330 331
    end

332 333
    # Returns both GET and POST parameters in a single hash.
    def parameters
334
      @parameters ||= request_parameters.merge(query_parameters).update(path_parameters).with_indifferent_access
D
Initial  
David Heinemeier Hansson 已提交
335
    end
336

337
    def path_parameters=(parameters) #:nodoc:
338
      @path_parameters = parameters
339 340
      @symbolized_path_parameters = @parameters = nil
    end
341

342 343
    # The same as <tt>path_parameters</tt> with explicitly symbolized keys 
    def symbolized_path_parameters 
344
      @symbolized_path_parameters ||= path_parameters.symbolize_keys
345
    end
D
Initial  
David Heinemeier Hansson 已提交
346

347 348
    # Returns a hash with the parameters used to form the path of the request.
    # Returned hash keys are strings.  See <tt>symbolized_path_parameters</tt> for symbolized keys.
349 350 351
    #
    # Example: 
    #
352
    #   {'action' => 'my_action', 'controller' => 'my_controller'}
353 354 355
    def path_parameters
      @path_parameters ||= {}
    end
356 357


D
Initial  
David Heinemeier Hansson 已提交
358 359 360
    #--
    # Must be implemented in the concrete request
    #++
361 362 363 364 365

    # The request body is an IO input stream.
    def body
    end

366
    def query_parameters #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
367 368
    end

369
    def request_parameters #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
370 371
    end

372
    def cookies #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
373 374
    end

375
    def session #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
376 377
    end

378 379 380 381
    def session=(session) #:nodoc:
      @session = session
    end

382
    def reset_session #:nodoc:
383
    end
384

385 386 387
    protected
      # The raw content type string. Use when you need parameters such as
      # charset or boundary which aren't included in the content_type MIME type.
388
      # Overridden by the X-POST_DATA_FORMAT header for backward compatibility.
389
      def content_type_with_parameters
390 391
        content_type_from_legacy_post_data_format_header ||
          env['CONTENT_TYPE'].to_s
392 393 394 395 396 397 398 399 400 401 402
      end

      # The raw content type string with its parameters stripped off.
      def content_type_without_parameters
        @content_type_without_parameters ||= self.class.extract_content_type_without_parameters(content_type_with_parameters)
      end

    private
      def content_type_from_legacy_post_data_format_header
        if x_post_format = @env['HTTP_X_POST_DATA_FORMAT']
          case x_post_format.to_s.downcase
403 404
            when 'yaml';  'application/x-yaml'
            when 'xml';   'application/xml'
405 406 407 408
          end
        end
      end

409
      def parse_formatted_request_parameters
410 411
        return {} if content_length.zero?

412
        content_type, boundary = self.class.extract_multipart_boundary(content_type_with_parameters)
413 414

        # Don't parse params for unknown requests.
415 416 417 418 419 420
        return {} if content_type.blank?

        mime_type = Mime::Type.lookup(content_type)
        strategy = ActionController::Base.param_parsers[mime_type]

        # Only multipart form parsing expects a stream.
J
Jeremy Kemper 已提交
421
        body = (strategy && strategy != :multipart_form) ? raw_post : self.body
422 423 424 425 426

        case strategy
          when Proc
            strategy.call(body)
          when :url_encoded_form
427 428
            self.class.clean_up_ajax_request_body! body
            self.class.parse_query_parameters(body)
429
          when :multipart_form
430
            self.class.parse_multipart_form_parameters(body, boundary, content_length, env)
431 432 433 434
          when :xml_simple, :xml_node
            body.blank? ? {} : Hash.from_xml(body).with_indifferent_access
          when :yaml
            YAML.load(body)
435 436 437 438 439 440 441 442
          when :json
            if body.blank?
              {}
            else
              data = ActiveSupport::JSON.decode(body)
              data = {:_json => data} unless data.is_a?(Hash)
              data.with_indifferent_access
            end
443 444 445 446 447 448
          else
            {}
        end
      rescue Exception => e # YAML, XML or Ruby code block errors
        raise
        { "body" => body,
449
          "content_type" => content_type_with_parameters,
450 451 452 453 454
          "content_length" => content_length,
          "exception" => "#{e.message} (#{e.class})",
          "backtrace" => e.backtrace }
      end

455 456 457 458
      def named_host?(host)
        !(host.nil? || /\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/.match(host))
      end

459
    class << self
460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498
      def parse_query_parameters(query_string)
        return {} if query_string.blank?

        pairs = query_string.split('&').collect do |chunk|
          next if chunk.empty?
          key, value = chunk.split('=', 2)
          next if key.empty?
          value = value.nil? ? nil : CGI.unescape(value)
          [ CGI.unescape(key), value ]
        end.compact

        UrlEncodedPairParser.new(pairs).result
      end

      def parse_request_parameters(params)
        parser = UrlEncodedPairParser.new

        params = params.dup
        until params.empty?
          for key, value in params
            if key.blank?
              params.delete key
            elsif !key.include?('[')
              # much faster to test for the most common case first (GET)
              # and avoid the call to build_deep_hash
              parser.result[key] = get_typed_value(value[0])
              params.delete key
            elsif value.is_a?(Array)
              parser.parse(key, get_typed_value(value.shift))
              params.delete key if value.empty?
            else
              raise TypeError, "Expected array, found #{value.inspect}"
            end
          end
        end

        parser.result
      end

499 500
      def parse_multipart_form_parameters(body, boundary, body_size, env)
        parse_request_parameters(read_multipart(body, boundary, body_size, env))
501
      end
502

503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520
      def extract_multipart_boundary(content_type_with_parameters)
        if content_type_with_parameters =~ MULTIPART_BOUNDARY
          ['multipart/form-data', $1.dup]
        else
          extract_content_type_without_parameters(content_type_with_parameters)
        end
      end

      def extract_content_type_without_parameters(content_type_with_parameters)
        $1.strip.downcase if content_type_with_parameters =~ /^([^,\;]*)/
      end

      def clean_up_ajax_request_body!(body)
        body.chop! if body[-1] == 0
        body.gsub!(/&_=$/, '')
      end


521 522 523 524 525 526 527 528 529 530
      private
        def get_typed_value(value)
          case value
            when String
              value
            when NilClass
              ''
            when Array
              value.map { |v| get_typed_value(v) }
            else
531
              if value.respond_to? :original_filename
532 533 534 535 536 537 538 539
                # Uploaded file
                if value.original_filename
                  value
                # Multipart param
                else
                  result = value.read
                  value.rewind
                  result
540 541 542 543 544 545 546 547 548 549 550 551
                end
              # Unknown value, neither string nor multipart.
              else
                raise "Unknown form value: #{value.inspect}"
              end
          end
        end

        MULTIPART_BOUNDARY = %r|\Amultipart/form-data.*boundary=\"?([^\";,]+)\"?|n

        EOL = "\015\012"

552
        def read_multipart(body, boundary, body_size, env)
553 554
          params = Hash.new([])
          boundary = "--" + boundary
555
          quoted_boundary = Regexp.quote(boundary)
556 557 558 559 560 561
          buf = ""
          bufsize = 10 * 1024
          boundary_end=""

          # start multipart/form-data
          body.binmode if defined? body.binmode
562 563 564 565 566 567
          case body
          when File
            body.set_encoding(Encoding::BINARY) if body.respond_to?(:set_encoding)
          when StringIO
            body.string.force_encoding(Encoding::BINARY) if body.string.respond_to?(:force_encoding)
          end
568
          boundary_size = boundary.size + EOL.size
569
          body_size -= boundary_size
570 571 572 573 574 575 576 577 578 579
          status = body.read(boundary_size)
          if nil == status
            raise EOFError, "no content body"
          elsif boundary + EOL != status
            raise EOFError, "bad content body"
          end

          loop do
            head = nil
            content =
580
              if 10240 < body_size
581
                UploadedTempfile.new("CGI")
582
              else
583
                UploadedStringIO.new
584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601
              end
            content.binmode if defined? content.binmode

            until head and /#{quoted_boundary}(?:#{EOL}|--)/n.match(buf)

              if (not head) and /#{EOL}#{EOL}/n.match(buf)
                buf = buf.sub(/\A((?:.|\n)*?#{EOL})#{EOL}/n) do
                  head = $1.dup
                  ""
                end
                next
              end

              if head and ( (EOL + boundary + EOL).size < buf.size )
                content.print buf[0 ... (buf.size - (EOL + boundary + EOL).size)]
                buf[0 ... (buf.size - (EOL + boundary + EOL).size)] = ""
              end

602
              c = if bufsize < body_size
603 604
                    body.read(bufsize)
                  else
605
                    body.read(body_size)
606 607 608 609 610
                  end
              if c.nil? || c.empty?
                raise EOFError, "bad content body"
              end
              buf.concat(c)
611
              body_size -= c.size
612 613 614 615 616
            end

            buf = buf.sub(/\A((?:.|\n)*?)(?:[\r\n]{1,2})?#{quoted_boundary}([\r\n]{1,2}|--)/n) do
              content.print $1
              if "--" == $2
617
                body_size = -1
618
              end
619
              boundary_end = $2.dup
620 621 622 623 624
              ""
            end

            content.rewind

625 626 627 628 629 630 631 632
            head =~ /Content-Disposition:.* filename=(?:"((?:\\.|[^\"])*)"|([^;]*))/ni
            if filename = $1 || $2
              if /Mac/ni.match(env['HTTP_USER_AGENT']) and
                  /Mozilla/ni.match(env['HTTP_USER_AGENT']) and
                  (not /MSIE/ni.match(env['HTTP_USER_AGENT']))
                filename = CGI.unescape(filename)
              end
              content.original_path = filename.dup
633 634
            end

635 636
            head =~ /Content-Type: ([^\r]*)/ni
            content.content_type = $1.dup if $1
637

638 639
            head =~ /Content-Disposition:.* name="?([^\";]*)"?/ni
            name = $1.dup if $1
640 641 642 643 644 645

            if params.has_key?(name)
              params[name].push(content)
            else
              params[name] = [content]
            end
646
            break if body_size == -1
647 648 649
          end
          raise EOFError, "bad boundary end of body part" unless boundary_end=~/--/

650
          begin
651
            body.rewind if body.respond_to?(:rewind)
652
          rescue Errno::ESPIPE
653 654
            # Handles exceptions raised by input streams that cannot be rewound
            # such as when using plain CGI under Apache
655
          end
656

657 658 659 660 661 662 663 664 665 666 667 668
          params
        end
    end
  end

  class UrlEncodedPairParser < StringScanner #:nodoc:
    attr_reader :top, :parent, :result

    def initialize(pairs = [])
      super('')
      @result = {}
      pairs.each { |key, value| parse(key, value) }
669
    end
670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733

    KEY_REGEXP = %r{([^\[\]=&]+)}
    BRACKETED_KEY_REGEXP = %r{\[([^\[\]=&]+)\]}

    # Parse the query string
    def parse(key, value)
      self.string = key
      @top, @parent = result, nil

      # First scan the bare key
      key = scan(KEY_REGEXP) or return
      key = post_key_check(key)

      # Then scan as many nestings as present
      until eos?
        r = scan(BRACKETED_KEY_REGEXP) or return
        key = self[1]
        key = post_key_check(key)
      end

      bind(key, value)
    end

    private
      # After we see a key, we must look ahead to determine our next action. Cases:
      #
      #   [] follows the key. Then the value must be an array.
      #   = follows the key. (A value comes next)
      #   & or the end of string follows the key. Then the key is a flag.
      #   otherwise, a hash follows the key.
      def post_key_check(key)
        if scan(/\[\]/) # a[b][] indicates that b is an array
          container(key, Array)
          nil
        elsif check(/\[[^\]]/) # a[b] indicates that a is a hash
          container(key, Hash)
          nil
        else # End of key? We do nothing.
          key
        end
      end

      # Add a container to the stack.
      def container(key, klass)
        type_conflict! klass, top[key] if top.is_a?(Hash) && top.key?(key) && ! top[key].is_a?(klass)
        value = bind(key, klass.new)
        type_conflict! klass, value unless value.is_a?(klass)
        push(value)
      end

      # Push a value onto the 'stack', which is actually only the top 2 items.
      def push(value)
        @parent, @top = @top, value
      end

      # Bind a key (which may be nil for items in an array) to the provided value.
      def bind(key, value)
        if top.is_a? Array
          if key
            if top[-1].is_a?(Hash) && ! top[-1].key?(key)
              top[-1][key] = value
            else
              top << {key => value}.with_indifferent_access
              push top.last
734
              value = top[key]
735 736 737 738 739 740 741
            end
          else
            top << value
          end
        elsif top.is_a? Hash
          key = CGI.unescape(key)
          parent << (@top = {}) if top.key?(key) && parent.is_a?(Array)
742 743
          top[key] ||= value
          return top[key]
744 745 746 747 748 749 750 751
        else
          raise ArgumentError, "Don't know what to do: top is #{top.inspect}"
        end

        return value
      end

      def type_conflict!(klass, value)
752
        raise TypeError, "Conflicting types for parameter containers. Expected an instance of #{klass} but found an instance of #{value.class}. This can be caused by colliding Array and Hash parameters like qs[]=value&qs[key]=value. (The parameters received were #{value.inspect}.)"
753
      end
D
Initial  
David Heinemeier Hansson 已提交
754
  end
755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790

  module UploadedFile
    def self.included(base)
      base.class_eval do
        attr_accessor :original_path, :content_type
        alias_method :local_path, :path
      end
    end

    # Take the basename of the upload's original filename.
    # This handles the full Windows paths given by Internet Explorer
    # (and perhaps other broken user agents) without affecting
    # those which give the lone filename.
    # The Windows regexp is adapted from Perl's File::Basename.
    def original_filename
      unless defined? @original_filename
        @original_filename =
          unless original_path.blank?
            if original_path =~ /^(?:.*[:\\\/])?(.*)/m
              $1
            else
              File.basename original_path
            end
          end
      end
      @original_filename
    end
  end

  class UploadedStringIO < StringIO
    include UploadedFile
  end

  class UploadedTempfile < Tempfile
    include UploadedFile
  end
791
end