如何使用模块动态地将类方法添加到Rails模型

时间:2013-12-17 12:50:49

标签: ruby-on-rails ruby ruby-on-rails-3 metaprogramming meta-search

我在模型类上有这个代码来添加meta_search gem使用的搜索方法:

class Model

  self.def created_at_lteq_at_end_of_day(date)
     where("created_at <= ?", DateTime.strptime(date, '%d/%m/%Y').end_of_day)
  end

  search_methods :created_at_lteq_at_end_of_day
end

此代码将搜索方法添加到日期字段。 现在,需要将此搜索方法添加到其他类和其他字段。 为此,创建了该模块:

LIB / meta_search / extended_search_methods.rb

module MetaSearch
  module ExtendedSearchMethods
    def self.included(base)
      base.extend ClassMethods
    end


    module ClassMethods
      def add_search_by_end_of_day(field, format)

        method_name = "#{field}_lteq_end_of_day"

        define_method method_name do |date|
          where("#{field} <= ?", DateTime.strptime(date, format).end_of_day) 
        end

        search_methods method_name
      end
    end
  end
end


class Model
   include MetaSearch::ExtendedSearchMethods
   add_search_by_end_of_day :created_at, '%d/%m/%Y'
end

添加模块后,此错误开始上升:

undefined method `created_at_lteq_end_of_day' for #<ActiveRecord::Relation:0x007fcd3cdb0e28>

其他解决方案:

define_method更改为define_singleton_method

2 个答案:

答案 0 :(得分:10)

ActiveSupport提供了一种非常惯用且很酷的方法,ActiveSupport::Concern

module Whatever
    extend ActiveSupport::Concern

    module ClassMethods
        def say_hello_to(to)
            puts "Hello #{to}"
        end
    end
end

class YourModel
    include Whatever

    say_hello_to "someone"
end

请参阅API doc。虽然它与您的问题没有直接关系,但included方法对模型或控制器(范围,辅助方法,过滤器等)非常有用,而且ActiveSupport::Concern免费处理模块之间的依赖关系(均为在自由和啤酒中。)

答案 1 :(得分:5)

您需要添加class << self,这样才能使用单件类。

module Library
  module Methods
    def self.included(base)
      base.extend ClassMethods
    end

    module ClassMethods
      def add_foo
        class << self
          define_method "foo" do
            puts "Foo!"
          end #define_method
        end #class << self
      end #add_foo
    end #ClassMethods
  end #Methods
end #Library

class Test
  include Library::Methods

  add_foo
end

puts Test.foo