gojee-sunspot 2.0.2

Sign up to get free protection for your applications and to get access to all the features.
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,14 @@
1
+ module Sunspot
2
+ module Query
3
+ class DateFieldFacet < AbstractFieldFacet
4
+ def to_params
5
+ params = super
6
+ params[:"facet.date"] = [@field.indexed_name]
7
+ params[qualified_param('date.start')] = @field.to_indexed(@options[:time_range].first)
8
+ params[qualified_param('date.end')] = @field.to_indexed(@options[:time_range].last)
9
+ params[qualified_param('date.gap')] = "+#{@options[:time_interval] || 86400}SECONDS"
10
+ params
11
+ end
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,132 @@
1
+ module Sunspot
2
+ module Query
3
+
4
+ #
5
+ # Solr full-text queries use Solr's DisMaxRequestHandler, a search handler
6
+ # designed to process user-entered phrases, and search for individual
7
+ # words across a union of several fields.
8
+ #
9
+ class Dismax
10
+ attr_writer :minimum_match, :phrase_slop, :query_phrase_slop, :tie
11
+
12
+ def initialize(keywords)
13
+ @keywords = keywords
14
+ @fulltext_fields = {}
15
+ @boost_queries = []
16
+ @boost_functions = []
17
+ @highlights = []
18
+ end
19
+
20
+ #
21
+ # The query as Solr parameters
22
+ #
23
+ def to_params
24
+ params = { :q => @keywords }
25
+ params[:fl] = '* score'
26
+ params[:qf] = @fulltext_fields.values.map { |field| field.to_boosted_field }.join(' ')
27
+ params[:defType] = 'dismax'
28
+ if @phrase_fields
29
+ params[:pf] = @phrase_fields.map { |field| field.to_boosted_field }.join(' ')
30
+ end
31
+ unless @boost_queries.empty?
32
+ params[:bq] = @boost_queries.map do |boost_query|
33
+ boost_query.to_boolean_phrase
34
+ end
35
+ end
36
+ unless @boost_functions.empty?
37
+ params[:bf] = @boost_functions.map do |boost_function|
38
+ boost_function.to_s
39
+ end
40
+ end
41
+ if @minimum_match
42
+ params[:mm] = @minimum_match
43
+ end
44
+ if @phrase_slop
45
+ params[:ps] = @phrase_slop
46
+ end
47
+ if @query_phrase_slop
48
+ params[:qs] = @query_phrase_slop
49
+ end
50
+ if @tie
51
+ params[:tie] = @tie
52
+ end
53
+ @highlights.each do |highlight|
54
+ Sunspot::Util.deep_merge!(params, highlight.to_params)
55
+ end
56
+ params
57
+ end
58
+
59
+ #
60
+ # Serialize the query as a Solr nested subquery.
61
+ #
62
+ def to_subquery
63
+ params = self.to_params
64
+ params.delete :defType
65
+ params.delete :fl
66
+ keywords = params.delete(:q)
67
+ options = params.map { |key, value| escape_param(key, value) }.join(' ')
68
+ "_query_:\"{!dismax #{options}}#{escape_quotes(keywords)}\""
69
+ end
70
+
71
+ #
72
+ # Assign a new boost query and return it.
73
+ #
74
+ def create_boost_query(factor)
75
+ @boost_queries << boost_query = BoostQuery.new(factor)
76
+ boost_query
77
+ end
78
+
79
+ #
80
+ # Add a boost function
81
+ #
82
+ def add_boost_function(function_query)
83
+ @boost_functions << function_query
84
+ end
85
+
86
+ #
87
+ # Add a fulltext field to be searched, with optional boost.
88
+ #
89
+ def add_fulltext_field(field, boost = nil)
90
+ @fulltext_fields[field.indexed_name] = TextFieldBoost.new(field, boost)
91
+ end
92
+
93
+ #
94
+ # Add a phrase field for extra boost.
95
+ #
96
+ def add_phrase_field(field, boost = nil)
97
+ @phrase_fields ||= []
98
+ @phrase_fields << TextFieldBoost.new(field, boost)
99
+ end
100
+
101
+ #
102
+ # Set highlighting options for the query. If fields is empty, the
103
+ # Highlighting object won't pass field names at all, which means
104
+ # the dismax's :qf parameter will be used by Solr.
105
+ #
106
+ def add_highlight(fields=[], options={})
107
+ @highlights << Highlighting.new(fields, options)
108
+ end
109
+
110
+ #
111
+ # Determine if a given field is being searched. Used by DSL to avoid
112
+ # overwriting boost parameters when injecting defaults.
113
+ #
114
+ def has_fulltext_field?(field)
115
+ @fulltext_fields.has_key?(field.indexed_name)
116
+ end
117
+
118
+
119
+ private
120
+
121
+ def escape_param(key, value)
122
+ "#{key}='#{escape_quotes(Array(value).join(" "))}'"
123
+ end
124
+
125
+ def escape_quotes(value)
126
+ return value unless value.is_a? String
127
+ value.gsub(/(['"])/, '\\\\\1')
128
+ end
129
+
130
+ end
131
+ end
132
+ end
@@ -0,0 +1,41 @@
1
+ module Sunspot
2
+ module Query
3
+ class FieldFacet < AbstractFieldFacet
4
+ def initialize(field, options)
5
+ if exclude_filters = options[:exclude]
6
+ @exclude_tag = Util.Array(exclude_filters).map do |filter|
7
+ filter.tag
8
+ end.join(',')
9
+ end
10
+ super
11
+ end
12
+
13
+ def to_params
14
+ super.merge(:"facet.field" => [field_name_with_local_params])
15
+ end
16
+
17
+ private
18
+
19
+ def local_params
20
+ @local_params ||=
21
+ begin
22
+ local_params = {}
23
+ local_params[:ex] = @exclude_tag if @exclude_tag
24
+ local_params[:key] = @options[:name] if @options[:name]
25
+ local_params
26
+ end
27
+ end
28
+
29
+ def field_name_with_local_params
30
+ if local_params.empty?
31
+ @field.indexed_name
32
+ else
33
+ pairs = local_params.map do |key, value|
34
+ "#{key}=#{value}"
35
+ end
36
+ "{!#{pairs.join(' ')}}#{@field.indexed_name}"
37
+ end
38
+ end
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,36 @@
1
+ module Sunspot
2
+ module Query
3
+ #
4
+ # A FieldGroup groups by the unique values of a given field.
5
+ #
6
+ class FieldGroup
7
+ attr_accessor :limit, :truncate
8
+
9
+ def initialize(field)
10
+ if field.multiple?
11
+ raise(ArgumentError, "#{field.name} cannot be used for grouping because it is a multiple-value field")
12
+ end
13
+ @field = field
14
+
15
+ @sort = SortComposite.new
16
+ end
17
+
18
+ def add_sort(sort)
19
+ @sort << sort
20
+ end
21
+
22
+ def to_params
23
+ params = {
24
+ :group => "true",
25
+ :"group.field" => @field.indexed_name,
26
+ }
27
+
28
+ params.merge!(@sort.to_params("group."))
29
+ params[:"group.limit"] = @limit if @limit
30
+ params[:"group.truncate"] = @truncate if @truncate
31
+
32
+ params
33
+ end
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,38 @@
1
+ module Sunspot
2
+ module Query
3
+ module Filter
4
+
5
+ #
6
+ # Express this filter as an :fq parameter; i.e., the boolean phrase,
7
+ # maybe prefixed by local params.
8
+ #
9
+ def to_filter_query
10
+ if tagged? then "{!tag=#{tag}}#{to_boolean_phrase}"
11
+ else to_boolean_phrase
12
+ end
13
+ end
14
+
15
+ #
16
+ # Generate and return a tag that can be attached to this restriction,
17
+ # for use with multiselect faceting. This needs to be unique, but doesn't
18
+ # really need to be human-readable, so just generate a string based on the
19
+ # hash of the boolean phrase.
20
+ #
21
+ def tag
22
+ @tag ||= to_boolean_phrase.hash.abs.to_s(36)
23
+ end
24
+
25
+ private
26
+
27
+ #
28
+ # True if a tag has been generated for this filter (e.g., if it's been
29
+ # excluded from a given facet). If a tag has not been generated at the
30
+ # time that the filter query param is requested, then it is not necessary
31
+ # to include a tag in the local params.
32
+ #
33
+ def tagged?
34
+ !!@tag
35
+ end
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,52 @@
1
+ module Sunspot
2
+ module Query
3
+ #
4
+ # Abstract class for function queries.
5
+ #
6
+ class FunctionQuery
7
+ include RSolr::Char
8
+ end
9
+
10
+ #
11
+ # Function query which represents a constant.
12
+ #
13
+ class ConstantFunctionQuery < FunctionQuery
14
+ def initialize(constant)
15
+ @constant = constant
16
+ end
17
+
18
+ def to_s
19
+ Type.to_literal(@constant)
20
+ end
21
+ end
22
+
23
+ #
24
+ # Function query which represents a field.
25
+ #
26
+ class FieldFunctionQuery < FunctionQuery
27
+ def initialize(field)
28
+ @field = field
29
+ end
30
+
31
+ def to_s
32
+ "#{escape(@field.indexed_name)}"
33
+ end
34
+ end
35
+
36
+ #
37
+ # Function query which represents an actual function invocation.
38
+ # Takes a function name and arguments as parameters.
39
+ # Arguments are in turn FunctionQuery objects.
40
+ #
41
+ class FunctionalFunctionQuery < FunctionQuery
42
+ def initialize(function_name, function_args)
43
+ @function_name, @function_args = function_name, function_args
44
+ end
45
+
46
+ def to_s
47
+ params = @function_args.map { |arg| arg.to_s }.join(",")
48
+ "#{@function_name}(#{params})"
49
+ end
50
+ end
51
+ end
52
+ end
@@ -0,0 +1,53 @@
1
+ begin
2
+ require 'geohash'
3
+ rescue LoadError => e
4
+ require 'pr_geohash'
5
+ end
6
+
7
+ module Sunspot
8
+ module Query
9
+ class Geo
10
+ MAX_PRECISION = 12
11
+ DEFAULT_PRECISION = 7
12
+ DEFAULT_PRECISION_FACTOR = 16.0
13
+
14
+ def initialize(field, lat, lng, options)
15
+ @field, @options = field, options
16
+ @geohash = GeoHash.encode(lat.to_f, lng.to_f, MAX_PRECISION)
17
+ end
18
+
19
+ def to_params
20
+ { :q => to_boolean_query }
21
+ end
22
+
23
+ def to_subquery
24
+ "(#{to_boolean_query})"
25
+ end
26
+
27
+ private
28
+
29
+ def to_boolean_query
30
+ queries = []
31
+ MAX_PRECISION.downto(precision) do |i|
32
+ star = i == MAX_PRECISION ? '' : '*'
33
+ precision_boost = Util.format_float(
34
+ boost * precision_factor ** (i-MAX_PRECISION).to_f, 3)
35
+ queries << "#{@field.indexed_name}:#{@geohash[0, i]}#{star}^#{precision_boost}"
36
+ end
37
+ queries.join(' OR ')
38
+ end
39
+
40
+ def precision
41
+ @options[:precision] || DEFAULT_PRECISION
42
+ end
43
+
44
+ def precision_factor
45
+ @options[:precision_factor] || DEFAULT_PRECISION_FACTOR
46
+ end
47
+
48
+ def boost
49
+ @options[:boost] || 1.0
50
+ end
51
+ end
52
+ end
53
+ end
@@ -0,0 +1,16 @@
1
+ module Sunspot
2
+ module Query
3
+ class Geofilt
4
+ def initialize(field, lat, lon, radius, options = {})
5
+ @field, @lat, @lon, @radius, @options = field, lat, lon, radius, options
6
+ end
7
+
8
+ def to_params
9
+ func = @options[:bbox] ? "bbox" : "geofilt"
10
+
11
+ filter = "{!#{func} sfield=#{@field.indexed_name} pt=#{@lat},#{@lon} d=#{@radius}}"
12
+ {:fq => filter}
13
+ end
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,62 @@
1
+ module Sunspot
2
+ module Query
3
+ #
4
+ # A query component that builds parameters for requesting highlights
5
+ #
6
+ class Highlighting #:nodoc:
7
+ def initialize(fields=[], options={})
8
+ @fields = fields
9
+ @options = options
10
+ end
11
+
12
+ #
13
+ # Return Solr highlighting params
14
+ #
15
+ def to_params
16
+ params = {
17
+ :hl => 'on',
18
+ :"hl.simple.pre" => '@@@hl@@@',
19
+ :"hl.simple.post" => '@@@endhl@@@'
20
+ }
21
+ unless @fields.empty?
22
+ params[:"hl.fl"] = @fields.map { |field| field.indexed_name }
23
+ end
24
+ if max_snippets = @options[:max_snippets]
25
+ params.merge!(make_params('snippets', max_snippets))
26
+ end
27
+ if fragment_size = @options[:fragment_size]
28
+ params.merge!(make_params('fragsize', fragment_size))
29
+ end
30
+ if @options[:merge_contiguous_fragments]
31
+ params.merge!(make_params('mergeContiguous', 'true'))
32
+ end
33
+ if @options[:phrase_highlighter]
34
+ params.merge!(make_params('usePhraseHighlighter', 'true'))
35
+ if @options[:require_field_match]
36
+ params.merge!(make_params('requireFieldMatch', 'true'))
37
+ end
38
+ end
39
+ if formatter = @options[:formatter]
40
+ params.merge!(make_params('formatter', formatter))
41
+ end
42
+ if fragmenter = @options[:fragmenter]
43
+ params.merge!(make_params('fragmenter', fragmenter))
44
+ end
45
+
46
+ params
47
+ end
48
+
49
+ private
50
+
51
+ def make_params(name, value)
52
+ if @fields.empty?
53
+ { :"hl.#{name}" => value }
54
+ else
55
+ @fields.inject({}) do |hash, field|
56
+ hash.merge!(:"f.#{field.indexed_name}.hl.#{name}" => value)
57
+ end
58
+ end
59
+ end
60
+ end
61
+ end
62
+ end