Ruby中的动态validation和元编程

我正在尝试开发一个应用程序,该应用程序可以向不同的用户提供相同的资源,并且资源可能具有基于用户的不同validation行为。

我曾尝试使用Ruby元编程以简单的方式解决这个问题,但看起来我缺少一些关于此事的关键知识。

我可以通过诸如此类的模型来举例说明

class Profile < ActiveRecord::Base # validates_presence_of :string1 end 

该模型有一个属性’string1’,有时是必需的,有时不是。 我想为每个用户创建子类(由于此简化中不明显的原因)并创建了一个我想要包含的模块:

 module ExtendProfile def self.included(base) base.extend(ClassMethods) end module ClassMethods def configure_validation(required) if required class_eval("ActiveRecord::Base.validates_presence_of :string1") end end end end 

它的唯一目的是添加一个方法,根据给定的参数添加条件validation。

当使用一个为true的参数调用时,它会添加validation,但它不会干净地执行。 看起来它并没有像我想象的那样分离子类。

它可以通过以下测试来说明:

 profile = Profile.new profile.save; profile.errors => [] 

默认情况下,配置文件可以保存而不会出错。

 Object::const_set('FirstExtendedProfile'.intern, Class::new(Profile) { include ExtendProfile }) FirstExtendedProfile.configure_validation(true) fep = FirstExtendedProfile.new; fep.save; fep.errors => {:string1=>["skal udfyldes", "skal udfyldes"]} 

创建一个新的子类并调用configuring_validation会增加validation,但由于某种原因,它在validation期间被调用两次(“skal udfyldes” – 是丹麦语,意味着它是必需的)。

 Object::const_set('SecondExtendedProfile'.intern, Class::new(Profile) { include ExtendProfile }) sep = SecondExtendedProfile.new; sep.save; sep.errors => {:string1=>["skal udfyldes"]} 

创建了另一个后代,即使未调用configure_validation它仍然会validationstring1属性(但现在只有一次)。

添加另一个后代并调用configure_validation再次添加validation…

为什么我无法将validation添加到特定的配置文件后代?

我使用的是Ruby 1.9.2和Rails 3.06。 请理解我想了解如何使这个动态类创建工作 – 我知道“标准”自定义validation。

由于在使用单表inheritance时变得清晰的原因,validation存储在“root”类中的变量中,该类是直接inheritanceActiveRecord :: Base的类。 在幕后做了相当多的工作,以确保你尝试做的事情不起作用。

我的建议是在每个类中存储一些配置数据,然后编写一个动态validation,检查配置并根据它在那里找到的内容进行validation。 然而,这可能是“我知道”标准“自定义validation”的含义。

  with_options :if => lambda { |obj| obj.class.validation_configuration[:string1]} do |t| t.validates_presence_of :string1 end 

这是解决问题的另一种方法…而不是使用元编程来扩展或不扩展类:

 class Profile with_options :if => lambda { |o| o.required? } do |on_required| on_required.validates_presence_of :string1 end end