6

我有一个很常见的habtm关系:

Photo has_and_belongs_to_many :tags
Tag has_and_belongs_to_many :photos

在我的照片模型中,我有一个“带有标签”的方法,我用它来查找一张用给定的 tag_ids 集标记的照片。此查询只需要匹配具有所有给定标签的照片,但不考虑是否存在任何其他标签。这是我的方法:

def self.with_terms( array )
  select('distinct photos.*').joins(:tags).where('tags.id' => array).group("photos." + self.column_names.join(', photos.')).having("count(*) = #{array.size}")
end

这按预期工作。

现在,为了更好地将它与我正在使用的其他一些库集成,我需要在 Arel 中重新编写它。(让它成为一个 Arel 节点?,不确定你通常称之为什么)。

我一直在尝试这个,但老实说我以前从未尝试过使用 Arel,所以我有点迷茫。我一直在控制台中进行试验并尝试过:

t = Photo.arel_table
q = t.join(:tags).on(t[:tags_id].in(array))
Photo.where(q)

但是,(1)我认为首先不是q正确的查询,并且(2)它创建一个Arel::SelectManager,当传递给 where 调用 raises 时Cannot visit Arel::SelectManager。所以,显然我做错了。

更新:在这里更具体一点,我希望返回一个 Arel 节点,因为我正在使用一个 gem(ransack),它希望您将 Arel 节点传递给它以用于搜索方法。Ransack 将把这个 Arel 节点与其他节点链接起来以生成复杂的搜索查询。

Arel 大师能告诉我如何正确地做到这一点吗?

4

1 回答 1

10

很难找到好的 Arel 文档,但是@Philip C整理了一些有用的幻灯片,在他对这个问题的回答中引用了这些幻灯片。

以下应该是您正在寻找的内容:

photos = Arel::Table.new(:photos)
tags = Arel::Table.new(:tags)
photo_tags = Arel::Table.new(:photo_tags)

q = photos[:id].in(
   photos.project(photos[:id])
  .join(photo_tags).on(photos[:id].eql(photo_tags[:photo_id]))
  .join(tags).on(photo_tags[:tag_id].eql(tags[:id]))
  .where(tags[:id].in(array))
  .group(photos.columns)
  .having(tags[:id].count.eq(array.length))
)

这会产生一个Arel::Nodes::In您应该能够直接使用的实例,如Photo.where(q).


更新:

在查看文档和一些 ransack 的源之后,似乎没有任何自然的方法来定义涉及子查询的自定义谓词,这在您的情况下是必要的(因为谓词必须适合where子句)。解决此问题的一种方法可能是利用:formatter谓词使用的优势,如下所示:

Ransack.configure do |config|
  config.add_predicate 'with_tag_ids',
                   :arel_predicate => 'in',
                   :formatter => proc {|tag_ids| tags_subquery(tag_ids) },
                   :validator => proc {|v| v.present?},
                   :compounds => true
end

您可以定义tags_subquery(tag_ids)为生成上述 arel 节点的方法,但在返回之前替换arraytag_ids并调用.to_sql它(格式化程序需要返回字符串,而不是节点)。

我还没有尝试过,所以如果它有效,我会很高兴!

于 2012-11-18T23:51:04.707 回答