gpt4 book ai didi

ruby-on-rails - 如何通过 ransack gem 搜索数组?

转载 作者:数据小太阳 更新时间:2023-10-29 08:19:42 24 4
gpt4 key购买 nike

我正在使用 ransack gem 在 Rails 应用程序中进行搜索。我需要在 User 表中搜索一组 email_ids

引用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"}')

我做错了什么?

最佳答案

我遇到了和你一样的问题。我正在使用 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

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

你能做到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\"]}')

是啊!

关于ruby-on-rails - 如何通过 ransack gem 搜索数组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38410076/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com