Skip to content

Commit 6d10abb

Browse files
committed
[MODEL] Fixed a problem where Hashie::Mash#min and #max returned unexpected values
While people can define names for aggregations such as `min` and `max`, these methods are defined in `Enumerable#min` and `Enumerable#max`, and return an unexpected value: result.aggregations # => {"price"=>{"doc_count"=>9428, "min"=>{"value"=>1.0}, "max"=>{"value"=>260000.0}}} result.aggregations.price # => {"doc_count"=>9428, "min"=>{"value"=>1.0}, "max"=>{"value"=>260000.0}} result.aggregations.price.min # => ["doc_count", 9428] Therefore, the whole Hash with aggregations is being traversed, in a wrapper class, and a `min` and `max` method is defined for Hashie::Mash instances, because these names are too common. The wrapper class can have its uses further down the line, so a dedicated unit test file has been added as well. Closes elastic#568 Related: elastic/elasticsearch-ruby#306
1 parent 31d04c7 commit 6d10abb

File tree

5 files changed

+98
-7
lines changed

5 files changed

+98
-7
lines changed

elasticsearch-model/lib/elasticsearch/model.rb

+1
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
require 'elasticsearch/model/response/results'
3232
require 'elasticsearch/model/response/records'
3333
require 'elasticsearch/model/response/pagination'
34+
require 'elasticsearch/model/response/aggregations'
3435
require 'elasticsearch/model/response/suggestions'
3536

3637
require 'elasticsearch/model/ext/active_record'

elasticsearch-model/lib/elasticsearch/model/response.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@ def shards
6969
# Returns a Hashie::Mash of the aggregations
7070
#
7171
def aggregations
72-
response['aggregations'] ? Hashie::Mash.new(response['aggregations']) : nil
72+
Aggregations.new(response['aggregations'])
7373
end
7474

7575
# Returns a Hashie::Mash of the suggestions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
module Elasticsearch
2+
module Model
3+
module Response
4+
5+
class Aggregations < Hashie::Mash
6+
def initialize(attributes={})
7+
__redefine_enumerable_methods super(attributes)
8+
end
9+
10+
# Fix the problem of Hashie::Mash returning unexpected values for `min` and `max` methods
11+
#
12+
# People can define names for aggregations such as `min` and `max`, but these
13+
# methods are defined in `Enumerable#min` and `Enumerable#max`
14+
#
15+
# { foo: 'bar' }.min
16+
# # => [:foo, "bar"]
17+
#
18+
# Therefore, any Hashie::Mash instance value has the `min` and `max`
19+
# methods redefined to return the real value
20+
#
21+
def __redefine_enumerable_methods(h)
22+
if h.respond_to?(:each_pair)
23+
h.each_pair { |k, v| v = __redefine_enumerable_methods(v) }
24+
end
25+
if h.is_a?(Hashie::Mash)
26+
class << h
27+
define_method(:min) { self[:min] }
28+
define_method(:max) { self[:max] }
29+
end
30+
end
31+
end
32+
end
33+
34+
end
35+
end
36+
end

elasticsearch-model/test/integration/active_record_basic_test.rb

+14-6
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ class ActiveRecordBasicIntegrationTest < Elasticsearch::Test::IntegrationTestCas
1212
create_table :articles do |t|
1313
t.string :title
1414
t.string :body
15+
t.integer :clicks, :default => 0
1516
t.datetime :created_at, :default => 'NOW()'
1617
end
1718
end
@@ -24,24 +25,25 @@ class ::Article < ActiveRecord::Base
2425
mapping do
2526
indexes :title, type: 'string', analyzer: 'snowball'
2627
indexes :body, type: 'string'
28+
indexes :clicks, type: 'integer'
2729
indexes :created_at, type: 'date'
2830
end
2931
end
3032

3133
def as_indexed_json(options = {})
3234
attributes
3335
.symbolize_keys
34-
.slice(:title, :body, :created_at)
36+
.slice(:title, :body, :clicks, :created_at)
3537
.merge(suggest_title: title)
3638
end
3739
end
3840

3941
Article.delete_all
4042
Article.__elasticsearch__.create_index! force: true
4143

42-
::Article.create! title: 'Test', body: ''
43-
::Article.create! title: 'Testing Coding', body: ''
44-
::Article.create! title: 'Coding', body: ''
44+
::Article.create! title: 'Test', body: '', clicks: 1
45+
::Article.create! title: 'Testing Coding', body: '', clicks: 2
46+
::Article.create! title: 'Coding', body: '', clicks: 3
4547

4648
Article.__elasticsearch__.refresh_index!
4749
end
@@ -217,11 +219,17 @@ def as_indexed_json(options = {})
217219

218220
should "allow dot access to response" do
219221
response = Article.search query: { match: { title: { query: 'test' } } },
220-
aggregations: { dates: { date_histogram: { field: 'created_at', interval: 'hour' } } },
222+
aggregations: {
223+
dates: { date_histogram: { field: 'created_at', interval: 'hour' } },
224+
clicks: { global: {}, aggregations: { min: { min: { field: 'clicks' } } } }
225+
},
221226
suggest: { text: 'tezt', title: { term: { field: 'title', suggest_mode: 'always' } } }
222227

223228
response.response.respond_to?(:aggregations)
224-
assert_equal 2, response.aggregations.dates.buckets.first.doc_count
229+
assert_equal 2, response.aggregations.dates.buckets.first.doc_count
230+
assert_equal 3, response.aggregations.clicks.doc_count
231+
assert_equal 1.0, response.aggregations.clicks.min.value
232+
assert_nil response.aggregations.clicks.max
225233

226234
response.response.respond_to?(:suggest)
227235
assert_equal 1, response.suggestions.title.first.options.size
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
require 'test_helper'
2+
3+
class Elasticsearch::Model::ResponseAggregationsTest < Test::Unit::TestCase
4+
context "Response aggregations" do
5+
class OriginClass
6+
def self.index_name; 'foo'; end
7+
def self.document_type; 'bar'; end
8+
end
9+
10+
RESPONSE = {
11+
'aggregations' => {
12+
'foo' => {'bar' => 10 },
13+
'price' => { 'doc_count' => 123,
14+
'min' => { 'value' => 1.0},
15+
'max' => { 'value' => 99 }
16+
}
17+
}
18+
}
19+
20+
setup do
21+
@search = Elasticsearch::Model::Searching::SearchRequest.new OriginClass, '*'
22+
@search.stubs(:execute!).returns(RESPONSE)
23+
end
24+
25+
should "access the aggregations" do
26+
@search.expects(:execute!).returns(RESPONSE)
27+
28+
response = Elasticsearch::Model::Response::Response.new OriginClass, @search
29+
assert_respond_to response, :aggregations
30+
assert_kind_of Elasticsearch::Model::Response::Aggregations, response.aggregations
31+
assert_kind_of Hashie::Mash, response.aggregations.foo
32+
assert_equal 10, response.aggregations.foo.bar
33+
end
34+
35+
should "properly return min and max values" do
36+
@search.expects(:execute!).returns(RESPONSE)
37+
38+
response = Elasticsearch::Model::Response::Response.new OriginClass, @search
39+
40+
assert_equal 123, response.aggregations.price.doc_count
41+
assert_equal 1, response.aggregations.price.min.value
42+
assert_equal 99, response.aggregations.price.max.value
43+
end
44+
45+
end
46+
end

0 commit comments

Comments
 (0)