Как мне динамически связать, где пункты в рельсах? - PullRequest
0 голосов
/ 12 сентября 2018

У меня есть модель Видео и модель Теги. Теги HABTM Видео и видео HABTM Метки.

class Tag < ApplicationRecord
  has_and_belongs_to_many :videos
  scope :with_tag, -> (id) { where(id: id) }
end

class Video < ApplicationRecord
  has_and_belongs_to_many :tags
end

Я хочу знать, как создать запрос, по которому от пользователя поступит неизвестное количество тегов. Поэтому я могу сказать: «Покажите мне все видео с тегами (123 И 124 И 125)». Конечная цель заключается в том, что я не знаю, сколько тегов будет передавать пользователь, поэтому мне нужно знать, как создать цепочку областей действия и, в конечном итоге, иметь возможность передать все это в Ransack для выполнения текстового поиска на некоторые поля результирующих видео. Как мне это сделать?

ОБНОВЛЕНИЕ: я хочу знать, как сделать это с Rails и ActiveRecord. Я знаю, как я могу сделать это с помощью SQL. Но выполнение этого только с помощью SQL не позволит мне передать полученное отношение в Ransack.

1 Ответ

0 голосов
/ 12 сентября 2018

Я создал несколько связанных таблиц для написания этого ответа и проверил запрос по данным, которые есть в таблице. Итак, я собираю посты, в которых есть теги 1 и 2.

В SQL:

rails-test_development=# select id from posts;
 id
----
  1
  2
  3
(3 rows)

rails-test_development=# select id from tags;
 id
----
  2
  3
  4
  5
  6
(5 rows)

rails-test_development=# select post_id, tag_id from posts_tags;
 post_id | tag_id
---------+--------
       1 |      2
       2 |      3
       2 |      2
       2 |      1
       3 |      1
       3 |      2
       1 |      4
(7 rows)

rails-test_development=# WITH posts_tags_cte AS (
rails-test_development(#         SELECT post_id, array_agg(tag_id) as tags
rails-test_development(#         FROM posts_tags
rails-test_development(#         WHERE tag_id in (1, 2)
rails-test_development(#         GROUP BY post_id
rails-test_development(# )
rails-test_development-# SELECT posts.id FROM posts_tags_cte JOIN posts ON posts.id = posts_tags_cte.post_id
rails-test_development-# WHERE posts_tags_cte.tags @> array[1, 2]::int8[];
 id
----
  3
  2
(2 rows)

Сейчас в Rails:

rails-test$ rails c
Running via Spring preloader in process 7361
Loading development environment (Rails 5.2.1)
2.5.1 :001 > Post.by_tag([1,2]).first.attributes
  Post Load (1.7ms)  SELECT  "posts".* FROM     (WITH posts_tags_cte AS (
      SELECT post_id, array_agg(tag_id) as tags
      FROM posts_tags
      WHERE tag_id in (1,2)
      GROUP BY post_id
    )
    SELECT posts.* FROM posts_tags_cte JOIN posts ON posts.id = posts_tags_cte.post_id
    WHERE posts_tags_cte.tags @> array[1,2]::int8[]) as posts
 ORDER BY "posts"."id" ASC LIMIT $1  [["LIMIT", 1]]
 => {"id"=>2, "name"=>"Postppp", "title"=>"Post", "content"=>"I don't know", "created_at"=>Sun, 09 Sep 2018 09:48:33 UTC +00:00, "updated_at"=>Sun, 09 Sep 2018 09:48:33 UTC +00:00, "month_and_year"=>Sun, 09 Sep 2018}
2.5.1 :002 > Post.by_tag([1,2]).last.attributes
  Post Load (1.3ms)  SELECT  "posts".* FROM     (WITH posts_tags_cte AS (
      SELECT post_id, array_agg(tag_id) as tags
      FROM posts_tags
      WHERE tag_id in (1,2)
      GROUP BY post_id
    )
    SELECT posts.* FROM posts_tags_cte JOIN posts ON posts.id = posts_tags_cte.post_id
    WHERE posts_tags_cte.tags @> array[1,2]::int8[]) as posts
 ORDER BY "posts"."id" DESC LIMIT $1  [["LIMIT", 1]]
 => {"id"=>3, "name"=>"Post A", "title"=>"Post title", "content"=>"Lorem Ipsum is simply dummy text of the printing and typesetting industry.", "created_at"=>Sun, 09 Sep 2018 09:52:57 UTC +00:00, "updated_at"=>Sun, 09 Sep 2018 09:52:57 UTC +00:00, "month_and_year"=>Sun, 09 Sep 2018}
2.5.1 :003 >

И я определил сферу как:

class Post < ApplicationRecord
  has_and_belongs_to_many :tags

  scope :by_tag, -> (tag_ids) {

    sql = sanitize_sql_array [<<-SQL, tag_ids, tag_ids]
    (WITH posts_tags_cte AS (
      SELECT post_id, array_agg(tag_id) as tags
      FROM posts_tags
      WHERE tag_id in (?)
      GROUP BY post_id
    )
    SELECT posts.* FROM posts_tags_cte JOIN posts ON posts.id = posts_tags_cte.post_id
    WHERE posts_tags_cte.tags @> array[?]::int8[]) as posts
    SQL

    from(sql)
  }
end

Я получил помощь от этого ответа .

...