Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

validate presence of has_and_belongs_to_many

Hi i'm using a has_and_belongs_to_many in a model. I want set the valitor of presence for kinds. and set the max number of kinds per core to 3

class Core < ActiveRecord::Base
  has_and_belongs_to_many :kinds, :foreign_key => 'core_id', :association_foreign_key => 'kind_id'
end

how can i do?

thanks

like image 637
Luca Romagnoli Avatar asked Jan 28 '10 21:01

Luca Romagnoli


2 Answers

validate :require_at_least_one_kind
validate :limit_to_three_kinds

private

def require_at_least_one_kind
  if kinds.count == 0
    errors.add_to_base "Please select at least one kind"
  end
end

def limit_to_three_kinds
  if kinds.count > 3
    errors.add_to_base "No more than 3 kinds, please"
  end
end
like image 106
Alex Reisner Avatar answered Oct 17 '22 01:10

Alex Reisner


You could try something like this (tested on Rails 2.3.4):

class Core < ActiveRecord::Base
  has_and_belongs_to_many :kinds, :foreign_key => 'core_id', :association_foreign_key => 'kind_id'
  validate :maximum_three_kinds
  validate :minimum_one_kind

  def minimum_one_kind
    errors.add(:kinds, "must total at least one") if (kinds.length < 1)
  end

  def maximum_three_kinds
    errors.add(:kinds, "must not total more than three") if (kinds.length > 3)
  end
end

... which works in the following way:

require 'test_helper'

class CoreTest < ActiveSupport::TestCase

  test "a Core may have kinds" do
    core = Core.new
    3.times { core.kinds << Kind.new }

    assert(core.save)
  end

  test "a Core may have no more than 3 kinds" do
    core = Core.new
    4.times { core.kinds << Kind.new }
    core.save

    assert_equal(1, core.errors.length)
    assert_not_nil(core.errors['kinds'])
  end

  test "a Core must have at least one kind" do
    core = Core.new
    core.save

    assert_equal(1, core.errors.length)
    assert_not_nil(core.errors['kinds'])
  end
end

Obviously the above isn't particularly DRY or production-ready, but you get the idea.

like image 2
Duncan Bayne Avatar answered Oct 17 '22 01:10

Duncan Bayne