Rails源代码分析(24):ActionController::Streaming

本文介绍了一种高效文件传输方法,包括send_file和send_data两个核心函数。send_file通过流式传输发送文件,适用于大文件传输;send_data则用于发送二进制数据作为文件下载。两者均支持设置内容类型、文件名等选项。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 这个module提供了两个方法:

1) send_file
      # Sends the file by streaming it 4096 bytes at a time. This way the
      # whole file doesn't need to be read into memory at once.  This makes
      # it feasible to send even large files.
      #
      # Be careful to sanitize the path parameter if it coming from a web
      # page. <tt>send_file(params[:path])</tt> allows a malicious user to
      # download any file on your server.

      # Options:
      # * <tt>:filename</tt> - suggests a filename for the browser to use.
      #   Defaults to <tt>File.basename(path)</tt>.
      # * <tt>:type</tt> - specifies an HTTP content type.
      #   Defaults to 'application/octet-stream'.
      # * <tt>:disposition</tt> - specifies whether the file will be shown inline or downloaded.
      #   Valid values are 'inline' and 'attachment' (default).
      # * <tt>:stream</tt> - whether to send the file to the user agent as it is read (+true+)
      #   or to read the entire file before sending (+false+). Defaults to +true+.
      # * <tt>:buffer_size</tt> - specifies size (in bytes) of the buffer used to stream the file.
      #   Defaults to 4096.
      # * <tt>:status</tt> - specifies the status code to send with the response. Defaults to '200 OK'.
      # * <tt>:url_based_filename</tt> - set to +true+ if you want the browser guess the filename from
      #   the URL, which is necessary for i18n filenames on certain browsers
      #   (setting <tt>:filename</tt> overrides this option).

      # Simple download:
      #
      #   send_file '/path/to.zip'
      #
      # Show a JPEG in the browser:
      #
      #   send_file '/path/to.jpeg', :type => 'image/jpeg', :disposition => 'inline'
      #
      # Show a 404 page in the browser:
      #
      #   send_file '/path/to/404.html', :type => 'text/html; charset=utf-8', :status => 404   

2)send_data
      # Send binary data to the user as a file download.  May set content type, apparent file name,
      # and specify whether to show data inline or download as an attachment.
      #
      # Options:
      # * <tt>:filename</tt> - suggests a filename for the browser to use.
      # * <tt>:type</tt> - specifies an HTTP content type.
      #   Defaults to 'application/octet-stream'.
      # * <tt>:disposition</tt> - specifies whether the file will be shown inline or downloaded.
      #   Valid values are 'inline' and 'attachment' (default).
      # * <tt>:status</tt> - specifies the status code to send with the response. Defaults to '200 OK'.
      
      # Generic data download:
      #
      #   send_data buffer
      #
      # Download a dynamically-generated tarball:
      #
      #   send_data generate_tgz('dir'), :filename => 'dir.tgz'
      #
      # Display an image Active Record in the browser:
      #
      #   send_data image.data, :type => image.content_type, :disposition => 'inline'
      #

2 实现
  1.   module Streaming
  2.     DEFAULT_SEND_FILE_OPTIONS = {
  3.       :type         => 'application/octet-stream'.freeze,
  4.       :disposition  => 'attachment'.freeze,
  5.       :stream       => true,
  6.       :buffer_size  => 4096,
  7.       :x_sendfile   => false
  8.     }.freeze
  9.     X_SENDFILE_HEADER = 'X-Sendfile'.freeze
  10.     protected
  11.       def send_file(path, options = {}) #:doc:
  12.         raise MissingFile, "Cannot read file #{path}" unless File.file?(path) and File.readable?(path) # 确认文件可读
  13.         options[:length]   ||= File.size(path)
  14.         options[:filename] ||= File.basename(path) unless options[:url_based_filename]
  15.         send_file_headers! options
  16.         @performed_render = false
  17.         if options[:x_sendfile] # 采用高速传输模式,不经过服务器的内存
  18.           logger.info "Sending #{X_SENDFILE_HEADER} header #{path}" if logger
  19.           head options[:status], X_SENDFILE_HEADER => path
  20.         else # 采用经过内存的读取模式
  21.           if options[:stream]
  22.             # 如果:stream设置为true 那么一边读取一边发送
  23.             render :status => options[:status], :text => Proc.new { |response, output|
  24.               logger.info "Streaming file #{path}" unless logger.nil?
  25.               len = options[:buffer_size] || 4096
  26.               File.open(path, 'rb'do |file|
  27.                 while buf = file.read(len)
  28.                   output.write(buf)
  29.                 end
  30.               end
  31.             }
  32.           else
  33.           # 如果:stream设置为false那么整个文件读取结束才发送
  34.             logger.info "Sending file #{path}" unless logger.nil?
  35.             File.open(path, 'rb') { |file| render :status => options[:status], :text => file.read }
  36.           end
  37.         end
  38.       end
  39.       # 直接发送data
  40.      def send_data(data, options = {}) #:doc:
  41.         logger.info "Sending data #{options[:filename]}" if logger
  42.         send_file_headers! options.merge(:length => data.size)
  43.         @performed_render = false
  44.         render :status => options[:status], :text => data
  45.       end
  46.     private
  47.       def send_file_headers!(options)
  48.         options.update(DEFAULT_SEND_FILE_OPTIONS.merge(options))
  49.         [:length:type:disposition].each do |arg|
  50.           raise ArgumentError, ":#{arg} option required" if options[arg].nil?
  51.         end #检查必须有的参数
  52.         disposition = options[:disposition].dup || 'attachment'
  53.         disposition <<= %(; filename="#{options[:filename]}"if options[:filename] #如果有:filename拼接到 Content-Disposition 
  54.         headers.update(
  55.           'Content-Length'            => options[:length],
  56.           'Content-Type'              => options[:type].to_s.strip,  # fixes a problem with extra '/r' with some browsers
  57.           'Content-Disposition'       => disposition,
  58.           'Content-Transfer-Encoding' => 'binary'
  59.         )
  60.         # Fix a problem with IE 6.0 on opening downloaded files:
  61.         # If Cache-Control: no-cache is set (which Rails does by default),
  62.         # IE removes the file it just downloaded from its cache immediately
  63.         # after it displays the "open/save" dialog, which means that if you
  64.         # hit "open" the file isn't there anymore when the application that
  65.         # is called for handling the download is run, so let's workaround that
  66.         headers['Cache-Control'] = 'private' if headers['Cache-Control'] == 'no-cache'
  67.       end
  68.   end




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值