如何通过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\"]}')
是啊!
- rails – 如何在保存后刷新关联
- 在尝试保存大(ish)整数值时,获取指示编号的错误“超出ActiveRecord :: Type :: Integer且限制为4”的范围
- 当我尝试做rake db:migrate时,我收到一个错误:ActiveRecord :: NODatabaseError角色“ubuntu”不存在
- ActiveRecord :: Relation对象如何调用类方法
- 在IRB上需要ActiveRecord – Ruby(NO Rails)
- Rails 2.3.x – 这个ActiveRecord范围如何工作?
- 错误:缺少表“self”的FROM子句条目(Rails 4 Active Record事务,postgresql 9.4))
- 向ON子句添加参数的任何可能方法都包括在rails上的左连接?
- Rails计数记录而不再查询