8

我正在使用 Rails 3.1.0.rc8 和 Chromium 15.0.874.102。

我想设置 CSV 下载的文件名。我正在关注这个SO 解决方案,但发现自己无法修改filename标题Content-Disposition

这是我的代码:

module ActionController 
  module CSVHelper

    def render_csv options={}
      if request.env['HTTP_USER_AGENT'] =~ /msie/i
        headers['Pragma'] = "public"
        headers['Content-Type'] = "text/plain"
        headers['Cache-Control'] = "no-cache, must-revalidate, post-check=0, pre-check=0"
        headers['Expires'] = "0"
      else
        headers['Content-Type'] = "text/csv"
      end

      filename = generate_filename options.delete(:basename)
      headers['Content-Disposition'] = "attachment; filename=#{filename}"
    end

    def generate_filename basename=nil, suffix="csv"
      filename = basename || params[:action]
      filename << ".#{suffix}"
      filename
    end

  end
end    

在我的控制器中:

respond_to do |format|
  format.html
  format.csv do
    render_csv(:basename => "my_filename")
    Rails.logger.debug "HEADERS: #{headers.to_s}"
  end
end  

在我的日志中:

[2011-11-28 12:25:49.611] DEBUG - HEADERS: {"Content-Type"=>"text/csv", "Content-Disposition"=>"attachment; filename=my_filename.csv"}

在 Chromium 网络检查器工具中,我在响应标头中看到以下内容:

Content-Type: text/plain
Content-Disposition: attachment; filename=index.csv

如果我将 更改Content-Type为类似的东西foo/bar,我会在我的网络检查器工具中看到更改。无论我设置什么filename,它仍然是index.csv.

谢谢,马克斯

4

4 回答 4

11

根据HTTP/1.1 规范附录的 Content-Disposition 部分,文件名必须是带引号的字符串

   filename-parm = "filename" "=" quoted-string
   ...

一个例子是

   Content-Disposition: attachment; filename="fname.ext"

因此,您可能需要进行以下更改(注意引号):

headers['Content-Disposition'] = "attachment; filename=\"#{filename}\""
于 2012-11-26T21:10:53.947 回答
2

我发现我的团队正在使用csv_builder@filename gem,它允许您通过在控制器中设置来调整下载文件的名称。仍然会向任何能够解释为什么我得到 OP 中描述的行为的人奖励正确答案。

于 2011-11-28T18:30:14.607 回答
2

我个人不使用render此类内容 - 我更喜欢使用send_data. 这是一个例子:

send_data data, :type => 'text/csv', :disposition => 'attachment; filename=my_file_name.csv'
于 2018-02-20T15:15:02.710 回答
1

我看不到您实际发送/渲染输出的位置...是在 .csv.erb 文件中还是什么?或者使用 csv_builder 作为其他答案?这可能是诀窍。

我想出了渲染 .xlsx 输出的相同方法。一旦我将文件放在磁盘上,我就可以这样做:

def example
  respond_to do |format|
    format.xlsx do
      path = some_method_generating_xlsx_file
      headers['Content-Disposition'] = "attachment; filename=test.xlsx"
      render :text => File.binread(path), :content_type => XLSX_MIME_TYPE
    end
  end
end

在我的控制器中,它工作得很好。也许标题被模板渲染破坏了。

于 2012-11-24T08:58:33.047 回答