1

我正在使用Toptal 的 Chewy gem来连接和查询我的 Elasticsearch,就像 ODM 一样。

我将Chewy与 Elasticsearch 6、Ruby on Rails 5.2 和 Active Record 一起使用。

我已经像这样定义了我的索引:

class OrdersIndex < Chewy::Index
  define_type Order.includes(:customer) do

    field :id, type: "keyword"

    field :customer do
      field :id, type: "keyword"
      field :name, type: "text"
      field :email, type: "keyword"
    end
  end
end

我的模型:

class Order < ApplicationRecord
  belongs_to :customer
end

这里的问题是,当我使用 Chewy 执行任何查询时,客户数据会被反序列化为哈希而不是对象,并且我不能使用点符号来访问嵌套数据。

results = OrdersIndex.query(query_string: { query: "test" })
results.first.id
# => "594d8e8b2cc640bb78bd115ae644637a1cc84dd460be6f69"

results.first.customer.name
# => NoMethodError: undefined method `name' for #<Hash:0x000000000931d928>

results.first.customer["name"]
# => "Frederique Schaefer"

如何使用点符号 ( result.customer.name) 访问嵌套关联?或者反序列化对象(例如结构)中的嵌套数据,这允许我使用点表示法?

4

2 回答 2

2

尝试使用

results = OrdersIndex.query(query_string: { query: "test" }).objects

它将查询结果转换为活动记录对象。所以点符号应该可以工作。如果要加载与上述结果的任何额外关联,可以在 Index 上使用 .load 方法。

如果您想将现有的 ES 嵌套对象转换为可使用点表示法访问,请尝试参考此答案。Open Struct 是在 ruby​​ 中完成任务的最佳方式。

无法对 ruby​​ 哈希使用点语法

另外,这个也有帮助

如果您需要 openStruct 为嵌套对象工作,请参阅此链接

于 2018-08-30T20:15:30.560 回答
0

将刚刚反序列化的结果转换为 JSON 字符串并使用 OpenStruct 再次对其进行反序列化object_class可能不是一个好主意,而且 CPU 成本很高。

我使用递归和 Ruby 的 native 以不同的方式解决了它Struct,保留了 Chewy gem 的惰性。

def convert_to_object(keys, values)
  schema = Struct.new(*keys.map(&:to_sym))
  object = schema.new(*values)

  object.each_pair do |key, value|
    if value.is_a?(Hash)
      object.send("#{key}=", convert_to_object(value.keys, value.values))
    end
  end

  object
end

OrdersIndex.query(query_string: { query: "test" }).lazy.map do |item|
  convert_to_object(item.attributes.keys, item.attributes.values)
end

convert_to_object获取一个键数组和另一个值,并从中创建一个结构。每当值数组项之一的类是哈希时,它就会递归地转换为结构,并传递哈希键和值。

为了呈现懒惰,那是 Chewy 最酷的部分,我用过Enumerator::LazyEnumerator#map. 将 ES 查询返回的每个值映射到convert_to_object函数中,使每个条目成为一个完整的结构。

该代码非常通用,适用于我拥有的每个索引。

于 2018-08-31T12:37:24.477 回答