簡體   English   中英

如何通過ransack gem搜索數組?

[英]How to search array through ransack gem?

我正在使用ransack gem在rails應用程序中進行搜索。 我需要在User表中搜索email_ids數組。

在洗劫時提到這個問題,我按照步驟將其添加到初始化文件夾ransack.rb

Ransack.configure do |config|
   {
    contained_within_array: :contained_within,
    contained_within_or_equals_array: :contained_within_or_equals,
    contains_array: :contains,
    contains_or_equals_array: :contains_or_equals,
    overlap_array: :overlap
   }.each do |rp, ap|
   config.add_predicate rp, arel_predicate: ap, wants_array: true
  end
end

在rails控制台中,如果我喜歡這樣:

a = User.search(email_contains_array: ['priti@gmail.com'])

它產生這樣的sql:

"SELECT \"users\".* FROM \"users\" WHERE \"users\".\"deleted_at\" IS NULL AND (\"users\".\"email\" >> '---\n- priti@gmail.com\n')"

並給出如下錯誤:

  User Load (1.8ms)  SELECT "users".* FROM "users" WHERE "users"."deleted_at" IS NULL AND ("users"."email" >> '---
- priti@gmail.com
')
 ActiveRecord::StatementInvalid: PG::UndefinedFunction: ERROR: operator does not exist: character varying >> unknown
 LINE 1: ...RE "users"."deleted_at" IS NULL AND ("users"."email" >> '---
                                                            ^
 HINT:  No operator matches the given name and argument type(s). You might need to add explicit type casts.
 : SELECT "users".* FROM "users" WHERE "users"."deleted_at" IS NULL AND ("users"."email" >> '---
 - priti@gmail.com
')

預期是這個查詢:

SELECT "users".* FROM "users"  WHERE ("users"."roles" @> '{"3","4"}')

我做錯了什么?

我遇到了和你一樣的問題。 我正在使用Rails 5,我需要在User表中搜索一組roles

您似乎已經在gemfile中添加了postgres_ext gem ,但如果您在Rails 5應用程序中使用它,則會出現一些問題。

因此,您可以自行選擇在Arel Node中添加contain查詢,而不是使用postgres_ext gem

如果您使用的是其他版本的Rails,我認為它的效果也很好。

我有一個User模型和一個數組屬性roles 我想要做的是使用ransack來搜索roles 這和你的情況一樣。

ransack無法搜索數組。 但是PostgresSQL可以像這樣搜索數組:

User.where("roles @> ?", '{admin}').to_sql)

它產生這樣的SQL查詢:

SELECT "users".* FROM "users" WHERE "users"."deleted_at" IS NULL AND (roles @> '{admin}')

所以我想要做的是在Arel Nodes添加一個類似的包含查詢

你能做到這樣

# app/config/initializers/arel.rb

require 'arel/nodes/binary'
require 'arel/predications'
require 'arel/visitors/postgresql'

module Arel
  class Nodes::ContainsArray < Arel::Nodes::Binary
    def operator
      :"@>"
    end
  end

  class Visitors::PostgreSQL
    private

    def visit_Arel_Nodes_ContainsArray(o, collector)
      infix_value o, collector, ' @> '
    end
  end

  module Predications
    def contains(other)
      Nodes::ContainsArray.new self, Nodes.build_quoted(other, self)
    end
  end
end

因為你可以自定義ransack謂詞,所以add contains Ransack謂詞,如下所示

# app/config/initializers/ransack.rb

Ransack.configure do |config|
  config.add_predicate 'contains',
    arel_predicate: 'contains',
    formatter: proc { |v| "{#{v}}" },
    validator: proc { |v| v.present? },
    type: :string
end

完成!

現在,您可以搜索數組:

User.ransack(roles_contains: 'admin')

SQL查詢將如下所示:

SELECT \"users\".* FROM \"users\" WHERE \"users\".\"deleted_at\" IS NULL AND (\"users\".\"roles\" @> '{[\"admin\"]}')

是啊!

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM