如何通过ransack gem搜索数组?

时间:2016-07-16 10:14:56

标签: ruby-on-rails ruby postgresql activerecord ransack

我使用ransack gem在rails应用程序中进行搜索。我需要在email_ids表格中搜索User数组。

在洗劫时提到this问题,我按照步骤将其添加到初始化文件夹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"}')

我做错了什么?

2 个答案:

答案 0 :(得分:7)

我遇到了和你一样的问题。我使用的是Rails 5,我需要在roles表中搜索User数组

您的gem文件中似乎已经添加了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

中添加一个类似的包含查询

你可以this way

# 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谓词, 所以添加contains Ransack谓词,如this

# 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\"]}')

呀!

答案 1 :(得分:0)

我使用可洗劫的范围实现了这一点。

其中 Author 有一列“affiliation”,由字符串数组组成。

scope :affiliation_includes, ->(str) {where("array_to_string(affiliation,',') ILIKE ?", "%#{str}%")}
def self.ransackable_scopes(auth_object=nil) 
    [:affiliation_includes]
  end

然后下面的作品

Author.ransack(affiliation_includes:'dresden')

与使用 postgres '@>' 相比,它的优势在于它可以匹配数组中的子字符串,因为在 where 命令中使用了 %(去掉整个字符串匹配的 %)。它也不区分大小写(使用 LIKE 而不是 ILIKE 区分大小写。