我有一个基于插件的系统,我用于Rails中的应用程序开发.每个插件都实现了一个带有MVC组件的引擎等.主应用程序只是将所有工作委托给所安装的插件的空线程.
我正在从Rails 2.3.5升级到Rails 3.1,并试图让资产管道与我的框架一起工作.
我遇到的问题是尝试以编程方式要求我的插件的资产,例如application.js清单.
我可以手动添加它们:
//= require <plugin_manifest_path>
一切都按预期工作.然而,由于我的框架中有几十个插件,每个安装都有不同的组合,所以我想根据安装的插件来更改这个清单.我试过这个解决方案:
<%= Rails.plugins.collect do |plugin| "//= require #{plugin}" end.join("\n") %>
但是我发现资产管道编译的需求/指令阶段是在再培训局扩充之前发生的,所以生成的意见只是作为评论结束.
是否有另一种机制包含编译可能有效的路径?在指令处理之前预先处理清单文件的任何方式?
如果我不能想到更好的东西,我可能必须编写一个在部署时生成一个plugin.js清单文件的耙子/部署任务,但如果可能,我会喜欢更清晰和更优雅的东西.谢谢!
解决方法
好的,所以这里的解决方案:
在内部,资产管道(也称为Sprockets)需要指令调用context.require_asset()来实际需要指令中指定的任何路径.原来,这意味着在ERB扩展期间,require_asset方法存在并可用.所以,正确的解决方案是:
// ... Standard application.js stuff here ... //= require_tree . <% Rails.plugins.each do |plugin| require_asset(plugin.to_s) end %>
补充说,并且都按预期工作.呼!
如果您想知道Rails.plugins位,我添加了一个扩展到Rails模块,以获取加载的插件的实际列表,按照加载顺序,基于config.plugins.为了完整,这里是:
module Rails def self.plugins # Get sorted list of all installed plugins all = Dir.glob(Rails.path('vendor/plugins/*/init.rb')).collect {|p| p.extract(/\/([^\/]+)\/init.rb$/) } all.sort! all.collect! {|p| p.to_sym } # Get our load order specification load_order = Rails.application.config.plugins # Split the load order out,and re-assemble replacing the :all keyword with the # set of plugins not in head or tail head,tail = load_order.split(:all) all -= head all -= tail # All set! head + all + tail end end
最后的细节是在每个插件的app / assets / javascripts目录中创建一个< plugin_name> .js清单文件.
希望能帮助别人!