Rails 命名空间模型与非命名空间模型冲突



我在 rails 3.2.6 中有两个类:

  • 类 Foo in models/foo.rb (默认在表 foos 中)
  • class Bar::Foo in models/bar/foo.rb (将self.table_name设置为 bar_foos

当我进入控制台时,我得到:

> Bar::Foo
=> Bar::Foo(id: ...)
> Foo # or ::Foo
LoadError: expected models/bar/foo.rb to define Foo

怎么了?

我们在 IRC 中解决了这个问题,但核心问题是有一个包含models/**作为负载路径的config.autoload_paths glob 集。

Rails的自动加载器迭代加载路径,并固定常量名称。找到存在的文件后,它会尝试加载该文件,然后在常量不可用时引发异常。

所以,发生的事情是Rails有一个负载路径列表,例如:

/models/bar/
/models/

它正在迭代路径,并在 /models/bar/foo.rb 处找到匹配项,然后加载该匹配项(这使Bar::Foo可用,但不可用Foo),然后抛出异常,因为Foo不可用。

在这种情况下,解决方案是删除autoload_paths设置,以便 Rails 不会找到要为根级常量加载的错误文件。

事实证明,config/applications.rb 中的这一行是问题所在:

 config.autoload_paths += Dir[Rails.root.join('app', 'models', '{**}')]

通过显式设置的自动加载,Rails感到困惑;它没有通过命名空间适当地查看模型/,而是查看了它拥有的第一个自动加载文件(错误地,是models/bar/foo.rb),并发现(true)它未能定义Foo(它定义了Bar::Foo)。

所以很明显,Rails 3已经知道在模型/子目录中查找命名空间模型。

感谢 freenode #RubyOnRails 上的 Antiarc 帮助解决这个问题。

最新更新