Подтвердить что ты не робот

Как настроить привязные области на основе отношений в Mongoid

Проблема исправлена ​​... Оказалось, что был активен активный метод записи, написанный, теперь все работает как ожидалось

Я пытаюсь настроить области, чтобы я мог сделать вызов, который выглядит как

Competitor.of_type(type).at_event(event)

который вернет всех участников типа, которые участвовали в мероприятии Мои модели выглядят примерно как

class Competitor < Competitor
  belongs_to :type
  has_and_belongs_to_many :events
  scope :at_event, ->(event) {where(:event_ids.in => event.competitor_ids)}
  scope :of_type, ->(type) where(:type_id => type.id)                
end

Следующие работы (возврат мангоидных критериев)

Competitor.of_type(type)
Competitor.at_event(event)

Но когда я их связываю, он печатает что-то похожее на это:

#<Competitor:0x00000109e2b210>
#<Competitor:0x00000109e2ab08>
-------=-=------------------------------------
=> #<Mongoid::Criteria
selector: {},
options:  {},
class:    Competitor,
embedded: false>

Для каждого из типов Competitor.of_type (type) (критерии первого набора) есть запись конкурента, и если я запустил .count по запросу, я получаю общее количество участников в базе данных.

В верхней части документации по mongoid для областей, она говорит, что все области привязаны и могут применяться к ассоциациям, а затем обсуждается в разделе отношений. К сожалению, я не видел раздел отношений, я не мог найти единственную ссылку на область видимости в разделе основных отношений.

Я смог получить следующее, чтобы вернуть результаты, которые я хотел:

where(:id.in => event.competitor_ids).where(:type_id => type.id)

но если какая-либо часть запроса разделяется на отдельный метод или область, она терпит неудачу и дает результат, который я показал выше.

4b9b3361

Ответ 1

прицелы

Подобно Active Record, Mongoid позволяет вам определять области на вашем моделей в качестве удобства для фильтрации наборов результатов. Определены области действия на уровне класса, либо используя макрос области видимости, либо определяя класс методы, возвращающие объект критериев. Все области применения могут быть применены и к ассоциациям, раздел отношений.

Именованные области определяются на уровне класса с использованием макроса области и могут быть привязаны для создания наборов результатов в красивой DSL.

class Person
  include Mongoid::Document
  field :occupation, type: String
  field :age, type: Integer

  scope :rock_n_rolla, where(occupation: "Rockstar")
  scope :washed_up, where(:age.gt => 30)
  scope :over, ->(limit) { where(:age.gt => limit) }
end

# Find all the rockstars.
Person.rock_n_rolla

# Find all rockstars that should probably quit.
Person.washed_up.rock_n_rolla

# Find a criteria with Keith Richards in it.
Person.rock_n_rolla.over(60)

Обратите внимание, что определения оцениваются во время загрузки класса. Для оценки во время выполнения, вы захотите определить, используя proc или лямбда. В следующем примере первая дата устанавливается как дата загрузки класса, где вторая область задает дату в то время область действия называется.

scope :current, where(:start_date.lte => Date.today)
scope :current, -> { where(:start_date.lte => Date.today) }

методы класса

Для тех, кто предпочитает синтаксис стиля Data Mapper, методы класса, возвращающие критерии, также могут рассматриваться как области с возможностью привязки.

class Person
  include Mongoid::Document
  field :occupation, type: String
  field :age, type: Integer

  class << self

    def rock_n_rolla
      where(occupation: "Rockstar")
    end

    def washed_up
      where(:age.gt => 30)
    end

    def over(limit)
      where(:age.gt => limit)
    end
  end
end

# Find all the rockstars.
Person.rock_n_rolla

# Find all rockstars that should probably quit.
Person.washed_up.rock_n_rolla

# Find a criteria with Keith Richards in it.
Person.rock_n_rolla.over(60)

Именованные области и методы класса, которые возвращают критерии, могут быть соединены вместе - что красота монгольских мощных критериев API.

class Person
  include Mongoid::Document
  field :occupation, type: String
  field :age, type: Integer

  scope :washed_up, where(:age.gt => 30)
  scope :over, ->(limit) { where(:age.gt => limit) }

  def self.rock_n_rolla
    where(occupation: "Rockstar")
  end
end

# Same queries apply here as well.
Person.rock_n_rolla
Person.washed_up.rock_n_rolla
Person.rock_n_rolla.over(60)