nuatt_sunspot 1.1.0.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (187) hide show
  1. data/History.txt +198 -0
  2. data/LICENSE +18 -0
  3. data/README.rdoc +239 -0
  4. data/Rakefile +11 -0
  5. data/TODO +13 -0
  6. data/VERSION.yml +4 -0
  7. data/bin/sunspot-installer +19 -0
  8. data/bin/sunspot-solr +74 -0
  9. data/installer/config/schema.yml +95 -0
  10. data/lib/light_config.rb +40 -0
  11. data/lib/sunspot.rb +529 -0
  12. data/lib/sunspot/adapters.rb +265 -0
  13. data/lib/sunspot/composite_setup.rb +202 -0
  14. data/lib/sunspot/configuration.rb +46 -0
  15. data/lib/sunspot/data_extractor.rb +50 -0
  16. data/lib/sunspot/dsl.rb +5 -0
  17. data/lib/sunspot/dsl/adjustable.rb +47 -0
  18. data/lib/sunspot/dsl/field_query.rb +266 -0
  19. data/lib/sunspot/dsl/fields.rb +113 -0
  20. data/lib/sunspot/dsl/fulltext.rb +243 -0
  21. data/lib/sunspot/dsl/function.rb +14 -0
  22. data/lib/sunspot/dsl/functional.rb +41 -0
  23. data/lib/sunspot/dsl/more_like_this_query.rb +56 -0
  24. data/lib/sunspot/dsl/paginatable.rb +28 -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/scope.rb +229 -0
  28. data/lib/sunspot/dsl/search.rb +30 -0
  29. data/lib/sunspot/dsl/standard_query.rb +125 -0
  30. data/lib/sunspot/field.rb +192 -0
  31. data/lib/sunspot/field_factory.rb +147 -0
  32. data/lib/sunspot/indexer.rb +131 -0
  33. data/lib/sunspot/installer.rb +31 -0
  34. data/lib/sunspot/installer/library_installer.rb +45 -0
  35. data/lib/sunspot/installer/schema_builder.rb +219 -0
  36. data/lib/sunspot/installer/solrconfig_updater.rb +106 -0
  37. data/lib/sunspot/installer/task_helper.rb +18 -0
  38. data/lib/sunspot/query.rb +10 -0
  39. data/lib/sunspot/query/abstract_field_facet.rb +52 -0
  40. data/lib/sunspot/query/boost_query.rb +24 -0
  41. data/lib/sunspot/query/common_query.rb +85 -0
  42. data/lib/sunspot/query/composite_fulltext.rb +31 -0
  43. data/lib/sunspot/query/connective.rb +191 -0
  44. data/lib/sunspot/query/date_field_facet.rb +14 -0
  45. data/lib/sunspot/query/dismax.rb +127 -0
  46. data/lib/sunspot/query/field_facet.rb +41 -0
  47. data/lib/sunspot/query/filter.rb +38 -0
  48. data/lib/sunspot/query/function_query.rb +52 -0
  49. data/lib/sunspot/query/highlighting.rb +55 -0
  50. data/lib/sunspot/query/local.rb +26 -0
  51. data/lib/sunspot/query/more_like_this.rb +60 -0
  52. data/lib/sunspot/query/more_like_this_query.rb +12 -0
  53. data/lib/sunspot/query/pagination.rb +38 -0
  54. data/lib/sunspot/query/query_facet.rb +16 -0
  55. data/lib/sunspot/query/restriction.rb +262 -0
  56. data/lib/sunspot/query/scope.rb +9 -0
  57. data/lib/sunspot/query/sort.rb +95 -0
  58. data/lib/sunspot/query/sort_composite.rb +33 -0
  59. data/lib/sunspot/query/standard_query.rb +20 -0
  60. data/lib/sunspot/query/text_field_boost.rb +17 -0
  61. data/lib/sunspot/schema.rb +151 -0
  62. data/lib/sunspot/search.rb +9 -0
  63. data/lib/sunspot/search/abstract_search.rb +302 -0
  64. data/lib/sunspot/search/date_facet.rb +35 -0
  65. data/lib/sunspot/search/facet_row.rb +27 -0
  66. data/lib/sunspot/search/field_facet.rb +88 -0
  67. data/lib/sunspot/search/highlight.rb +38 -0
  68. data/lib/sunspot/search/hit.rb +136 -0
  69. data/lib/sunspot/search/more_like_this_search.rb +31 -0
  70. data/lib/sunspot/search/query_facet.rb +62 -0
  71. data/lib/sunspot/search/standard_search.rb +21 -0
  72. data/lib/sunspot/server.rb +152 -0
  73. data/lib/sunspot/session.rb +252 -0
  74. data/lib/sunspot/session_proxy.rb +71 -0
  75. data/lib/sunspot/session_proxy/abstract_session_proxy.rb +29 -0
  76. data/lib/sunspot/session_proxy/class_sharding_session_proxy.rb +66 -0
  77. data/lib/sunspot/session_proxy/id_sharding_session_proxy.rb +89 -0
  78. data/lib/sunspot/session_proxy/master_slave_session_proxy.rb +43 -0
  79. data/lib/sunspot/session_proxy/sharding_session_proxy.rb +215 -0
  80. data/lib/sunspot/session_proxy/thread_local_session_proxy.rb +37 -0
  81. data/lib/sunspot/setup.rb +366 -0
  82. data/lib/sunspot/text_field_setup.rb +29 -0
  83. data/lib/sunspot/type.rb +340 -0
  84. data/lib/sunspot/util.rb +253 -0
  85. data/lib/sunspot/version.rb +3 -0
  86. data/solr/etc/jetty.xml +214 -0
  87. data/solr/etc/webdefault.xml +379 -0
  88. data/solr/lib/jetty-6.1.3.jar +0 -0
  89. data/solr/lib/jetty-util-6.1.3.jar +0 -0
  90. data/solr/lib/jsp-2.1/ant-1.6.5.jar +0 -0
  91. data/solr/lib/jsp-2.1/core-3.1.1.jar +0 -0
  92. data/solr/lib/jsp-2.1/jsp-2.1.jar +0 -0
  93. data/solr/lib/jsp-2.1/jsp-api-2.1.jar +0 -0
  94. data/solr/lib/servlet-api-2.5-6.1.3.jar +0 -0
  95. data/solr/solr/conf/admin-extra.html +31 -0
  96. data/solr/solr/conf/elevate.xml +36 -0
  97. data/solr/solr/conf/mapping-ISOLatin1Accent.txt +246 -0
  98. data/solr/solr/conf/protwords.txt +21 -0
  99. data/solr/solr/conf/schema.xml +240 -0
  100. data/solr/solr/conf/scripts.conf +24 -0
  101. data/solr/solr/conf/solrconfig.xml +938 -0
  102. data/solr/solr/conf/spellings.txt +2 -0
  103. data/solr/solr/conf/stopwords.es.txt +345 -0
  104. data/solr/solr/conf/stopwords.txt +58 -0
  105. data/solr/solr/conf/synonyms.txt +31 -0
  106. data/solr/solr/lib/lucene-spatial-2.9.1.jar +0 -0
  107. data/solr/solr/lib/solr-spatial-light-0.0.6.jar +0 -0
  108. data/solr/start.jar +0 -0
  109. data/solr/webapps/solr.war +0 -0
  110. data/spec/api/adapters_spec.rb +33 -0
  111. data/spec/api/binding_spec.rb +38 -0
  112. data/spec/api/indexer/attributes_spec.rb +149 -0
  113. data/spec/api/indexer/batch_spec.rb +46 -0
  114. data/spec/api/indexer/dynamic_fields_spec.rb +42 -0
  115. data/spec/api/indexer/fixed_fields_spec.rb +57 -0
  116. data/spec/api/indexer/fulltext_spec.rb +43 -0
  117. data/spec/api/indexer/removal_spec.rb +53 -0
  118. data/spec/api/indexer/spec_helper.rb +1 -0
  119. data/spec/api/indexer_spec.rb +14 -0
  120. data/spec/api/query/advanced_manipulation_examples.rb +35 -0
  121. data/spec/api/query/connectives_examples.rb +176 -0
  122. data/spec/api/query/dsl_spec.rb +18 -0
  123. data/spec/api/query/dynamic_fields_examples.rb +165 -0
  124. data/spec/api/query/faceting_examples.rb +399 -0
  125. data/spec/api/query/fulltext_examples.rb +315 -0
  126. data/spec/api/query/function_spec.rb +70 -0
  127. data/spec/api/query/highlighting_examples.rb +225 -0
  128. data/spec/api/query/local_examples.rb +38 -0
  129. data/spec/api/query/more_like_this_spec.rb +140 -0
  130. data/spec/api/query/ordering_pagination_examples.rb +97 -0
  131. data/spec/api/query/scope_examples.rb +256 -0
  132. data/spec/api/query/spec_helper.rb +1 -0
  133. data/spec/api/query/standard_spec.rb +28 -0
  134. data/spec/api/query/text_field_scoping_examples.rb +30 -0
  135. data/spec/api/query/types_spec.rb +20 -0
  136. data/spec/api/search/dynamic_fields_spec.rb +33 -0
  137. data/spec/api/search/faceting_spec.rb +356 -0
  138. data/spec/api/search/highlighting_spec.rb +69 -0
  139. data/spec/api/search/hits_spec.rb +138 -0
  140. data/spec/api/search/results_spec.rb +79 -0
  141. data/spec/api/search/search_spec.rb +23 -0
  142. data/spec/api/search/spec_helper.rb +1 -0
  143. data/spec/api/server_spec.rb +91 -0
  144. data/spec/api/session_proxy/class_sharding_session_proxy_spec.rb +85 -0
  145. data/spec/api/session_proxy/id_sharding_session_proxy_spec.rb +30 -0
  146. data/spec/api/session_proxy/master_slave_session_proxy_spec.rb +41 -0
  147. data/spec/api/session_proxy/sharding_session_proxy_spec.rb +77 -0
  148. data/spec/api/session_proxy/spec_helper.rb +9 -0
  149. data/spec/api/session_proxy/thread_local_session_proxy_spec.rb +50 -0
  150. data/spec/api/session_spec.rb +198 -0
  151. data/spec/api/spec_helper.rb +3 -0
  152. data/spec/api/sunspot_spec.rb +18 -0
  153. data/spec/ext.rb +11 -0
  154. data/spec/helpers/indexer_helper.rb +29 -0
  155. data/spec/helpers/query_helper.rb +38 -0
  156. data/spec/helpers/search_helper.rb +80 -0
  157. data/spec/integration/dynamic_fields_spec.rb +55 -0
  158. data/spec/integration/faceting_spec.rb +238 -0
  159. data/spec/integration/highlighting_spec.rb +22 -0
  160. data/spec/integration/indexing_spec.rb +33 -0
  161. data/spec/integration/keyword_search_spec.rb +317 -0
  162. data/spec/integration/local_search_spec.rb +91 -0
  163. data/spec/integration/more_like_this_spec.rb +43 -0
  164. data/spec/integration/scoped_search_spec.rb +324 -0
  165. data/spec/integration/spec_helper.rb +7 -0
  166. data/spec/integration/stored_fields_spec.rb +10 -0
  167. data/spec/integration/test_pagination.rb +32 -0
  168. data/spec/mocks/adapters.rb +32 -0
  169. data/spec/mocks/blog.rb +3 -0
  170. data/spec/mocks/comment.rb +21 -0
  171. data/spec/mocks/connection.rb +122 -0
  172. data/spec/mocks/mock_adapter.rb +30 -0
  173. data/spec/mocks/mock_class_sharding_session_proxy.rb +24 -0
  174. data/spec/mocks/mock_record.rb +52 -0
  175. data/spec/mocks/mock_sharding_session_proxy.rb +15 -0
  176. data/spec/mocks/photo.rb +12 -0
  177. data/spec/mocks/post.rb +76 -0
  178. data/spec/mocks/super_class.rb +2 -0
  179. data/spec/mocks/user.rb +8 -0
  180. data/spec/spec_helper.rb +52 -0
  181. data/tasks/gemspec.rake +33 -0
  182. data/tasks/rcov.rake +28 -0
  183. data/tasks/rdoc.rake +27 -0
  184. data/tasks/schema.rake +19 -0
  185. data/tasks/spec.rake +24 -0
  186. data/tasks/todo.rake +4 -0
  187. metadata +355 -0
@@ -0,0 +1,46 @@
1
+ module Sunspot
2
+ # The Sunspot::Configuration module provides a factory method for Sunspot
3
+ # configuration objects. Available properties are:
4
+ #
5
+ # Sunspot.config.solr.url::
6
+ # The URL at which to connect to Solr
7
+ # (default: 'http://localhost:8983/solr')
8
+ # Sunspot.config.pagination.default_per_page::
9
+ # Solr always paginates its results. This sets Sunspot's default result
10
+ # count per page if it is not explicitly specified in the query.
11
+ #
12
+ module Configuration
13
+ class <<self
14
+ # Factory method to build configuration instances.
15
+ #
16
+ # ==== Returns
17
+ #
18
+ # LightConfig::Configuration:: new configuration instance with defaults
19
+ #
20
+ def build #:nodoc:
21
+ LightConfig.build do
22
+ solr do
23
+ url 'http://127.0.0.1:8983/solr'
24
+ end
25
+ master_solr do
26
+ url nil
27
+ end
28
+ pagination do
29
+ default_per_page 30
30
+ end
31
+ end
32
+ end
33
+
34
+ # Location for the default solr configuration files,
35
+ # required for bootstrapping a new solr installation
36
+ #
37
+ # ==== Returns
38
+ #
39
+ # String:: Directory with default solr config files
40
+ #
41
+ def solr_default_configuration_location
42
+ File.join( File.dirname(__FILE__), '../../solr/solr/conf' )
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,50 @@
1
+ module Sunspot
2
+ #
3
+ # DataExtractors present an internal API for the indexer to use to extract
4
+ # field values from models for indexing. They must implement the #value_for
5
+ # method, which takes an object and returns the value extracted from it.
6
+ #
7
+ module DataExtractor #:nodoc: all
8
+ #
9
+ # AttributeExtractors extract data by simply calling a method on the block.
10
+ #
11
+ class AttributeExtractor
12
+ def initialize(attribute_name)
13
+ @attribute_name = attribute_name
14
+ end
15
+
16
+ def value_for(object)
17
+ object.send(@attribute_name)
18
+ end
19
+ end
20
+
21
+ #
22
+ # BlockExtractors extract data by evaluating a block in the context of the
23
+ # object instance, or if the block takes an argument, by passing the object
24
+ # as the argument to the block. Either way, the return value of the block is
25
+ # the value returned by the extractor.
26
+ #
27
+ class BlockExtractor
28
+ def initialize(&block)
29
+ @block = block
30
+ end
31
+
32
+ def value_for(object)
33
+ Util.instance_eval_or_call(object, &@block)
34
+ end
35
+ end
36
+
37
+ #
38
+ # Constant data extractors simply return the same value for every object.
39
+ #
40
+ class Constant
41
+ def initialize(value)
42
+ @value = value
43
+ end
44
+
45
+ def value_for(object)
46
+ @value
47
+ end
48
+ end
49
+ end
50
+ end
@@ -0,0 +1,5 @@
1
+ %w(fields scope paginatable adjustable field_query standard_query query_facet
2
+ functional fulltext restriction search more_like_this_query
3
+ function).each do |file|
4
+ require File.join(File.dirname(__FILE__), 'dsl', file)
5
+ end
@@ -0,0 +1,47 @@
1
+ module Sunspot
2
+ module DSL #:nodoc:
3
+ module Adjustable #:nodoc
4
+ # <strong>Expert:</strong> Adjust or reset the parameters passed to Solr.
5
+ # The adjustment will take place just before sending the params to solr,
6
+ # after Sunspot builds the Solr params based on the methods called in the
7
+ # DSL.
8
+ #
9
+ # Under normal circumstances, using this method should not be necessary;
10
+ # if you find that it is, please consider submitting a feature request.
11
+ # Using this method requires knowledge of Sunspot's internal Solr schema
12
+ # and Solr query representations, which are not part of Sunspot's public
13
+ # API; they could change at any time. <strong>This method is unsupported
14
+ # and your mileage may vary.</strong>
15
+ #
16
+ # ==== Examples
17
+ #
18
+ # Sunspot.search(Post) do
19
+ # adjust_solr_params do |params|
20
+ # params[:q] += ' AND something_s:more'
21
+ # end
22
+ # end
23
+ #
24
+ # Sunspot.more_like_this(my_post) do
25
+ # adjust_solr_params do |params|
26
+ # params["mlt.match.include"] = true
27
+ # end
28
+ # end
29
+ #
30
+ def adjust_solr_params( &block )
31
+ @query.solr_parameter_adjustment = block
32
+ end
33
+
34
+ #
35
+ # <strong>Expert:</strong> Use a custom request handler for this search.
36
+ # The general use case for this would be a request handler configuration
37
+ # you've defined in solrconfig that has different search components,
38
+ # defaults, etc. Using this to point at an entirely different type of
39
+ # request handler that Sunspot doesn't support probably won't get you very
40
+ # far.
41
+ #
42
+ def request_handler(request_handler)
43
+ @search.request_handler = request_handler
44
+ end
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,266 @@
1
+ module Sunspot
2
+ module DSL
3
+ #
4
+ # Provides an API for areas of the query DSL that operate on specific
5
+ # fields. This functionality is provided by the query DSL and the dynamic
6
+ # query DSL.
7
+ #
8
+ class FieldQuery < Scope
9
+ def initialize(search, query, setup) #:nodoc:
10
+ @search, @query = search, query
11
+ super(query.scope, setup)
12
+ end
13
+
14
+ # Specify the order that results should be returned in. This method can
15
+ # be called multiple times; precedence will be in the order given.
16
+ #
17
+ # ==== Parameters
18
+ #
19
+ # field_name<Symbol>:: the field to use for ordering
20
+ # direction<Symbol>:: :asc or :desc (default :asc)
21
+ #
22
+ def order_by(field_name, direction = nil)
23
+ sort =
24
+ if special = Sunspot::Query::Sort.special(field_name)
25
+ special.new(direction)
26
+ else
27
+ Sunspot::Query::Sort::FieldSort.new(
28
+ @setup.field(field_name), direction
29
+ )
30
+ end
31
+ @query.add_sort(sort)
32
+ end
33
+
34
+ #
35
+ # DEPRECATED Use <code>order_by(:random)</code>
36
+ #
37
+ def order_by_random
38
+ order_by(:random)
39
+ end
40
+
41
+ #
42
+ # Request a facet on the search query. A facet is a feature of Solr that
43
+ # determines the number of documents that match the existing search *and*
44
+ # an additional criterion. This allows you to build powerful drill-down
45
+ # interfaces for search, at each step presenting the searcher with a set
46
+ # of refinements that are known to return results.
47
+ #
48
+ # In Sunspot, each facet returns zero or more rows, each of which
49
+ # represents a particular criterion conjoined with the actual query being
50
+ # performed. For _field_ _facets_, each row represents a particular value
51
+ # for a given field. For _query_ _facets_, each row represents an
52
+ # arbitrary scope; the facet itself is just a means of logically grouping
53
+ # the scopes.
54
+ #
55
+ # === Examples
56
+ #
57
+ # ==== Field Facets
58
+ #
59
+ # A field facet is specified by passing one or more Symbol arguments to
60
+ # this method:
61
+ #
62
+ # Sunspot.search(Post) do
63
+ # with(:blog_id, 1)
64
+ # facet(:category_id)
65
+ # end
66
+ #
67
+ # The facet specified above will have a row for each category_id that is
68
+ # present in a document which also has a blog_id of 1.
69
+ #
70
+ # ==== Multiselect Facets
71
+ #
72
+ # In certain circumstances, it is beneficial to exclude certain query
73
+ # scopes from a facet; the most common example is multi-select faceting,
74
+ # where the user has selected a certain value, but the facet should still
75
+ # show all options that would be available if they had not:
76
+ #
77
+ # Sunspot.search(Post) do
78
+ # with(:blog_id, 1)
79
+ # category_filter = with(:category_id, 2)
80
+ # facet(:category_id, :exclude => category_filter)
81
+ # end
82
+ #
83
+ # Although the results of the above search will be restricted to those
84
+ # with a category_id of 2, the category_id facet will operate as if a
85
+ # category had not been selected, allowing the user to select additional
86
+ # categories (which will presumably be ORed together).
87
+ #
88
+ # <strong>As far as I can tell, Solr only supports multi-select with
89
+ # field facets; if +:exclude+ is passed to a query facet, this method will
90
+ # raise an error. Also, the +:only+ and +:extra+ options use query
91
+ # faceting under the hood, so these can't be used with +:extra+ either.
92
+ # </strong>
93
+ #
94
+ # ==== Query Facets
95
+ #
96
+ # A query facet is a collection of arbitrary scopes, each of which
97
+ # represents a row. This is specified by passing a block into the #facet
98
+ # method; the block then contains one or more +row+ blocks, each of which
99
+ # creates a query facet row. The +row+ blocks follow the usual Sunspot
100
+ # scope DSL.
101
+ #
102
+ # For example, a query facet can be used to facet over a set of ranges:
103
+ #
104
+ # Sunspot.search(Post) do
105
+ # facet(:average_rating) do
106
+ # row(1.0..2.0) do
107
+ # with(:average_rating, 1.0..2.0)
108
+ # end
109
+ # row(2.0..3.0) do
110
+ # with(:average_rating, 2.0..3.0)
111
+ # end
112
+ # row(3.0..4.0) do
113
+ # with(:average_rating, 3.0..4.0)
114
+ # end
115
+ # row(4.0..5.0) do
116
+ # with(:average_rating, 4.0..5.0)
117
+ # end
118
+ # end
119
+ # end
120
+ #
121
+ # Note that the arguments to the +facet+ and +row+ methods simply provide
122
+ # labels for the facet and its rows, so that they can be retrieved and
123
+ # identified from the Search object. They are not passed to Solr and no
124
+ # semantic meaning is attached to them. The label for +facet+ should be
125
+ # a symbol; the label for +row+ can be whatever you'd like.
126
+ #
127
+ # ==== Parameters
128
+ #
129
+ # field_names...<Symbol>:: fields for which to return field facets
130
+ #
131
+ # ==== Options
132
+ #
133
+ # :sort<Symbol>::
134
+ # Either :count (values matching the most terms first) or :index (lexical)
135
+ # :limit<Integer>::
136
+ # The maximum number of facet rows to return
137
+ # :minimum_count<Integer>::
138
+ # The minimum count a facet row must have to be returned
139
+ # :zeros<Boolean>::
140
+ # Return facet rows for which there are no matches (equivalent to
141
+ # :minimum_count => 0). Default is false.
142
+ # :exclude<Object,Array>::
143
+ # Exclude one or more filters when performing the faceting (see
144
+ # Multiselect Faceting above). The object given for this argument should
145
+ # be the return value(s) of a scoping method (+with+, +any_of+,
146
+ # +all_of+, etc.). <strong>Only can be used for field facets that do not
147
+ # use the +:extra+ or +:only+ options.</strong>
148
+ # :name<Symbol>::
149
+ # Give a custom name to a field facet. The main use case for this option
150
+ # is for requesting the same field facet multiple times, using different
151
+ # filter exclusions (see Multiselect Faceting above). If you pass this
152
+ # option, it is also the argument that should be passed to Search#facet
153
+ # when retrieving the facet result.
154
+ # :only<Array>::
155
+ # Only return facet rows for the given values. Useful if you are only
156
+ # interested in faceting on a subset of values for a given field.
157
+ # <strong>Only applies to field facets.</strong>
158
+ # :extra<Symbol,Array>::
159
+ # One or more of :any and :none. :any returns a facet row with a count
160
+ # of all matching documents that have some value for this field. :none
161
+ # returns a facet row with a count of all matching documents that have
162
+ # no value for this field. The facet row(s) corresponding to the extras
163
+ # have a value of the symbol passed. <strong>Only applies to field
164
+ # facets.</strong>
165
+ #
166
+ def facet(*field_names, &block)
167
+ options = Sunspot::Util.extract_options_from(field_names)
168
+
169
+ if block
170
+ if field_names.length != 1
171
+ raise(
172
+ ArgumentError,
173
+ "wrong number of arguments (#{field_names.length} for 1)"
174
+ )
175
+ end
176
+ if options.has_key?(:exclude)
177
+ raise(
178
+ ArgumentError,
179
+ "can't use :exclude with query facets"
180
+ )
181
+ end
182
+ search_facet = @search.add_query_facet(field_names.first, options)
183
+ Sunspot::Util.instance_eval_or_call(
184
+ QueryFacet.new(@query, @setup, search_facet),
185
+ &block
186
+ )
187
+ elsif options[:only]
188
+ if options.has_key?(:exclude)
189
+ raise(
190
+ ArgumentError,
191
+ "can't use :exclude with :only (see documentation)"
192
+ )
193
+ end
194
+ field_names.each do |field_name|
195
+ field = @setup.field(field_name)
196
+ search_facet = @search.add_field_facet(field, options)
197
+ Util.Array(options[:only]).each do |value|
198
+ facet = Sunspot::Query::QueryFacet.new
199
+ facet.add_restriction(field, Sunspot::Query::Restriction::EqualTo, value)
200
+ @query.add_query_facet(facet)
201
+ search_facet.add_row(value, facet.to_boolean_phrase)
202
+ end
203
+ end
204
+ else
205
+ field_names.each do |field_name|
206
+ search_facet = nil
207
+ field = @setup.field(field_name)
208
+ facet =
209
+ if options[:time_range]
210
+ unless field.type.is_a?(Sunspot::Type::TimeType)
211
+ raise(
212
+ ArgumentError,
213
+ ':time_range can only be specified for Date or Time fields'
214
+ )
215
+ end
216
+ search_facet = @search.add_date_facet(field, options)
217
+ Sunspot::Query::DateFieldFacet.new(field, options)
218
+ else
219
+ search_facet = @search.add_field_facet(field, options)
220
+ Sunspot::Query::FieldFacet.new(field, options)
221
+ end
222
+ @query.add_field_facet(facet)
223
+ Util.Array(options[:extra]).each do |extra|
224
+ if options.has_key?(:exclude)
225
+ raise(
226
+ ArgumentError,
227
+ "can't use :exclude with :extra (see documentation)"
228
+ )
229
+ end
230
+ extra_facet = Sunspot::Query::QueryFacet.new
231
+ case extra
232
+ when :any
233
+ extra_facet.add_negated_restriction(
234
+ field,
235
+ Sunspot::Query::Restriction::EqualTo,
236
+ nil
237
+ )
238
+ when :none
239
+ extra_facet.add_restriction(
240
+ field,
241
+ Sunspot::Query::Restriction::EqualTo,
242
+ nil
243
+ )
244
+ else
245
+ raise(
246
+ ArgumentError,
247
+ "Allowed values for :extra are :any and :none"
248
+ )
249
+ end
250
+ search_facet.add_row(extra, extra_facet.to_boolean_phrase)
251
+ @query.add_query_facet(extra_facet)
252
+ end
253
+ end
254
+ end
255
+ end
256
+
257
+ def dynamic(base_name, &block)
258
+ dynamic_field_factory = @setup.dynamic_field_factory(base_name)
259
+ Sunspot::Util.instance_eval_or_call(
260
+ FieldQuery.new(@search, @query, dynamic_field_factory),
261
+ &block
262
+ )
263
+ end
264
+ end
265
+ end
266
+ end
@@ -0,0 +1,113 @@
1
+ module Sunspot
2
+ module DSL #:nodoc:
3
+ # The Fields class provides a DSL for specifying field definitions in the
4
+ # Sunspot.setup block. As well as the #text method, which creates fulltext
5
+ # fields, uses #method_missing to allow definition of typed fields. The
6
+ # available methods are determined by the constants defined in
7
+ # Sunspot::Type - in theory (though this is untested), plugin developers
8
+ # should be able to add support for new types simply by creating new
9
+ # implementations in Sunspot::Type
10
+ #
11
+ class Fields
12
+ def initialize(setup) #:nodoc:
13
+ @setup = setup
14
+ end
15
+
16
+ # Add a text field. Text fields are tokenized before indexing and are
17
+ # the only fields searched in fulltext searches. If a block is passed,
18
+ # create a virtual field; otherwise create an attribute field.
19
+ #
20
+ # If options are passed, they will be applied to all the given fields.
21
+ #
22
+ # ==== Parameters
23
+ #
24
+ # names...<Symbol>:: One or more field names
25
+ #
26
+ # ==== Options
27
+ #
28
+ # :boost<Float>::
29
+ # Index-time boost that should be applied to this field for keyword search
30
+ # :default_boost<Float>::
31
+ # Default search-time boost to apply to this field during keyword
32
+ # search. Can be overriden with DSL::Fulltext#fields or
33
+ # DSL::Fulltext#boost_fields method.
34
+ #
35
+ def text(*names, &block)
36
+ options = names.pop if names.last.is_a?(Hash)
37
+ names.each do |name|
38
+ @setup.add_text_field_factory(
39
+ name,
40
+ options || {},
41
+ &block
42
+ )
43
+ end
44
+ end
45
+
46
+ #
47
+ # Specify a method or block that returns the geographical coordinates
48
+ # associated with the document. The object returned must respond to #first
49
+ # and #last (e.g., a two-element Array); or to #lat and one of #lng, #lon,
50
+ # or #long
51
+ #
52
+ def coordinates(name = nil, &block)
53
+ @setup.set_coordinates_field(name, &block)
54
+ end
55
+
56
+ #
57
+ # Specify a document-level boost. As with fields, you have the option of
58
+ # passing an attribute name which will be called on each model, or a block
59
+ # to be evaluated in the model's context. As well as these two options,
60
+ # this method can also take a constant number, meaning that all indexed
61
+ # documents of this class will have the specified boost.
62
+ #
63
+ # ==== Parameters
64
+ #
65
+ # attr_name<Symbol,~.to_f>:: Attribute name to call or a numeric constant
66
+ #
67
+ def boost(attr_name = nil, &block)
68
+ @setup.add_document_boost(attr_name, &block)
69
+ end
70
+
71
+ # method_missing is used to provide access to typed fields, because
72
+ # developers should be able to add new Sunspot::Type implementations
73
+ # dynamically and have them recognized inside the Fields DSL. Like #text,
74
+ # these methods will create a VirtualField if a block is passed, or an
75
+ # AttributeField if not.
76
+ #
77
+ # ==== Example
78
+ #
79
+ # Sunspot.setup(File) do
80
+ # time :mtime
81
+ # end
82
+ #
83
+ # The call to +time+ will create a field of type Sunspot::Types::TimeType
84
+ #
85
+ def method_missing(method, *args, &block)
86
+ options = Util.extract_options_from(args)
87
+ type_const_name = "#{Util.camel_case(method.to_s.sub(/^dynamic_/, ''))}Type"
88
+ trie = options.delete(:trie)
89
+ type_const_name = "Trie#{type_const_name}" if trie
90
+ begin
91
+ type_class = Type.const_get(type_const_name)
92
+ rescue(NameError)
93
+ if trie
94
+ raise ArgumentError, "Trie fields are only valid for numeric and time types"
95
+ else
96
+ super(method, *args, &block)
97
+ end
98
+ end
99
+ type = type_class.instance
100
+ name = args.shift
101
+ if method.to_s =~ /^dynamic_/
102
+ if type.accepts_dynamic?
103
+ @setup.add_dynamic_field_factory(name, type, options, &block)
104
+ else
105
+ super(method, *args, &block)
106
+ end
107
+ else
108
+ @setup.add_field_factory(name, type, options, &block)
109
+ end
110
+ end
111
+ end
112
+ end
113
+ end