因此,在我的控制器中,我有这个:
def new @pmdocument = Pmdocument.new @pmdocument.projectmilestone_id=params[:projectmilestone_id]
其中projectmilestone_id是我项目的foreign_key(用作我的桶名)
我的模型是这样的:
class Pmdocument < ActiveRecord::Base belongs_to :projectmilestone attr_accessible :id,:name,:description,:projectmilestone_id,:pmdoc,:projectmilestone_attributes attr_protected :pmdoc_content_type,:pmdoc_size accepts_nested_attributes_for :projectmilestone,:allow_destroy => false has_attached_file :pmdoc,:storage => :s3,:bucket => self.projectmilestone_id.to_s,:s3_credentials => File.join(Rails.root,'config','s3.yml')
当我加载页面时,我收到此错误:
用于#的未定义方法`projectmilestone_id’
我检查了我的控制器,并在那里正确加载了projectmilestone_id字段.
我试图将铲斗线更改为:bucket => self.name然后错误消失了.
该模型工作正常,因为projectmilestone_id正确存储在db中.
怎么了?非常感谢!!!
我真的不明白:
我决定不再改变我的桶了(不管怎样,坏主意因为所有S3的名称都必须是唯一的),而是改变我的路径.
这是代码:
:path => proc { |attachment| "#{attachment.istance.projectname}/:attachment/:id/:basename.:extension" },
未创建具有项目名称的第一个文件夹.如果我按名称或甚至描述(pmdocuments的另一个字段)替换projectname,它可以工作,但不能用projectname.当然,我检查了projectname是否正确填充.原因在于其他地方.
任何线索?
解决方法
但Paperclip非常友好,可以满足您的需求. documentation on the S3 storage说:
You can define the bucket as a Proc if
you want to determine it’s name at
runtime. Paperclip will call that Proc
with attachment as the only argument.
在你的情况下,它将是这样的:
has_attached_file :pmdoc,:bucket => proc { |attachment| attachment.instance.projectmilestone_id.to_s },'s3.yml')
现在您将Proc传递给has_attached_file.在加载类时不会评估块的内容,但在以后需要时会对其进行评估. Paperclip然后使用attachment作为参数调用块,并使用返回的值作为存储桶名称.
编辑:
不幸的是,这个块在分配文件时运行,而不是在保存记录时运行.因此,您的所有属性可能尚未设置(当您执行Pmdocument.new(params [:pmdocument]时)属性分配的顺序未确定).我想让Paperclip以另一种方式工作,但与此同时我看到两个选择:
您可以从控制器中的参数中删除该文件,并在其他所有准备就绪时进行设置:
pmdoc = params[:pmdocument].delete(:pmdoc) @pmdocument = Pmdocument.new(params[:pmdocument]) @pmdocument.pmdoc = pmdoc
或者,您可以通过使用before_post_process禁用Paperclip后处理(请参阅README的“事件”部分)并在after_save
callback中运行它来延迟Paperclip后处理.