honkster-acts_as_solr 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (154) hide show
  1. data/CHANGE_LOG +233 -0
  2. data/LICENSE +19 -0
  3. data/README.markdown +118 -0
  4. data/README.rdoc +107 -0
  5. data/Rakefile +68 -0
  6. data/TESTING_THE_PLUGIN +25 -0
  7. data/VERSION +1 -0
  8. data/lib/acts_as_solr/acts_methods.rb +279 -0
  9. data/lib/acts_as_solr/class_methods.rb +236 -0
  10. data/lib/acts_as_solr/common_methods.rb +89 -0
  11. data/lib/acts_as_solr/deprecation.rb +61 -0
  12. data/lib/acts_as_solr/instance_methods.rb +166 -0
  13. data/lib/acts_as_solr/lazy_document.rb +18 -0
  14. data/lib/acts_as_solr/parser_methods.rb +201 -0
  15. data/lib/acts_as_solr/search_results.rb +68 -0
  16. data/lib/acts_as_solr/solr_fixtures.rb +13 -0
  17. data/lib/acts_as_solr/tasks/database.rake +16 -0
  18. data/lib/acts_as_solr/tasks/solr.rake +132 -0
  19. data/lib/acts_as_solr/tasks/test.rake +5 -0
  20. data/lib/acts_as_solr/tasks.rb +10 -0
  21. data/lib/acts_as_solr.rb +64 -0
  22. data/lib/solr/connection.rb +177 -0
  23. data/lib/solr/document.rb +75 -0
  24. data/lib/solr/exception.rb +13 -0
  25. data/lib/solr/field.rb +36 -0
  26. data/lib/solr/importer/array_mapper.rb +26 -0
  27. data/lib/solr/importer/delimited_file_source.rb +38 -0
  28. data/lib/solr/importer/hpricot_mapper.rb +27 -0
  29. data/lib/solr/importer/mapper.rb +51 -0
  30. data/lib/solr/importer/solr_source.rb +41 -0
  31. data/lib/solr/importer/xpath_mapper.rb +35 -0
  32. data/lib/solr/importer.rb +19 -0
  33. data/lib/solr/indexer.rb +52 -0
  34. data/lib/solr/request/add_document.rb +58 -0
  35. data/lib/solr/request/base.rb +36 -0
  36. data/lib/solr/request/commit.rb +29 -0
  37. data/lib/solr/request/delete.rb +48 -0
  38. data/lib/solr/request/dismax.rb +46 -0
  39. data/lib/solr/request/index_info.rb +22 -0
  40. data/lib/solr/request/modify_document.rb +46 -0
  41. data/lib/solr/request/optimize.rb +19 -0
  42. data/lib/solr/request/ping.rb +36 -0
  43. data/lib/solr/request/select.rb +54 -0
  44. data/lib/solr/request/spellcheck.rb +30 -0
  45. data/lib/solr/request/standard.rb +402 -0
  46. data/lib/solr/request/update.rb +23 -0
  47. data/lib/solr/request.rb +26 -0
  48. data/lib/solr/response/add_document.rb +17 -0
  49. data/lib/solr/response/base.rb +42 -0
  50. data/lib/solr/response/commit.rb +15 -0
  51. data/lib/solr/response/delete.rb +13 -0
  52. data/lib/solr/response/dismax.rb +8 -0
  53. data/lib/solr/response/index_info.rb +26 -0
  54. data/lib/solr/response/modify_document.rb +17 -0
  55. data/lib/solr/response/optimize.rb +14 -0
  56. data/lib/solr/response/ping.rb +26 -0
  57. data/lib/solr/response/ruby.rb +42 -0
  58. data/lib/solr/response/select.rb +17 -0
  59. data/lib/solr/response/spellcheck.rb +20 -0
  60. data/lib/solr/response/standard.rb +60 -0
  61. data/lib/solr/response/xml.rb +39 -0
  62. data/lib/solr/response.rb +27 -0
  63. data/lib/solr/solrtasks.rb +27 -0
  64. data/lib/solr/util.rb +32 -0
  65. data/lib/solr/xml.rb +44 -0
  66. data/lib/solr.rb +26 -0
  67. data/solr/CHANGES.txt +1207 -0
  68. data/solr/LICENSE.txt +712 -0
  69. data/solr/NOTICE.txt +90 -0
  70. data/solr/etc/jetty.xml +205 -0
  71. data/solr/etc/webdefault.xml +379 -0
  72. data/solr/lib/easymock.jar +0 -0
  73. data/solr/lib/jetty-6.1.3.jar +0 -0
  74. data/solr/lib/jetty-util-6.1.3.jar +0 -0
  75. data/solr/lib/jsp-2.1/ant-1.6.5.jar +0 -0
  76. data/solr/lib/jsp-2.1/core-3.1.1.jar +0 -0
  77. data/solr/lib/jsp-2.1/jsp-2.1.jar +0 -0
  78. data/solr/lib/jsp-2.1/jsp-api-2.1.jar +0 -0
  79. data/solr/lib/servlet-api-2.4.jar +0 -0
  80. data/solr/lib/servlet-api-2.5-6.1.3.jar +0 -0
  81. data/solr/lib/xpp3-1.1.3.4.O.jar +0 -0
  82. data/solr/solr/README.txt +52 -0
  83. data/solr/solr/bin/abc +176 -0
  84. data/solr/solr/bin/abo +176 -0
  85. data/solr/solr/bin/backup +108 -0
  86. data/solr/solr/bin/backupcleaner +142 -0
  87. data/solr/solr/bin/commit +128 -0
  88. data/solr/solr/bin/optimize +129 -0
  89. data/solr/solr/bin/readercycle +129 -0
  90. data/solr/solr/bin/rsyncd-disable +77 -0
  91. data/solr/solr/bin/rsyncd-enable +76 -0
  92. data/solr/solr/bin/rsyncd-start +145 -0
  93. data/solr/solr/bin/rsyncd-stop +105 -0
  94. data/solr/solr/bin/scripts-util +83 -0
  95. data/solr/solr/bin/snapcleaner +148 -0
  96. data/solr/solr/bin/snapinstaller +168 -0
  97. data/solr/solr/bin/snappuller +248 -0
  98. data/solr/solr/bin/snappuller-disable +77 -0
  99. data/solr/solr/bin/snappuller-enable +77 -0
  100. data/solr/solr/bin/snapshooter +109 -0
  101. data/solr/solr/conf/admin-extra.html +31 -0
  102. data/solr/solr/conf/protwords.txt +21 -0
  103. data/solr/solr/conf/schema.xml +126 -0
  104. data/solr/solr/conf/scripts.conf +24 -0
  105. data/solr/solr/conf/solrconfig.xml +458 -0
  106. data/solr/solr/conf/stopwords.txt +57 -0
  107. data/solr/solr/conf/synonyms.txt +31 -0
  108. data/solr/solr/conf/xslt/example.xsl +132 -0
  109. data/solr/solr/conf/xslt/example_atom.xsl +63 -0
  110. data/solr/solr/conf/xslt/example_rss.xsl +62 -0
  111. data/solr/start.jar +0 -0
  112. data/solr/webapps/solr.war +0 -0
  113. data/test/config/solr.yml +2 -0
  114. data/test/db/connections/mysql/connection.rb +10 -0
  115. data/test/db/connections/sqlite/connection.rb +8 -0
  116. data/test/db/migrate/001_create_books.rb +15 -0
  117. data/test/db/migrate/002_create_movies.rb +12 -0
  118. data/test/db/migrate/003_create_categories.rb +11 -0
  119. data/test/db/migrate/004_create_electronics.rb +16 -0
  120. data/test/db/migrate/005_create_authors.rb +12 -0
  121. data/test/db/migrate/006_create_postings.rb +9 -0
  122. data/test/db/migrate/007_create_posts.rb +13 -0
  123. data/test/db/migrate/008_create_gadgets.rb +11 -0
  124. data/test/fixtures/authors.yml +9 -0
  125. data/test/fixtures/books.yml +13 -0
  126. data/test/fixtures/categories.yml +7 -0
  127. data/test/fixtures/db_definitions/mysql.sql +41 -0
  128. data/test/fixtures/electronics.yml +49 -0
  129. data/test/fixtures/movies.yml +9 -0
  130. data/test/fixtures/postings.yml +10 -0
  131. data/test/functional/acts_as_solr_test.rb +413 -0
  132. data/test/functional/association_indexing_test.rb +37 -0
  133. data/test/functional/faceted_search_test.rb +163 -0
  134. data/test/functional/multi_solr_search_test.rb +51 -0
  135. data/test/models/author.rb +10 -0
  136. data/test/models/book.rb +10 -0
  137. data/test/models/category.rb +8 -0
  138. data/test/models/electronic.rb +21 -0
  139. data/test/models/gadget.rb +9 -0
  140. data/test/models/movie.rb +17 -0
  141. data/test/models/novel.rb +2 -0
  142. data/test/models/post.rb +3 -0
  143. data/test/models/posting.rb +11 -0
  144. data/test/test_helper.rb +51 -0
  145. data/test/unit/acts_methods_shoulda.rb +68 -0
  146. data/test/unit/class_methods_shoulda.rb +85 -0
  147. data/test/unit/common_methods_shoulda.rb +111 -0
  148. data/test/unit/instance_methods_shoulda.rb +318 -0
  149. data/test/unit/lazy_document_shoulda.rb +34 -0
  150. data/test/unit/parser_instance.rb +19 -0
  151. data/test/unit/parser_methods_shoulda.rb +268 -0
  152. data/test/unit/solr_instance.rb +44 -0
  153. data/test/unit/test_helper.rb +22 -0
  154. metadata +239 -0
@@ -0,0 +1,89 @@
1
+ module ActsAsSolr #:nodoc:
2
+
3
+ module CommonMethods
4
+
5
+ # Converts field types into Solr types
6
+ def get_solr_field_type(field_type)
7
+ if field_type.is_a?(Symbol)
8
+ case field_type
9
+ when :float
10
+ return "f"
11
+ when :integer
12
+ return "i"
13
+ when :boolean
14
+ return "b"
15
+ when :string
16
+ return "s"
17
+ when :date
18
+ return "d"
19
+ when :range_float
20
+ return "rf"
21
+ when :range_integer
22
+ return "ri"
23
+ when :facet
24
+ return "facet"
25
+ when :text
26
+ return "t"
27
+ else
28
+ raise "Unknown field_type symbol: #{field_type}"
29
+ end
30
+ elsif field_type.is_a?(String)
31
+ return field_type
32
+ else
33
+ raise "Unknown field_type class: #{field_type.class}: #{field_type}"
34
+ end
35
+ end
36
+
37
+ # Sets a default value when value being set is nil.
38
+ def set_value_if_nil(field_type)
39
+ case field_type
40
+ when "b", :boolean
41
+ return "false"
42
+ when "s", "t", "d", :date, :string, :text
43
+ return ""
44
+ when "f", "rf", :float, :range_float
45
+ return 0.00
46
+ when "i", "ri", :integer, :range_integer
47
+ return 0
48
+ else
49
+ return ""
50
+ end
51
+ end
52
+
53
+ # Sends an add command to Solr
54
+ def solr_add(add_xml)
55
+ ActsAsSolr::Post.execute(Solr::Request::AddDocument.new(add_xml))
56
+ end
57
+
58
+ # Sends the delete command to Solr
59
+ def solr_delete(solr_ids)
60
+ ActsAsSolr::Post.execute(Solr::Request::Delete.new(:id => solr_ids))
61
+ end
62
+
63
+ # Sends the commit command to Solr
64
+ def solr_commit
65
+ ActsAsSolr::Post.execute(Solr::Request::Commit.new)
66
+ end
67
+
68
+ # Optimizes the Solr index. Solr says:
69
+ #
70
+ # Optimizations can take nearly ten minutes to run.
71
+ # We are presuming optimizations should be run once following large
72
+ # batch-like updates to the collection and/or once a day.
73
+ #
74
+ # One of the solutions for this would be to create a cron job that
75
+ # runs every day at midnight and optmizes the index:
76
+ # 0 0 * * * /your_rails_dir/script/runner -e production "Model.solr_optimize"
77
+ #
78
+ def solr_optimize
79
+ ActsAsSolr::Post.execute(Solr::Request::Optimize.new)
80
+ end
81
+
82
+ # Returns the id for the given instance
83
+ def record_id(object)
84
+ eval "object.#{object.class.primary_key}"
85
+ end
86
+
87
+ end
88
+
89
+ end
@@ -0,0 +1,61 @@
1
+ module ActsAsSolr #:nodoc:
2
+
3
+ class Post
4
+ def initialize(body, mode = :search)
5
+ @body = body
6
+ @mode = mode
7
+ puts "The method ActsAsSolr::Post.new(body, mode).execute_post is depracated. " +
8
+ "Use ActsAsSolr::Post.execute(body, mode) instead!"
9
+ end
10
+
11
+ def execute_post
12
+ ActsAsSolr::Post.execute(@body, @mode)
13
+ end
14
+ end
15
+
16
+ module ClassMethods
17
+ def find_with_facet(query, options={})
18
+ Deprecation.plog "The method find_with_facet is deprecated. Use find_by_solr instead, passing the " +
19
+ "arguments the same way you used to do with find_with_facet."
20
+ find_by_solr(query, options)
21
+ end
22
+ end
23
+
24
+ class Deprecation
25
+ # Validates the options passed during query
26
+ def self.validate_query options={}
27
+ if options[:field_types]
28
+ plog "The option :field_types for searching is deprecated. " +
29
+ "The field types are automatically traced back when you specify a field type in your model."
30
+ end
31
+ if options[:sort_by]
32
+ plog "The option :sort_by is deprecated, use :order instead!"
33
+ options[:order] ||= options[:sort_by]
34
+ end
35
+ if options[:start]
36
+ plog "The option :start is deprecated, use :offset instead!"
37
+ options[:offset] ||= options[:start]
38
+ end
39
+ if options[:rows]
40
+ plog "The option :rows is deprecated, use :limit instead!"
41
+ options[:limit] ||= options[:rows]
42
+ end
43
+ end
44
+
45
+ # Validates the options passed during indexing
46
+ def self.validate_index options={}
47
+ if options[:background]
48
+ plog "The :background option is being deprecated. There are better and more efficient " +
49
+ "ways to handle delayed saving of your records."
50
+ end
51
+ end
52
+
53
+ # This will print the text to stdout and log the text
54
+ # if rails logger is available
55
+ def self.plog text
56
+ puts text
57
+ RAILS_DEFAULT_LOGGER.warn text if defined? RAILS_DEFAULT_LOGGER
58
+ end
59
+ end
60
+
61
+ end
@@ -0,0 +1,166 @@
1
+ module ActsAsSolr #:nodoc:
2
+
3
+ module InstanceMethods
4
+
5
+ # Solr id is <class.name>:<id> to be unique across all models
6
+ def solr_id
7
+ "#{self.class.name}:#{record_id(self)}"
8
+ end
9
+
10
+ # saves to the Solr index
11
+ def solr_save
12
+ return true if indexing_disabled?
13
+ if evaluate_condition(:if, self)
14
+ logger.debug "solr_save: #{self.class.name} : #{record_id(self)}"
15
+ solr_add to_solr_doc
16
+ solr_commit if configuration[:auto_commit]
17
+ true
18
+ else
19
+ solr_destroy
20
+ end
21
+ end
22
+
23
+ def indexing_disabled?
24
+ evaluate_condition(:offline, self) || !configuration[:if]
25
+ end
26
+
27
+ # remove from index
28
+ def solr_destroy
29
+ return true if indexing_disabled?
30
+ logger.debug "solr_destroy: #{self.class.name} : #{record_id(self)}"
31
+ solr_delete solr_id
32
+ solr_commit if configuration[:auto_commit]
33
+ true
34
+ end
35
+
36
+ # convert instance to Solr document
37
+ def to_solr_doc
38
+ logger.debug "to_solr_doc: creating doc for class: #{self.class.name}, id: #{record_id(self)}"
39
+ doc = Solr::Document.new
40
+ doc.boost = validate_boost(configuration[:boost]) if configuration[:boost]
41
+
42
+ doc << {:id => solr_id,
43
+ solr_configuration[:type_field] => self.class.name,
44
+ solr_configuration[:primary_key_field] => record_id(self).to_s}
45
+
46
+ # iterate through the fields and add them to the document,
47
+ configuration[:solr_fields].each do |field_name, options|
48
+ #field_type = configuration[:facets] && configuration[:facets].include?(field) ? :facet : :text
49
+
50
+ field_boost = options[:boost] || solr_configuration[:default_boost]
51
+ field_type = get_solr_field_type(options[:type])
52
+ solr_name = options[:as] || field_name
53
+
54
+ value = self.send("#{field_name}_for_solr")
55
+ value = set_value_if_nil(field_type) if value.to_s == ""
56
+
57
+ # add the field to the document, but only if it's not the id field
58
+ # or the type field (from single table inheritance), since these
59
+ # fields have already been added above.
60
+ if field_name.to_s != self.class.primary_key and field_name.to_s != "type"
61
+ suffix = get_solr_field_type(field_type)
62
+ # This next line ensures that e.g. nil dates are excluded from the
63
+ # document, since they choke Solr. Also ignores e.g. empty strings,
64
+ # but these can't be searched for anyway:
65
+ # http://www.mail-archive.com/solr-dev@lucene.apache.org/msg05423.html
66
+ next if value.nil? || value.to_s.strip.empty?
67
+ [value].flatten.each do |v|
68
+ v = set_value_if_nil(suffix) if value.to_s == ""
69
+ field = Solr::Field.new("#{solr_name}_#{suffix}" => ERB::Util.html_escape(v.to_s))
70
+ field.boost = validate_boost(field_boost)
71
+ doc << field
72
+ end
73
+ end
74
+ end
75
+
76
+ add_includes(doc)
77
+ logger.debug doc.to_xml
78
+ doc
79
+ end
80
+
81
+ private
82
+ def add_includes(doc)
83
+ if configuration[:solr_includes].respond_to?(:each)
84
+ configuration[:solr_includes].each do |association, options|
85
+ data = options[:multivalued] ? [] : ""
86
+ field_name = options[:as] || association.to_s.singularize
87
+ field_type = get_solr_field_type(options[:type])
88
+ field_boost = options[:boost] || solr_configuration[:default_boost]
89
+ suffix = get_solr_field_type(field_type)
90
+ case self.class.reflect_on_association(association).macro
91
+ when :has_many, :has_and_belongs_to_many
92
+ records = self.send(association).to_a
93
+ unless records.empty?
94
+ records.each {|r| data << include_value(r, options)}
95
+ [data].flatten.each do |value|
96
+ field = Solr::Field.new("#{field_name}_#{suffix}" => value)
97
+ field.boost = validate_boost(field_boost)
98
+ doc << field
99
+ end
100
+ end
101
+ when :has_one, :belongs_to
102
+ record = self.send(association)
103
+ unless record.nil?
104
+ doc["#{field_name}_#{suffix}"] = include_value(record, options)
105
+ end
106
+ end
107
+ end
108
+ end
109
+ end
110
+
111
+ def include_value(record, options)
112
+ if options[:using].is_a? Proc
113
+ options[:using].call(record)
114
+ elsif options[:using].is_a? Symbol
115
+ record.send(options[:using])
116
+ else
117
+ record.attributes.inject([]){|k,v| k << "#{v.first}=#{ERB::Util.html_escape(v.last)}"}.join(" ")
118
+ end
119
+ end
120
+
121
+ def validate_boost(boost)
122
+ boost_value = case boost
123
+ when Float
124
+ return solr_configuration[:default_boost] if boost < 0
125
+ boost
126
+ when Proc
127
+ boost.call(self)
128
+ when Symbol
129
+ if self.respond_to?(boost)
130
+ self.send(boost)
131
+ end
132
+ end
133
+
134
+ boost_value || solr_configuration[:default_boost]
135
+ end
136
+
137
+ def condition_block?(condition)
138
+ condition.respond_to?("call") && (condition.arity == 1 || condition.arity == -1)
139
+ end
140
+
141
+ def evaluate_condition(which_condition, field)
142
+ condition = configuration[which_condition]
143
+ case condition
144
+ when Symbol
145
+ field.send(condition)
146
+ when String
147
+ eval(condition, binding)
148
+ when FalseClass, NilClass
149
+ false
150
+ when TrueClass
151
+ true
152
+ else
153
+ if condition_block?(condition)
154
+ condition.call(field)
155
+ else
156
+ raise(
157
+ ArgumentError,
158
+ "The :#{which_condition} option has to be either a symbol, string (to be eval'ed), proc/method, true/false, or " +
159
+ "class implementing a static validation method"
160
+ )
161
+ end
162
+ end
163
+ end
164
+
165
+ end
166
+ end
@@ -0,0 +1,18 @@
1
+ module ActsAsSolr
2
+ class LazyDocument
3
+ attr_reader :id, :clazz
4
+
5
+ def initialize(id, clazz)
6
+ @id = id
7
+ @clazz = clazz
8
+ end
9
+
10
+ def method_missing(name, *args)
11
+ unless @__instance
12
+ @__instance = @clazz.find(@id)
13
+ end
14
+
15
+ @__instance.send(name, *args)
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,201 @@
1
+ module ActsAsSolr #:nodoc:
2
+ module ParserMethods
3
+ protected
4
+
5
+ # Method used by mostly all the ClassMethods when doing a search
6
+ def parse_query(query=nil, options={}, models=nil)
7
+ valid_options = [:offset, :limit, :facets, :models, :results_format, :order, :scores, :operator, :include, :lazy]
8
+ query_options = {}
9
+
10
+ return nil if (query.nil? || query.strip == '')
11
+
12
+ raise "Invalid parameters: #{(options.keys - valid_options).join(',')}" unless (options.keys - valid_options).empty?
13
+ begin
14
+ Deprecation.validate_query(options)
15
+ query_options[:start] = options[:offset]
16
+ query_options[:rows] = options[:limit]
17
+ query_options[:operator] = options[:operator]
18
+
19
+ # first steps on the facet parameter processing
20
+ if options[:facets]
21
+ query_options[:facets] = {}
22
+ query_options[:facets][:limit] = -1 # TODO: make this configurable
23
+ query_options[:facets][:sort] = :count if options[:facets][:sort]
24
+ query_options[:facets][:mincount] = 0
25
+ query_options[:facets][:mincount] = 1 if options[:facets][:zeros] == false
26
+ # override the :zeros (it's deprecated anyway) if :mincount exists
27
+ query_options[:facets][:mincount] = options[:facets][:mincount] if options[:facets][:mincount]
28
+ query_options[:facets][:fields] = options[:facets][:fields].collect{|k| "#{k}_facet"} if options[:facets][:fields]
29
+ query_options[:filter_queries] = replace_types([*options[:facets][:browse]].collect{|k| "#{k.sub!(/ *: */,"_facet:")}"}) if options[:facets][:browse]
30
+ query_options[:facets][:queries] = replace_types(options[:facets][:query].collect{|k| "#{k.sub!(/ *: */,"_t:")}"}) if options[:facets][:query]
31
+
32
+
33
+ if options[:facets][:dates]
34
+ query_options[:date_facets] = {}
35
+ # if options[:facets][:dates][:fields] exists then :start, :end, and :gap must be there
36
+ if options[:facets][:dates][:fields]
37
+ [:start, :end, :gap].each { |k| raise "#{k} must be present in faceted date query" unless options[:facets][:dates].include?(k) }
38
+ query_options[:date_facets][:fields] = []
39
+ options[:facets][:dates][:fields].each { |f|
40
+ if f.kind_of? Hash
41
+ key = f.keys[0]
42
+ query_options[:date_facets][:fields] << {"#{key}_d" => f[key]}
43
+ validate_date_facet_other_options(f[key][:other]) if f[key][:other]
44
+ else
45
+ query_options[:date_facets][:fields] << "#{f}_d"
46
+ end
47
+ }
48
+ end
49
+
50
+ query_options[:date_facets][:start] = options[:facets][:dates][:start] if options[:facets][:dates][:start]
51
+ query_options[:date_facets][:end] = options[:facets][:dates][:end] if options[:facets][:dates][:end]
52
+ query_options[:date_facets][:gap] = options[:facets][:dates][:gap] if options[:facets][:dates][:gap]
53
+ query_options[:date_facets][:hardend] = options[:facets][:dates][:hardend] if options[:facets][:dates][:hardend]
54
+ query_options[:date_facets][:filter] = replace_types([*options[:facets][:dates][:filter]].collect{|k| "#{k.sub!(/ *:(?!\d) */,"_d:")}"}) if options[:facets][:dates][:filter]
55
+
56
+ if options[:facets][:dates][:other]
57
+ validate_date_facet_other_options(options[:facets][:dates][:other])
58
+ query_options[:date_facets][:other] = options[:facets][:dates][:other]
59
+ end
60
+
61
+ end
62
+ end
63
+
64
+ if models.nil?
65
+ # TODO: use a filter query for type, allowing Solr to cache it individually
66
+ models = "AND #{solr_type_condition}"
67
+ field_list = solr_configuration[:primary_key_field]
68
+ else
69
+ field_list = "id"
70
+ end
71
+
72
+ query_options[:field_list] = [field_list, 'score']
73
+ query = "(#{query.gsub(/ *: */,"_t:")}) #{models}"
74
+ order = options[:order].split(/\s*,\s*/).collect{|e| e.gsub(/\s+/,'_t ').gsub(/\bscore_t\b/, 'score') }.join(',') if options[:order]
75
+ query_options[:query] = replace_types([query])[0] # TODO adjust replace_types to work with String or Array
76
+
77
+ if options[:order]
78
+ # TODO: set the sort parameter instead of the old ;order. style.
79
+ query_options[:query] << ';' << replace_types([order], false)[0]
80
+ end
81
+
82
+ ActsAsSolr::Post.execute(Solr::Request::Standard.new(query_options))
83
+ rescue
84
+ raise "There was a problem executing your search: #{$!} in #{$!.backtrace.first}"
85
+ end
86
+ end
87
+
88
+ def solr_type_condition
89
+ subclasses.inject("(#{solr_configuration[:type_field]}:#{self.name}") do |condition, subclass|
90
+ condition << " OR #{solr_configuration[:type_field]}:#{subclass.name}"
91
+ end << ')'
92
+ end
93
+
94
+ # Parses the data returned from Solr
95
+ def parse_results(solr_data, options = {})
96
+ results = {
97
+ :docs => [],
98
+ :total => 0
99
+ }
100
+
101
+ configuration = {
102
+ :format => :objects
103
+ }
104
+ results.update(:facets => {'facet_fields' => []}) if options[:facets]
105
+ return SearchResults.new(results) if (solr_data.nil? || solr_data.total_hits == 0)
106
+
107
+ configuration.update(options) if options.is_a?(Hash)
108
+
109
+ ids = solr_data.hits.collect {|doc| doc["#{solr_configuration[:primary_key_field]}"]}.flatten
110
+
111
+ result = find_objects(ids, options, configuration)
112
+
113
+ add_scores(result, solr_data) if configuration[:format] == :objects && options[:scores]
114
+
115
+ results.update(:facets => solr_data.data['facet_counts']) if options[:facets]
116
+ results.update({:docs => result, :total => solr_data.total_hits, :max_score => solr_data.max_score, :query_time => solr_data.data['responseHeader']['QTime']})
117
+ SearchResults.new(results)
118
+ end
119
+
120
+
121
+ def find_objects(ids, options, configuration)
122
+ result = if configuration[:lazy] && configuration[:format] != :ids
123
+ ids.collect {|id| ActsAsSolr::LazyDocument.new(id, self)}
124
+ elsif configuration[:format] == :objects
125
+ conditions = [ "#{self.table_name}.#{primary_key} in (?)", ids ]
126
+ find_options = {:conditions => conditions}
127
+ find_options[:include] = options[:include] if options[:include]
128
+ result = reorder(self.find(:all, find_options), ids)
129
+ else
130
+ ids
131
+ end
132
+
133
+ result
134
+ end
135
+
136
+ # Reorders the instances keeping the order returned from Solr
137
+ def reorder(things, ids)
138
+ ordered_things = Array.new(things.size)
139
+ raise "Out of sync! Found #{ids.size} items in index, but only #{things.size} were found in database!" unless things.size == ids.size
140
+ things.each do |thing|
141
+ position = ids.index(thing.id)
142
+ ordered_things[position] = thing
143
+ end
144
+ ordered_things
145
+ end
146
+
147
+ # Replaces the field types based on the types (if any) specified
148
+ # on the acts_as_solr call
149
+ def replace_types(strings, include_colon=true)
150
+ suffix = include_colon ? ":" : ""
151
+ if configuration[:solr_fields]
152
+ configuration[:solr_fields].each do |name, options|
153
+ solr_name = options[:as] || name.to_s
154
+ solr_type = get_solr_field_type(options[:type])
155
+ field = "#{solr_name}_#{solr_type}#{suffix}"
156
+ strings.each_with_index {|s,i| strings[i] = s.gsub(/#{solr_name.to_s}_t#{suffix}/,field) }
157
+ end
158
+ end
159
+ if configuration[:solr_includes]
160
+ configuration[:solr_includes].each do |association, options|
161
+ solr_name = options[:as] || association.to_s.singularize
162
+ solr_type = get_solr_field_type(options[:type])
163
+ field = "#{solr_name}_#{solr_type}#{suffix}"
164
+ strings.each_with_index {|s,i| strings[i] = s.gsub(/#{solr_name.to_s}_t#{suffix}/,field) }
165
+ end
166
+ end
167
+ strings
168
+ end
169
+
170
+ # Adds the score to each one of the instances found
171
+ def add_scores(results, solr_data)
172
+ with_score = []
173
+ solr_data.hits.each do |doc|
174
+ with_score.push([doc["score"],
175
+ results.find {|record| scorable_record?(record, doc) }])
176
+ end
177
+ with_score.each do |score, object|
178
+ class << object; attr_accessor :solr_score; end
179
+ object.solr_score = score
180
+ end
181
+ end
182
+
183
+ def scorable_record?(record, doc)
184
+ doc_id = doc["#{solr_configuration[:primary_key_field]}"]
185
+ if doc_id.nil?
186
+ doc_id = doc["id"]
187
+ "#{record.class.name}:#{record_id(record)}" == doc_id.first.to_s
188
+ else
189
+ record_id(record).to_s == doc_id.to_s
190
+ end
191
+ end
192
+
193
+ def validate_date_facet_other_options(options)
194
+ valid_other_options = [:after, :all, :before, :between, :none]
195
+ options = [options] unless options.kind_of? Array
196
+ bad_options = options.map {|x| x.to_sym} - valid_other_options
197
+ raise "Invalid option#{'s' if bad_options.size > 1} for faceted date's other param: #{bad_options.join(', ')}. May only be one of :after, :all, :before, :between, :none" if bad_options.size > 0
198
+ end
199
+
200
+ end
201
+ end