0

我有一个用户模型,其中存储用户地址等。创建帐户(user_sessions 控制器)时,我只想验证他们的姓名是否存在,地址是可选的。

当他们升级他们的帐户(升级控制器)时,我想验证是否输入了他们的完整地址。

是否有一种基于控制器进行条件验证的优雅方式?

4

2 回答 2

3

您是否尝试过创建一个继承旧模型的新模型。所以你可以有:

class User < ActiveRecord::Base
  attr_accessible :name, :address
  validates_presence_of :name
end

class UpgradedUser < User
  validates_precense_of :address
end

由于它继承自User该类,因此UpgradedUser该类对地址和名称都有验证。而且这个解决方案仍然只使用一个数据库表。因为它们仍然是一个表,所以您必须添加一:account_status列或其他内容来确定帐户是否已升级。

在您的 upgrade_users 控制器中,您将有一个标准的更新方法:

class upgraded_users_controller   

  def update
    @upgraded_user = UpgradedUser.find params[:id]
    respond_to do |format|
      if @upgraded_user.update_attributes params[:user]
        format.html { redirect_to @upgraded_user }
      end
    end

此方法将像以前一样访问同一个用户表users_controller ,但它只是通过UpgradedUser模型来访问,因此将在保存时检查验证器是否存在名称和地址。

于 2012-11-24T23:10:02.557 回答
2

装饰器/实例扩展

您可以使用装饰器模式来完成此操作。实例扩展是另一种选择。因为我们使用的是 ruby​​,所以这里的实例扩展方式更有趣:

class User < ActiveRecord::Base
  validate :decorator_validations

  protected

  def decorator_validations
    # noop at this point
  end
end


module User::Upgrader

  def upgrade!
    # do whatever upgrade operation is needed
    save
  end

  protected

  def decorator_validations
    super
    self.errors.add(:address, 'needs to be provided') unless self.address.present?
  end
end

在你的控制器中:

def upgrade
  @user = User.find(params[:id])
  @user.extend User::Upgrader
  if @user.upgrade!
    # success case
  else
    # failure case
  end
end

命令模式

另一种方法是使用命令模式:

class UserUpgrader

 def initialize(user)
   @user = user
 end

 def call
   validate_upgrade_state
   return false if @user.errors.present?
   @user.update_attribute(:upgraded, true)
 end

 protected

 def validate_upgrade_state
   @user.errors.add(:address, 'must be present') unless @user.address.present?
 end
end

在这种情况下,您的控制器将如下所示:

@user = User.find(params[:id])
if UserUpgrader.new(@user).call
 # success case
else
 # failure case
end

这两种解决方案的好处是它们非常可测试。

于 2012-11-25T02:17:57.530 回答