gojee-sunspot 2.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (176) hide show
  1. data/.gitignore +12 -0
  2. data/Gemfile +5 -0
  3. data/History.txt +252 -0
  4. data/LICENSE +18 -0
  5. data/Rakefile +13 -0
  6. data/TODO +13 -0
  7. data/lib/light_config.rb +40 -0
  8. data/lib/sunspot.rb +579 -0
  9. data/lib/sunspot/adapters.rb +265 -0
  10. data/lib/sunspot/batcher.rb +62 -0
  11. data/lib/sunspot/class_set.rb +23 -0
  12. data/lib/sunspot/composite_setup.rb +202 -0
  13. data/lib/sunspot/configuration.rb +53 -0
  14. data/lib/sunspot/data_extractor.rb +50 -0
  15. data/lib/sunspot/dsl.rb +5 -0
  16. data/lib/sunspot/dsl/adjustable.rb +47 -0
  17. data/lib/sunspot/dsl/field_group.rb +57 -0
  18. data/lib/sunspot/dsl/field_query.rb +327 -0
  19. data/lib/sunspot/dsl/fields.rb +103 -0
  20. data/lib/sunspot/dsl/fulltext.rb +243 -0
  21. data/lib/sunspot/dsl/function.rb +27 -0
  22. data/lib/sunspot/dsl/functional.rb +44 -0
  23. data/lib/sunspot/dsl/more_like_this_query.rb +56 -0
  24. data/lib/sunspot/dsl/paginatable.rb +32 -0
  25. data/lib/sunspot/dsl/query_facet.rb +36 -0
  26. data/lib/sunspot/dsl/restriction.rb +25 -0
  27. data/lib/sunspot/dsl/restriction_with_near.rb +160 -0
  28. data/lib/sunspot/dsl/scope.rb +217 -0
  29. data/lib/sunspot/dsl/search.rb +30 -0
  30. data/lib/sunspot/dsl/standard_query.rb +123 -0
  31. data/lib/sunspot/field.rb +193 -0
  32. data/lib/sunspot/field_factory.rb +129 -0
  33. data/lib/sunspot/indexer.rb +136 -0
  34. data/lib/sunspot/query.rb +11 -0
  35. data/lib/sunspot/query/abstract_field_facet.rb +52 -0
  36. data/lib/sunspot/query/bbox.rb +15 -0
  37. data/lib/sunspot/query/boost_query.rb +24 -0
  38. data/lib/sunspot/query/common_query.rb +96 -0
  39. data/lib/sunspot/query/composite_fulltext.rb +36 -0
  40. data/lib/sunspot/query/connective.rb +206 -0
  41. data/lib/sunspot/query/date_field_facet.rb +14 -0
  42. data/lib/sunspot/query/dismax.rb +132 -0
  43. data/lib/sunspot/query/field_facet.rb +41 -0
  44. data/lib/sunspot/query/field_group.rb +36 -0
  45. data/lib/sunspot/query/filter.rb +38 -0
  46. data/lib/sunspot/query/function_query.rb +52 -0
  47. data/lib/sunspot/query/geo.rb +53 -0
  48. data/lib/sunspot/query/geofilt.rb +16 -0
  49. data/lib/sunspot/query/highlighting.rb +62 -0
  50. data/lib/sunspot/query/more_like_this.rb +61 -0
  51. data/lib/sunspot/query/more_like_this_query.rb +12 -0
  52. data/lib/sunspot/query/pagination.rb +42 -0
  53. data/lib/sunspot/query/query_facet.rb +16 -0
  54. data/lib/sunspot/query/restriction.rb +262 -0
  55. data/lib/sunspot/query/scope.rb +9 -0
  56. data/lib/sunspot/query/sort.rb +109 -0
  57. data/lib/sunspot/query/sort_composite.rb +34 -0
  58. data/lib/sunspot/query/standard_query.rb +16 -0
  59. data/lib/sunspot/query/text_field_boost.rb +17 -0
  60. data/lib/sunspot/schema.rb +151 -0
  61. data/lib/sunspot/search.rb +9 -0
  62. data/lib/sunspot/search/abstract_search.rb +281 -0
  63. data/lib/sunspot/search/date_facet.rb +35 -0
  64. data/lib/sunspot/search/facet_row.rb +27 -0
  65. data/lib/sunspot/search/field_facet.rb +88 -0
  66. data/lib/sunspot/search/field_group.rb +32 -0
  67. data/lib/sunspot/search/group.rb +50 -0
  68. data/lib/sunspot/search/highlight.rb +38 -0
  69. data/lib/sunspot/search/hit.rb +150 -0
  70. data/lib/sunspot/search/hit_enumerable.rb +72 -0
  71. data/lib/sunspot/search/more_like_this_search.rb +31 -0
  72. data/lib/sunspot/search/paginated_collection.rb +57 -0
  73. data/lib/sunspot/search/query_facet.rb +67 -0
  74. data/lib/sunspot/search/standard_search.rb +21 -0
  75. data/lib/sunspot/session.rb +262 -0
  76. data/lib/sunspot/session_proxy.rb +95 -0
  77. data/lib/sunspot/session_proxy/abstract_session_proxy.rb +29 -0
  78. data/lib/sunspot/session_proxy/class_sharding_session_proxy.rb +66 -0
  79. data/lib/sunspot/session_proxy/id_sharding_session_proxy.rb +89 -0
  80. data/lib/sunspot/session_proxy/master_slave_session_proxy.rb +43 -0
  81. data/lib/sunspot/session_proxy/multicore_session_proxy.rb +67 -0
  82. data/lib/sunspot/session_proxy/sharding_session_proxy.rb +222 -0
  83. data/lib/sunspot/session_proxy/silent_fail_session_proxy.rb +42 -0
  84. data/lib/sunspot/session_proxy/thread_local_session_proxy.rb +37 -0
  85. data/lib/sunspot/setup.rb +350 -0
  86. data/lib/sunspot/text_field_setup.rb +29 -0
  87. data/lib/sunspot/type.rb +393 -0
  88. data/lib/sunspot/util.rb +252 -0
  89. data/lib/sunspot/version.rb +3 -0
  90. data/script/console +10 -0
  91. data/spec/api/adapters_spec.rb +33 -0
  92. data/spec/api/batcher_spec.rb +112 -0
  93. data/spec/api/binding_spec.rb +50 -0
  94. data/spec/api/class_set_spec.rb +24 -0
  95. data/spec/api/hit_enumerable_spec.rb +47 -0
  96. data/spec/api/indexer/attributes_spec.rb +149 -0
  97. data/spec/api/indexer/batch_spec.rb +72 -0
  98. data/spec/api/indexer/dynamic_fields_spec.rb +42 -0
  99. data/spec/api/indexer/fixed_fields_spec.rb +57 -0
  100. data/spec/api/indexer/fulltext_spec.rb +43 -0
  101. data/spec/api/indexer/removal_spec.rb +53 -0
  102. data/spec/api/indexer/spec_helper.rb +1 -0
  103. data/spec/api/indexer_spec.rb +14 -0
  104. data/spec/api/query/advanced_manipulation_examples.rb +35 -0
  105. data/spec/api/query/connectives_examples.rb +189 -0
  106. data/spec/api/query/dsl_spec.rb +18 -0
  107. data/spec/api/query/dynamic_fields_examples.rb +165 -0
  108. data/spec/api/query/faceting_examples.rb +397 -0
  109. data/spec/api/query/fulltext_examples.rb +313 -0
  110. data/spec/api/query/function_spec.rb +79 -0
  111. data/spec/api/query/geo_examples.rb +68 -0
  112. data/spec/api/query/group_spec.rb +32 -0
  113. data/spec/api/query/highlighting_examples.rb +245 -0
  114. data/spec/api/query/more_like_this_spec.rb +140 -0
  115. data/spec/api/query/ordering_pagination_examples.rb +116 -0
  116. data/spec/api/query/scope_examples.rb +275 -0
  117. data/spec/api/query/spatial_examples.rb +27 -0
  118. data/spec/api/query/spec_helper.rb +1 -0
  119. data/spec/api/query/standard_spec.rb +29 -0
  120. data/spec/api/query/text_field_scoping_examples.rb +30 -0
  121. data/spec/api/query/types_spec.rb +20 -0
  122. data/spec/api/search/dynamic_fields_spec.rb +33 -0
  123. data/spec/api/search/faceting_spec.rb +360 -0
  124. data/spec/api/search/highlighting_spec.rb +69 -0
  125. data/spec/api/search/hits_spec.rb +131 -0
  126. data/spec/api/search/paginated_collection_spec.rb +36 -0
  127. data/spec/api/search/results_spec.rb +72 -0
  128. data/spec/api/search/search_spec.rb +23 -0
  129. data/spec/api/search/spec_helper.rb +1 -0
  130. data/spec/api/session_proxy/class_sharding_session_proxy_spec.rb +85 -0
  131. data/spec/api/session_proxy/id_sharding_session_proxy_spec.rb +30 -0
  132. data/spec/api/session_proxy/master_slave_session_proxy_spec.rb +41 -0
  133. data/spec/api/session_proxy/sharding_session_proxy_spec.rb +77 -0
  134. data/spec/api/session_proxy/silent_fail_session_proxy_spec.rb +24 -0
  135. data/spec/api/session_proxy/spec_helper.rb +9 -0
  136. data/spec/api/session_proxy/thread_local_session_proxy_spec.rb +39 -0
  137. data/spec/api/session_spec.rb +232 -0
  138. data/spec/api/spec_helper.rb +3 -0
  139. data/spec/api/sunspot_spec.rb +29 -0
  140. data/spec/ext.rb +11 -0
  141. data/spec/helpers/indexer_helper.rb +17 -0
  142. data/spec/helpers/integration_helper.rb +8 -0
  143. data/spec/helpers/mock_session_helper.rb +13 -0
  144. data/spec/helpers/query_helper.rb +26 -0
  145. data/spec/helpers/search_helper.rb +68 -0
  146. data/spec/integration/dynamic_fields_spec.rb +57 -0
  147. data/spec/integration/faceting_spec.rb +251 -0
  148. data/spec/integration/field_grouping_spec.rb +66 -0
  149. data/spec/integration/geospatial_spec.rb +85 -0
  150. data/spec/integration/highlighting_spec.rb +44 -0
  151. data/spec/integration/indexing_spec.rb +55 -0
  152. data/spec/integration/keyword_search_spec.rb +317 -0
  153. data/spec/integration/local_search_spec.rb +64 -0
  154. data/spec/integration/more_like_this_spec.rb +43 -0
  155. data/spec/integration/scoped_search_spec.rb +354 -0
  156. data/spec/integration/stored_fields_spec.rb +12 -0
  157. data/spec/integration/test_pagination.rb +43 -0
  158. data/spec/integration/unicode_spec.rb +15 -0
  159. data/spec/mocks/adapters.rb +32 -0
  160. data/spec/mocks/blog.rb +3 -0
  161. data/spec/mocks/comment.rb +21 -0
  162. data/spec/mocks/connection.rb +126 -0
  163. data/spec/mocks/mock_adapter.rb +30 -0
  164. data/spec/mocks/mock_class_sharding_session_proxy.rb +24 -0
  165. data/spec/mocks/mock_record.rb +52 -0
  166. data/spec/mocks/mock_sharding_session_proxy.rb +15 -0
  167. data/spec/mocks/photo.rb +11 -0
  168. data/spec/mocks/post.rb +86 -0
  169. data/spec/mocks/super_class.rb +2 -0
  170. data/spec/mocks/user.rb +13 -0
  171. data/spec/spec_helper.rb +40 -0
  172. data/sunspot.gemspec +42 -0
  173. data/tasks/rdoc.rake +27 -0
  174. data/tasks/schema.rake +19 -0
  175. data/tasks/todo.rake +4 -0
  176. metadata +409 -0
@@ -0,0 +1,35 @@
1
+ module Sunspot
2
+ module Search
3
+ class DateFacet
4
+ def initialize(field, search, options)
5
+ @field, @search, @options = field, search, options
6
+ end
7
+
8
+ def field_name
9
+ @field.name
10
+ end
11
+
12
+ def rows
13
+ @rows ||=
14
+ begin
15
+ data = @search.facet_response['facet_dates'][@field.indexed_name]
16
+ gap = (@options[:time_interval] || 86400).to_i
17
+ rows = []
18
+ data.each_pair do |value, count|
19
+ if value =~ /^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}Z$/
20
+ start_time = @field.cast(value)
21
+ end_time = start_time + gap
22
+ rows << FacetRow.new(start_time..end_time, count, self)
23
+ end
24
+ end
25
+ if @options[:sort] == :count
26
+ rows.sort! { |lrow, rrow| rrow.count <=> lrow.count }
27
+ else
28
+ rows.sort! { |lrow, rrow| lrow.value.first <=> rrow.value.first }
29
+ end
30
+ rows
31
+ end
32
+ end
33
+ end
34
+ end
35
+ end
@@ -0,0 +1,27 @@
1
+ module Sunspot
2
+ module Search
3
+ class FacetRow
4
+ attr_reader :value, :count
5
+ attr_writer :instance #:nodoc:
6
+
7
+ def initialize(value, count, facet) #:nodoc:
8
+ @value, @count, @facet = value, count, facet
9
+ end
10
+
11
+ #
12
+ # Return the instance referenced by this facet row. Only valid for field
13
+ # facets whose fields are defined with the :references key.
14
+ #
15
+ def instance
16
+ if !defined?(@instance)
17
+ @facet.populate_instances
18
+ end
19
+ @instance
20
+ end
21
+
22
+ def inspect
23
+ "<Sunspot::Search::FacetRow:#{value.inspect} (#{count})>"
24
+ end
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,88 @@
1
+ module Sunspot
2
+ module Search
3
+ #
4
+ # A FieldFacet is a facet whose rows are all values for a certain field, in
5
+ # contrast to a QueryFacet, whose rows represent arbitrary queries.
6
+ #
7
+ class FieldFacet < QueryFacet
8
+ def initialize(field, search, options) #:nodoc:
9
+ super((options[:name] || field.name).to_sym, search, options)
10
+ @field = field
11
+ end
12
+
13
+ def field_name
14
+ @field.name
15
+ end
16
+
17
+ #
18
+ # Get the rows returned for this facet.
19
+ #
20
+ # ==== Options (options)
21
+ #
22
+ # :verify::
23
+ # Only return rows for which the referenced object exists in the data
24
+ # store. This option is ignored unless the field associated with this
25
+ # facet is configured with a :references argument.
26
+ #
27
+ # ==== Returns
28
+ #
29
+ # Array:: Array of FacetRow objects
30
+ #
31
+ def rows(options = {})
32
+ if options[:verify]
33
+ verified_rows
34
+ else
35
+ @rows ||=
36
+ begin
37
+ rows = super
38
+ has_query_facets = !rows.empty?
39
+ if @search.facet_response['facet_fields']
40
+ if data = @search.facet_response['facet_fields'][key]
41
+ data.each_slice(2) do |value, count|
42
+ row = FacetRow.new(@field.cast(value), count, self)
43
+ rows << row
44
+ end
45
+ end
46
+ end
47
+ sort_rows!(rows) if has_query_facets
48
+ rows
49
+ end
50
+ end
51
+ end
52
+
53
+ #
54
+ # If this facet references a model class, populate the rows with instances
55
+ # of the model class by loading them out of the appropriate adapter.
56
+ #
57
+ def populate_instances #:nodoc:
58
+ if reference = @field.reference
59
+ values_hash = rows.inject({}) do |hash, row|
60
+ hash[row.value] = row
61
+ hash
62
+ end
63
+ instances = Adapters::DataAccessor.create(Sunspot::Util.full_const_get(reference)).load_all(
64
+ values_hash.keys
65
+ )
66
+ instances.each do |instance|
67
+ values_hash[Adapters::InstanceAdapter.adapt(instance).id].instance = instance
68
+ end
69
+ true
70
+ end
71
+ end
72
+
73
+ private
74
+
75
+ def verified_rows
76
+ if @field.reference
77
+ @verified_rows ||= rows.select { |row| row.instance }
78
+ else
79
+ rows
80
+ end
81
+ end
82
+
83
+ def key
84
+ @key ||= (@options[:name] || @field.indexed_name).to_s
85
+ end
86
+ end
87
+ end
88
+ end
@@ -0,0 +1,32 @@
1
+ module Sunspot
2
+ module Search
3
+ class FieldGroup
4
+ def initialize(field, search, options) #:nodoc:
5
+ @field, @search, @options = field, search, options
6
+ end
7
+
8
+ def groups
9
+ @groups ||=
10
+ begin
11
+ if solr_response
12
+ solr_response['groups'].map do |group|
13
+ Group.new(group['groupValue'], group['doclist'], @search)
14
+ end
15
+ end
16
+ end
17
+ end
18
+
19
+ def matches
20
+ if solr_response
21
+ solr_response['matches'].to_i
22
+ end
23
+ end
24
+
25
+ private
26
+
27
+ def solr_response
28
+ @search.group_response[@field.indexed_name.to_s]
29
+ end
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,50 @@
1
+ require 'sunspot/search/hit_enumerable'
2
+
3
+ module Sunspot
4
+ module Search
5
+ class Group
6
+ attr_reader :value
7
+
8
+ include HitEnumerable
9
+
10
+ def initialize(value, doclist, search)
11
+ @value, @doclist, @search = value, doclist, search
12
+ end
13
+
14
+ def hits(options = {})
15
+ if options[:verify]
16
+ super
17
+ else
18
+ @hits ||= super
19
+ end
20
+ end
21
+
22
+ def verified_hits
23
+ @verified_hits ||= super
24
+ end
25
+
26
+ def results
27
+ @results ||= verified_hits.map { |hit| hit.instance }
28
+ end
29
+
30
+ def highlights_for(doc)
31
+ @search.highlights_for(doc)
32
+ end
33
+
34
+ def solr_docs
35
+ @doclist['docs']
36
+ end
37
+
38
+ #
39
+ # The total number of documents matching the query for this group
40
+ #
41
+ # ==== Returns
42
+ #
43
+ # Integer:: Total matching documents
44
+ #
45
+ def total
46
+ @doclist['numFound']
47
+ end
48
+ end
49
+ end
50
+ end
@@ -0,0 +1,38 @@
1
+ module Sunspot
2
+ module Search
3
+ #
4
+ # A Highlight represents a single highlighted fragment of text from a
5
+ # document. Depending on the highlighting parameters used for search, there
6
+ # may be more than one Highlight object for a given field in a given result.
7
+ #
8
+ class Highlight
9
+ HIGHLIGHT_MATCHER = /@@@hl@@@(.*?)@@@endhl@@@/ #:nodoc:
10
+
11
+ #
12
+ # The name of the field in which the highlight appeared.
13
+ #
14
+ attr_reader :field_name
15
+
16
+ def initialize(field_name, highlight) #:nodoc:
17
+ @field_name = field_name.to_sym
18
+ @highlight = highlight.to_s.strip
19
+ end
20
+
21
+ #
22
+ # Returns the highlighted text with formatting according to the template given in &block.
23
+ # When no block is given, &lt;em&gt; and &lt;/em&gt; are used to surround the highlight.
24
+ #
25
+ # ==== Example
26
+ #
27
+ # search.highlights(:body).first.format { |word| "<strong>#{word}</strong>" }
28
+ #
29
+ def format(&block)
30
+ block ||= proc { |word| "<em>#{word}</em>" }
31
+ @highlight.gsub(HIGHLIGHT_MATCHER) do
32
+ block.call(Regexp.last_match[1])
33
+ end
34
+ end
35
+ alias_method :formatted, :format
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,150 @@
1
+ module Sunspot
2
+ module Search
3
+ #
4
+ # Hit objects represent the raw information returned by Solr for a single
5
+ # document. As well as the primary key and class name, hit objects give
6
+ # access to stored field values, keyword relevance score, and keyword
7
+ # highlighting.
8
+ #
9
+ class Hit
10
+ SPECIAL_KEYS = Set.new(%w(id type score)) #:nodoc:
11
+
12
+ #
13
+ # Primary key of object associated with this hit, as string.
14
+ #
15
+ attr_reader :primary_key
16
+ #
17
+ # Class name of object associated with this hit, as string.
18
+ #
19
+ attr_reader :class_name
20
+ #
21
+ # Keyword relevance score associated with this result. Nil if this hit
22
+ # is not from a keyword search.
23
+ #
24
+ attr_reader :score
25
+ #
26
+
27
+ attr_writer :result #:nodoc:
28
+
29
+ def initialize(raw_hit, highlights, search) #:nodoc:
30
+ @class_name, @primary_key = *raw_hit['id'].match(/([^ ]+) (.+)/)[1..2]
31
+ @score = raw_hit['score']
32
+ @search = search
33
+ @stored_values = raw_hit
34
+ @stored_cache = {}
35
+ @highlights = highlights
36
+ end
37
+
38
+ #
39
+ # Returns all highlights for this hit when called without parameters.
40
+ # When a field_name is provided, returns only the highlight for this field.
41
+ #
42
+ def highlights(field_name = nil)
43
+ if field_name.nil?
44
+ highlights_cache.values.flatten
45
+ else
46
+ highlights_cache[field_name.to_sym]
47
+ end || []
48
+ end
49
+
50
+ #
51
+ # Return the first highlight found for a given field, or nil if there is
52
+ # none.
53
+ #
54
+ def highlight(field_name)
55
+ highlights(field_name).first
56
+ end
57
+
58
+ #
59
+ # Retrieve stored field value. For any attribute field configured with
60
+ # :stored => true, the Hit object will contain the stored value for
61
+ # that field. The value of this field will be typecast according to the
62
+ # type of the field.
63
+ #
64
+ # ==== Parameters
65
+ #
66
+ # field_name<Symbol>::
67
+ # The name of the field for which to retrieve the stored value.
68
+ # dynamic_field_name<Symbol>::
69
+ # If you want to access a stored dynamic field, this should be the
70
+ # dynamic component of the field name.
71
+ #
72
+ def stored(field_name, dynamic_field_name = nil)
73
+ field_key =
74
+ if dynamic_field_name
75
+ [field_name.to_sym, dynamic_field_name.to_sym]
76
+ else
77
+ field_name.to_sym
78
+ end
79
+ return @stored_cache[field_key] if @stored_cache.has_key?(field_key)
80
+ @stored_cache[field_key] = stored_value(field_name, dynamic_field_name)
81
+ end
82
+
83
+ #
84
+ # Retrieve the instance associated with this hit. This is lazy-loaded, but
85
+ # the first time it is called on any hit, all the hits for the search will
86
+ # load their instances using the adapter's #load_all method.
87
+ #
88
+ def result
89
+ return @result if defined?(@result)
90
+ @search.populate_hits
91
+ @result
92
+ end
93
+ alias_method :instance, :result
94
+
95
+ def inspect #:nodoc:
96
+ "#<Sunspot::Search::Hit:#{@class_name} #{@primary_key}>"
97
+ end
98
+
99
+ #
100
+ # Returns the instance primary key when the Hit is used to generate urls
101
+ # For example, using a search that stores the :name attribute:
102
+ #
103
+ # hits = Sunspot.search(Object) do ...
104
+ #
105
+ # hits.each do |hit|
106
+ # link_to hit.stored(:name), edit_object_path(hit)
107
+ # end
108
+ #
109
+ def to_param
110
+ self.primary_key
111
+ end
112
+
113
+ private
114
+
115
+ def setup
116
+ @setup ||= Sunspot::Setup.for(Util.full_const_get(@class_name))
117
+ end
118
+
119
+ def highlights_cache
120
+ @highlights_cache ||=
121
+ begin
122
+ cache = {}
123
+ if @highlights
124
+ @highlights.each_pair do |indexed_field_name, highlight_strings|
125
+ field_name = indexed_field_name.sub(/_[a-z]+$/, '').to_sym
126
+ cache[field_name] = highlight_strings.map do |highlight_string|
127
+ Highlight.new(field_name, highlight_string)
128
+ end
129
+ end
130
+ end
131
+ cache
132
+ end
133
+ end
134
+
135
+ def stored_value(field_name, dynamic_field_name)
136
+ setup.stored_fields(field_name, dynamic_field_name).each do |field|
137
+ value = @stored_values[field.indexed_name]
138
+
139
+ if Array === value
140
+ return value.map { |item| field.cast(item) }
141
+ elsif !value.nil?
142
+ return field.cast(value)
143
+ end
144
+ end
145
+
146
+ nil
147
+ end
148
+ end
149
+ end
150
+ end
@@ -0,0 +1,72 @@
1
+ module Sunspot
2
+ module Search
3
+ module HitEnumerable #:nodoc:
4
+ def hits(options = {})
5
+ if options[:verify]
6
+ verified_hits
7
+ elsif solr_docs
8
+ solr_docs.map { |d| Hit.new(d, highlights_for(d), self) }
9
+ else
10
+ []
11
+ end
12
+ end
13
+
14
+ def verified_hits
15
+ hits.select { |h| h.result }
16
+ end
17
+
18
+ #
19
+ # Populate the Hit objects with their instances. This is invoked the first
20
+ # time any hit has its instance requested, and all hits are loaded as a
21
+ # batch.
22
+ #
23
+ def populate_hits #:nodoc:
24
+ id_hit_hash = Hash.new { |h, k| h[k] = {} }
25
+ hits.each do |hit|
26
+ id_hit_hash[hit.class_name][hit.primary_key] = hit
27
+ end
28
+ id_hit_hash.each_pair do |class_name, hits|
29
+ ids = hits.map { |id, hit| hit.primary_key }
30
+ data_accessor = data_accessor_for(Util.full_const_get(class_name))
31
+ hits_for_class = id_hit_hash[class_name]
32
+ data_accessor.load_all(ids).each do |result|
33
+ hit = hits_for_class.delete(Adapters::InstanceAdapter.adapt(result).id.to_s)
34
+ hit.result = result
35
+ end
36
+ hits_for_class.values.each { |hit| hit.result = nil }
37
+ end
38
+ end
39
+
40
+ #
41
+ # Convenience method to iterate over hit and result objects. Block is
42
+ # yielded a Sunspot::Server::Hit instance and a Sunspot::Server::Result
43
+ # instance.
44
+ #
45
+ # Note that this method iterates over verified hits (see #hits method
46
+ # for more information).
47
+ #
48
+ def each_hit_with_result
49
+ verified_hits.each do |hit|
50
+ yield(hit, hit.result)
51
+ end
52
+ end
53
+
54
+ #
55
+ # Get the data accessor that will be used to load a particular class out of
56
+ # persistent storage. Data accessors can implement any methods that may be
57
+ # useful for refining how data is loaded out of storage. When building a
58
+ # search manually (e.g., using the Sunspot#new_search method), this should
59
+ # be used before calling #execute(). Use the
60
+ # Sunspot::DSL::Search#data_accessor_for method when building searches using
61
+ # the block DSL.
62
+ #
63
+ def data_accessor_for(clazz) #:nodoc:
64
+ # FIXME: This method does not belong here, but I was getting bogged
65
+ # down trying to figure out where it should go. Punted for now.
66
+ # - AL 27 Nov 2011
67
+ (@data_accessors ||= {})[clazz.name.to_sym] ||=
68
+ Adapters::DataAccessor.create(clazz)
69
+ end
70
+ end
71
+ end
72
+ end