如何在父app的模型和挂载的引擎模型之间设置has_many关联?

时间:2015-01-26 17:50:33

标签: ruby-on-rails ruby ruby-on-rails-4

按照RailsGuides指令,我在我的应用程序中创建了一个用于博客系统的引擎。此博客引擎安装为/ blog。

RailsGuides显示如何将belongs_to关联添加到已挂载引擎的Article模型。但是,父应用的User模型仍需要has_many与引擎的Article模型关联,该模型位于不同的命名空间中。

如何在父app的模型和挂载的引擎模型之间设置has_many关联?

  • Ruby 2.2.0
  • Rails 4.2.0

提前致谢。

2 个答案:

答案 0 :(得分:3)

在rails应用程序中,您知道包含哪个模块,因此您只需指定与类名称的关系;)

has_many :articles, class_name: 'Blog::Article'

检查这是否是数据库适配器的正确语法,例如我将它用于Mongoid,但它应该与ActiveRecord AFAIK相同

答案 1 :(得分:0)

接受的答案需要手动修改main_app的父模型,以便将has_many关系设置为引擎的子模型。因此,每次将引擎添加到main_apps中时,您都必须进入main_apps模型并手动设置所有必需的关系。

更强大但更复杂的解决方案是在引擎中使用装饰器模式,以便引擎自动配置main_app的父模型及其所需的关系。

通过使用此方法,您只需要在main_app中为引擎初始化程序添加一个设置,引擎将处理其余部分。

在引擎中:

blog.gemspec.rb

s.add_dependency 'decorators' #this will install the decorators gem for use in engine

LIB /博客/ blog.rb

module Blog
  class Engine < ::Rails::Engine
    isolate_namespace Blog
    engine_name 'blog'

    #to set up main_app objects via decorators in engine
    config.to_prepare do
      Decorators.register! Engine.root, Rails.root
    end

  end
end

LIB / blog.rb

require 'decorators'

module Blog

  mattr_accessor :user_class #Can now reference this setting as Blog.user_class

class << self

    #the following lets us add functionality to main_app user model

    def decorate_user_class!
      Blog.user_class.class_eval do
        has_many :articles, :class_name => "Blog::Article", :foreign_key => "user_id"
      end
    end

  end
end

应用程序/装饰/ LIB /博客/ user_class_decorator.rb

if Blog.user_class
  Blog.decorate_user_class!
else
  raise "Blog.user_class must be set in main_app blog.rb initializer"
end

在主应用中:

应用程序/初始化/ blog.rb

Blog.user_class = User

如果您从主应用程序运行rails控制台,您将看到关系已正确设置。引擎中的装饰器模式也可用于以不同方式扩展main_app的模型和控制器,而不仅仅是Activerecord关系。实现了几乎完全解耦!