如何使用RSpec针对真实应用开发Rails3引擎?

时间:2011-07-05 18:49:00

标签: ruby-on-rails-3 rspec rails-engines

有很多关于引擎开发和使用虚拟应用程序进行测试的文章。

在我们的案例中,我们正在开发一个不是独立实体的引擎,但它依赖于真正的Rails 3应用程序。我们仍然希望此代码存在于引擎中,而不是成为应用程序的一部分,因为引擎的工作是从具有自己的表和模型映射的遗留系统导入数据,并且我们希望最终再次删除它。 / p>

旧的遗留表和新模式之间的数据映射很复杂,我们想要TDD(使用rspec)引擎。

  • 我已关注Jose Valim's book "Crafting Rails Appliations"并正在使用enginex gem
  • 我用一个指向真正的Rails 3应用程序的git子模块替换了/spec/dummy_app
  • 我无法从引擎加载模型(未定义的符号错误),因为真实应用程序的Gemfile没有指向引擎,我也无法修改config/application.rb,需要引擎(正如虚拟应用程序所做的那样,正如本书第15-16页所述)。
  • 我将引擎的lib文件夹包含在$:的加载路径spec_helper中,路径可用。
  • require放入spec_helper.rb并没有解决问题。
  • 我想知道是否有一个内部Rails API(或一个聪明的猴子补丁)挂钩到真实应用程序的启动序列并需要引擎,而不必修改真实应用程序的代码(因为它在子模块中) )。
  • 另一个我不完全确定的问题是我有2个Gemfiles(一个在引擎中,一个在应用程序中),当引擎处于活动状态时,它们都应该被使用。

思想?

1 个答案:

答案 0 :(得分:8)

所以,我想出了一些事情并且会回答我自己的问题,现在我已经开始工作了。

为简单起见,我将把我宝石的名称称为“my_gem”和“MyGem”:

engine.rb文件中,我添加了:

require 'my_gem'
require 'rails'

这种类型的固定错误:

my_gem/lib/my_gem/engine.rb:2: uninitialized constant MyGem::Rails (NameError)

spec_helper.rb中,我直接添加了:

require 'bundler/setup'
require 'my_gem'

这是为了确保Bundler立即初始化,而不是通过应用程序。这样我可以在这里加载MyGem,它将被挂钩到应用程序的初始化序列中。这修复了引擎模型类的NameError个异常。

这留下了Gemfile使用的问题。问题是我的应用程序有自己的gemfile,而gem /引擎需要在其自己的Gemfile中单独的依赖项。

我找不到Bundler的任何API来传递两个Gemfile,实际上Bundler似乎是围绕一个权威Gemfile的假设而构建的。所以我在spec_helper中生成一个。我使用应用程序的gemfile并附加gemspec,它指向gem的GemSpec格式的依赖项。 (顺便说一句,Jose Valim的书中没有提到关于gemspec的提示。)

我不知道在测试启动期间是否有比连接文件更好的方法。如果您知道其中一个,请回复。

有用的资源是: