Rails habtm и поиск записи без ассоциации - PullRequest
12 голосов
/ 12 августа 2011

У меня есть 2 модели:

class User < ActiveRecord::Base
    has_and_belongs_to_many :groups
end

class Group < ActiveRecord::Base
    has_and_belongs_to_many :users
end

Я хочу сделать область действия (это важно - для эффективности и для возможности цепочки областей), которая возвращает пользователей, которые не принадлежатдо ЛЮБЫХ групп.После многих попыток я потерпел неудачу в создании метода вместо области видимости, что делает collect на User.all, что уродливо и ... не правильно.

Любая помощь?2-й вопрос: мне удалось создать область, которая возвращает пользователей, принадлежащих к какой-либо из указанных групп (представленных в виде массива идентификаторов).

scope :in_groups, lambda { |g|
        {
          :joins      => :groups,
          :conditions => {:groups => {:id => g}},
          :select     => "DISTINCT `users`.*" # kill duplicates
        }
      }

Может ли она быть лучше / красивее?(Использование Rails 3.0.9)

Ответы [ 2 ]

17 голосов
/ 12 августа 2011

Ваша таблица неявного соединения была бы названа groups_users в соответствии с соглашениями об именах.Подтвердите это один раз в вашей БД.Предполагая, что это:

В более новой версии Rails:

scope :not_in_any_group -> {
    joins("LEFT JOIN groups_users ON users.id = groups_users.user_id")
    .where("groups_users.user_id IS NULL")
}

Для более старых версий Rails:

scope :not_in_any_group, {
    :joins      => "LEFT JOIN groups_users ON users.id = groups_users.user_id",
    :conditions => "groups_users.user_id IS NULL",
    :select     => "DISTINCT users.*"
}
3 голосов
/ 20 июля 2015

Если вы преобразуете из HABTM в has_many через (более гибкую) связь, то вы можете использовать что-то вроде этого:

class Group < ActiveRecord::Base
  has_many :groups_users, dependent: :destroy
  has_many :users, through: :groups_users, uniq: true

  scope :in_groups, -> { includes(:groups_users).where(groups_users: {group_id: nil}) }
end

class User < ActiveRecord::Base
  has_many :groups_users, dependent: :destroy
  has_many :groups, through: :groups_users
end

class GroupsUser < ActiveRecord::Base
  belongs_to :group
  belongs_to :user
end
...