【问题标题】:Rails namespaced model conflicting with non-namespaced modelRails 命名空间模型与非命名空间模型冲突
【发布时间】:2012-09-10 04:13:48
【问题描述】:
我在 rails 3.2.6 中有两个类:
- models/foo.rb 中的类 Foo(默认在表 foos 中)
- models/bar/foo.rb 中的类 Bar::Foo(将
self.table_name 设置为 bar_foos)
当我进入控制台时,我得到:
> Bar::Foo
=> Bar::Foo(id: ...)
> Foo # or ::Foo
LoadError: expected models/bar/foo.rb to define Foo
怎么了?
【问题讨论】:
标签:
ruby-on-rails
activerecord
namespaces
【解决方案1】:
我们在 IRC 中解决了这个问题,但核心问题是有一个 config.autoload_paths glob 集包含 models/** 作为加载路径。
Rails 的自动加载器会迭代加载路径,并添加常量名称。一旦它找到一个存在的文件,它就会尝试加载它,如果常量不可用则抛出异常。
所以,Rails 有一个加载路径列表,例如:
/models/bar/
/models/
它正在迭代路径,并会在 /models/bar/foo.rb 找到匹配项,然后加载该匹配项(这使 Bar::Foo 可用,但 Foo 不可用),然后抛出异常,因为 Foo 不可用.
这种情况下的解决方案是删除autoload_paths 设置,这样Rails 就不会为根级常量找到要加载的错误文件。
【解决方案2】:
原来 config/applications.rb 中的这一行是问题所在:
config.autoload_paths += Dir[Rails.root.join('app', 'models', '{**}')]
通过显式设置自动加载,Rails 感到困惑;它没有通过适当的命名空间来查看 models/ 下,而是查看了它拥有的第一个自动加载文件(错误地是 models/bar/foo.rb)并发现(true)它未能定义 Foo(它定义了 Bar::富)。
很明显,Rails 3 已经知道在模型/子目录中查找命名空间模型。
感谢 freenode 上的 Antiarc #RubyOnRails 帮助解决这个问题。