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

D
Initial  
David Heinemeier Hansson 已提交
5
module ActionController
6
  # CgiRequest and TestRequest provide concrete implementations.
D
Initial  
David Heinemeier Hansson 已提交
7
  class AbstractRequest
8
    cattr_accessor :relative_url_root
9
    remove_method :relative_url_root
10

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

15 16 17
    # The HTTP request method as a lowercase symbol, such as :get.
    # Note, HEAD is returned as :get since the two are functionally
    # equivalent from the application's perspective.
D
Initial  
David Heinemeier Hansson 已提交
18
    def method
19 20 21 22 23 24 25
      @request_method ||=
        if @env['REQUEST_METHOD'] == 'POST' && !parameters[:_method].blank?
          parameters[:_method].to_s.downcase.to_sym
        else
          @env['REQUEST_METHOD'].downcase.to_sym
        end

26
      @request_method == :head ? :get : @request_method
D
Initial  
David Heinemeier Hansson 已提交
27 28
    end

29
    # Is this a GET (or HEAD) request?  Equivalent to request.method == :get
D
Initial  
David Heinemeier Hansson 已提交
30 31 32 33
    def get?
      method == :get
    end

34
    # Is this a POST request?  Equivalent to request.method == :post
D
Initial  
David Heinemeier Hansson 已提交
35 36 37 38
    def post?
      method == :post
    end

39
    # Is this a PUT request?  Equivalent to request.method == :put
D
Initial  
David Heinemeier Hansson 已提交
40 41 42 43
    def put?
      method == :put
    end

44
    # Is this a DELETE request?  Equivalent to request.method == :delete
D
Initial  
David Heinemeier Hansson 已提交
45 46 47 48
    def delete?
      method == :delete
    end

49 50
    # Is this a HEAD request? request.method sees HEAD as :get, so check the
    # HTTP method directly.
51
    def head?
52
      @env['REQUEST_METHOD'].downcase.to_sym == :head
53
    end
54

55 56 57 58
    def headers
      @env
    end

59 60 61 62
    def content_length
      @content_length ||= env['CONTENT_LENGTH'].to_i
    end

63
    # The MIME type of the HTTP request, such as Mime::XML.
64 65 66
    #
    # For backward compatibility, the post format is extracted from the
    # X-Post-Data-Format HTTP header if present.
67
    def content_type
68
      @content_type ||= Mime::Type.lookup(content_type_without_parameters)
69 70
    end

71
    # Returns the accepted MIME type for the request
72
    def accepts
73 74
      @accepts ||=
        if @env['HTTP_ACCEPT'].to_s.strip.empty?
75
          [ content_type, Mime::ALL ].compact # make sure content_type being nil is not included
76 77 78
        else
          Mime::Type.parse(@env['HTTP_ACCEPT'])
        end
79
    end
80

81 82 83 84 85 86 87
    # Returns the Mime type for the format used in the request. If there is no format available, the first of the 
    # accept types will be used. Examples:
    #
    #   GET /posts/5.xml   | request.format => Mime::XML
    #   GET /posts/5.xhtml | request.format => Mime::HTML
    #   GET /posts/5       | request.format => request.accepts.first (usually Mime::HTML for browsers)
    def format
88
      @format ||= parameters[:format] ? Mime::Type.lookup_by_extension(parameters[:format]) : accepts.first
89
    end
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106
    
    
    # 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
      format
    end
107

108 109 110
    # Returns true if the request's "X-Requested-With" header contains
    # "XMLHttpRequest". (The Prototype Javascript library sends this header with
    # every Ajax request.)
111
    def xml_http_request?
112
      !(@env['HTTP_X_REQUESTED_WITH'] !~ /XMLHttpRequest/i)
113 114 115
    end
    alias xhr? :xml_http_request?

D
Initial  
David Heinemeier Hansson 已提交
116 117 118 119 120 121
    # Determine originating IP address.  REMOTE_ADDR is the standard
    # but will fail if the user is behind a proxy.  HTTP_CLIENT_IP and/or
    # HTTP_X_FORWARDED_FOR are set by proxies so check for these before
    # falling back to REMOTE_ADDR.  HTTP_X_FORWARDED_FOR may be a comma-
    # delimited list in the case of multiple chained proxies; the first is
    # the originating IP.
122
    #
123 124 125 126
    # Security note: do not use if IP spoofing is a concern for your
    # application. Since remote_ip checks HTTP headers for addresses forwarded
    # by proxies, the client may send any IP. remote_addr can't be spoofed but
    # also doesn't work behind a proxy, since it's always the proxy's IP.
D
Initial  
David Heinemeier Hansson 已提交
127
    def remote_ip
128
      return @env['HTTP_CLIENT_IP'] if @env.include? 'HTTP_CLIENT_IP'
129

130 131
      if @env.include? 'HTTP_X_FORWARDED_FOR' then
        remote_ips = @env['HTTP_X_FORWARDED_FOR'].split(',').reject do |ip|
132
          ip.strip =~ /^unknown$|^(10|172\.(1[6-9]|2[0-9]|30|31)|192\.168)\./i
133 134 135
        end

        return remote_ips.first.strip unless remote_ips.empty?
D
Initial  
David Heinemeier Hansson 已提交
136
      end
137

138
      @env['REMOTE_ADDR']
D
Initial  
David Heinemeier Hansson 已提交
139 140
    end

141 142 143 144 145 146 147 148
    # 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 已提交
149
      protocol + host_with_port + request_uri
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
    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
      host + port_string
    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

191 192 193
    # 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)
194
      return nil if !/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/.match(host).nil? or host.nil?
195

196
      host.split('.').last(1 + tld_length).join('.')
197 198 199 200 201 202
    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)
203
      return [] unless host
204
      parts = host.split('.')
205
      parts[0..-(tld_length+2)]
206 207
    end

208 209 210 211 212 213 214 215 216
    # 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

217 218
    # Return the request URI, accounting for server idiosyncracies.
    # WEBrick includes the full URL. IIS leaves REQUEST_URI blank.
D
Initial  
David Heinemeier Hansson 已提交
219
    def request_uri
220
      if uri = @env['REQUEST_URI']
221 222 223 224
        # 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.
225 226
        script_filename = @env['SCRIPT_NAME'].to_s.match(%r{[^/]+$})
        uri = @env['PATH_INFO']
227
        uri = uri.sub(/#{script_filename}\//, '') unless script_filename.nil?
228
        unless (env_qs = @env['QUERY_STRING']).nil? || env_qs.empty?
229 230
          uri << '?' << env_qs
        end
231 232 233 234 235 236 237

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

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

245
      # Cut off the path to the installation directory if given
246 247
      path.sub!(%r/^#{relative_url_root}/, '')
      path || ''      
248
    end
249
    
250
    # Returns the path minus the web server relative installation directory.
251 252 253
    # 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.
254
    def relative_url_root
255 256 257 258 259 260 261 262
      @@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 已提交
263 264 265
    end


266 267
    # Read the request body. This is useful for web services that need to
    # work with raw requests directly.
268
    def raw_post
269 270 271 272 273
      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']
274 275
    end

276 277 278
    # Returns both GET and POST parameters in a single hash.
    def parameters
      @parameters ||= request_parameters.update(query_parameters).update(path_parameters).with_indifferent_access
D
Initial  
David Heinemeier Hansson 已提交
279
    end
280

281
    def path_parameters=(parameters) #:nodoc:
282
      @path_parameters = parameters
283 284
      @symbolized_path_parameters = @parameters = nil
    end
285

286 287
    # The same as <tt>path_parameters</tt> with explicitly symbolized keys 
    def symbolized_path_parameters 
288
      @symbolized_path_parameters ||= path_parameters.symbolize_keys
289
    end
D
Initial  
David Heinemeier Hansson 已提交
290

291 292 293 294 295
    # Returns a hash with the parameters used to form the path of the request 
    #
    # Example: 
    #
    #   {:action => 'my_action', :controller => 'my_controller'}
296 297 298
    def path_parameters
      @path_parameters ||= {}
    end
299 300


D
Initial  
David Heinemeier Hansson 已提交
301 302 303
    #--
    # Must be implemented in the concrete request
    #++
304 305 306 307 308

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

309
    def query_parameters #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
310 311
    end

312
    def request_parameters #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
313 314
    end

315
    def cookies #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
316 317
    end

318
    def session #:nodoc:
D
Initial  
David Heinemeier Hansson 已提交
319 320
    end

321 322 323 324
    def session=(session) #:nodoc:
      @session = session
    end

325
    def reset_session #:nodoc:
326
    end
327

328 329 330
    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.
331
      # Overridden by the X-POST_DATA_FORMAT header for backward compatibility.
332
      def content_type_with_parameters
333 334
        content_type_from_legacy_post_data_format_header ||
          env['CONTENT_TYPE'].to_s
335 336 337 338 339 340 341 342 343 344 345
      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
346 347
            when 'yaml';  'application/x-yaml'
            when 'xml';   'application/xml'
348 349 350 351
          end
        end
      end

352
      def parse_formatted_request_parameters
353 354
        return {} if content_length.zero?

355
        content_type, boundary = self.class.extract_multipart_boundary(content_type_with_parameters)
356 357

        # Don't parse params for unknown requests.
358 359 360 361 362 363
        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 已提交
364
        body = (strategy && strategy != :multipart_form) ? raw_post : self.body
365 366 367 368 369

        case strategy
          when Proc
            strategy.call(body)
          when :url_encoded_form
370 371
            self.class.clean_up_ajax_request_body! body
            self.class.parse_query_parameters(body)
372
          when :multipart_form
373
            self.class.parse_multipart_form_parameters(body, boundary, content_length, env)
374 375 376 377 378 379 380 381 382 383
          when :xml_simple, :xml_node
            body.blank? ? {} : Hash.from_xml(body).with_indifferent_access
          when :yaml
            YAML.load(body)
          else
            {}
        end
      rescue Exception => e # YAML, XML or Ruby code block errors
        raise
        { "body" => body,
384
          "content_type" => content_type_with_parameters,
385 386 387 388 389
          "content_length" => content_length,
          "exception" => "#{e.message} (#{e.class})",
          "backtrace" => e.backtrace }
      end

390
    class << self
391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431
      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

      def parse_multipart_form_parameters(body, boundary, content_length, env)
        parse_request_parameters(read_multipart(body, boundary, content_length, env))
432
      end
433

434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451
      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


452 453 454 455 456 457 458 459 460 461
      private
        def get_typed_value(value)
          case value
            when String
              value
            when NilClass
              ''
            when Array
              value.map { |v| get_typed_value(v) }
            else
462 463 464 465 466 467 468 469 470
              if value.is_a?(UploadedFile)
                # Uploaded file
                if value.original_filename
                  value
                # Multipart param
                else
                  result = value.read
                  value.rewind
                  result
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 499 500 501 502 503 504 505 506
                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"

        def read_multipart(body, boundary, content_length, env)
          params = Hash.new([])
          boundary = "--" + boundary
          quoted_boundary = Regexp.quote(boundary, "n")
          buf = ""
          bufsize = 10 * 1024
          boundary_end=""

          # start multipart/form-data
          body.binmode if defined? body.binmode
          boundary_size = boundary.size + EOL.size
          content_length -= boundary_size
          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 =
              if 10240 < content_length
507
                UploadedTempfile.new("CGI")
508
              else
509
                UploadedStringIO.new
510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550
              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

              c = if bufsize < content_length
                    body.read(bufsize)
                  else
                    body.read(content_length)
                  end
              if c.nil? || c.empty?
                raise EOFError, "bad content body"
              end
              buf.concat(c)
              content_length -= c.size
            end

            buf = buf.sub(/\A((?:.|\n)*?)(?:[\r\n]{1,2})?#{quoted_boundary}([\r\n]{1,2}|--)/n) do
              content.print $1
              if "--" == $2
                content_length = -1
              end
             boundary_end = $2.dup
              ""
            end

            content.rewind

551 552 553 554 555 556 557 558
            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
559 560
            end

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

564 565
            head =~ /Content-Disposition:.* name="?([^\";]*)"?/ni
            name = $1.dup if $1
566 567 568 569 570 571 572 573 574 575 576

            if params.has_key?(name)
              params[name].push(content)
            else
              params[name] = [content]
            end
            break if buf.size == 0
            break if content_length == -1
          end
          raise EOFError, "bad boundary end of body part" unless boundary_end=~/--/

577
          body.rewind if body.respond_to?(:rewind)
578 579 580 581 582 583 584 585 586 587 588 589
          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) }
590
    end
591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672

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

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

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

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

      bind(key, value)
    end

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

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

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

      # Bind a key (which may be nil for items in an array) to the provided value.
      def bind(key, value)
        if top.is_a? Array
          if key
            if top[-1].is_a?(Hash) && ! top[-1].key?(key)
              top[-1][key] = value
            else
              top << {key => value}.with_indifferent_access
              push top.last
            end
          else
            top << value
          end
        elsif top.is_a? Hash
          key = CGI.unescape(key)
          parent << (@top = {}) if top.key?(key) && parent.is_a?(Array)
          return top[key] ||= value
        else
          raise ArgumentError, "Don't know what to do: top is #{top.inspect}"
        end

        return value
      end

      def type_conflict!(klass, value)
        raise TypeError, "Conflicting types for parameter containers. Expected an instance of #{klass} but found an instance of #{value.class}. This can be caused by colliding Array and Hash parameters like qs[]=value&qs[key]=value."
      end
D
Initial  
David Heinemeier Hansson 已提交
673
  end
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

  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
710
end