thinking-sphinx 3.0.1 → 3.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 (45) hide show
  1. checksums.yaml +7 -0
  2. data/.travis.yml +5 -2
  3. data/Appraisals +4 -0
  4. data/Gemfile +1 -1
  5. data/HISTORY +18 -0
  6. data/README.textile +7 -7
  7. data/gemfiles/rails_3_1.gemfile +1 -1
  8. data/gemfiles/rails_3_2.gemfile +1 -1
  9. data/gemfiles/rails_4_0.gemfile +11 -0
  10. data/lib/thinking_sphinx.rb +1 -0
  11. data/lib/thinking_sphinx/active_record/associations.rb +8 -1
  12. data/lib/thinking_sphinx/active_record/database_adapters/abstract_adapter.rb +4 -0
  13. data/lib/thinking_sphinx/active_record/database_adapters/mysql_adapter.rb +4 -0
  14. data/lib/thinking_sphinx/active_record/filtered_reflection.rb +32 -14
  15. data/lib/thinking_sphinx/active_record/sql_builder.rb +4 -4
  16. data/lib/thinking_sphinx/active_record/sql_source.rb +3 -1
  17. data/lib/thinking_sphinx/active_record/sql_source/template.rb +5 -1
  18. data/lib/thinking_sphinx/capistrano.rb +6 -7
  19. data/lib/thinking_sphinx/configuration.rb +56 -33
  20. data/lib/thinking_sphinx/core/index.rb +1 -1
  21. data/lib/thinking_sphinx/errors.rb +2 -0
  22. data/lib/thinking_sphinx/masks/group_enumerators_mask.rb +4 -0
  23. data/lib/thinking_sphinx/masks/pagination_mask.rb +4 -0
  24. data/lib/thinking_sphinx/masks/scopes_mask.rb +2 -2
  25. data/lib/thinking_sphinx/masks/weight_enumerator_mask.rb +4 -0
  26. data/lib/thinking_sphinx/middlewares/active_record_translator.rb +4 -3
  27. data/lib/thinking_sphinx/middlewares/sphinxql.rb +15 -3
  28. data/lib/thinking_sphinx/railtie.rb +13 -0
  29. data/lib/thinking_sphinx/search.rb +3 -2
  30. data/lib/thinking_sphinx/search/query.rb +4 -0
  31. data/spec/acceptance/specifying_sql_spec.rb +4 -4
  32. data/spec/internal/app/models/tweet.rb +1 -1
  33. data/spec/thinking_sphinx/active_record/associations_spec.rb +49 -4
  34. data/spec/thinking_sphinx/active_record/sql_builder_spec.rb +3 -3
  35. data/spec/thinking_sphinx/active_record/sql_source_spec.rb +14 -1
  36. data/spec/thinking_sphinx/configuration_spec.rb +25 -1
  37. data/spec/thinking_sphinx/connection_spec.rb +4 -4
  38. data/spec/thinking_sphinx/errors_spec.rb +7 -0
  39. data/spec/thinking_sphinx/middlewares/active_record_translator_spec.rb +3 -3
  40. data/spec/thinking_sphinx/middlewares/sphinxql_spec.rb +35 -4
  41. data/spec/thinking_sphinx/search/query_spec.rb +20 -0
  42. data/spec/thinking_sphinx/search_spec.rb +5 -0
  43. data/thinking-sphinx.gemspec +3 -3
  44. metadata +22 -48
  45. data/lib/thinking_sphinx/search/translator.rb +0 -50
@@ -12,7 +12,7 @@ describe ThinkingSphinx::Middlewares::ActiveRecordTranslator do
12
12
  let(:middleware) {
13
13
  ThinkingSphinx::Middlewares::ActiveRecordTranslator.new app }
14
14
  let(:context) { {:raw => [], :results => []} }
15
- let(:model) { double('model') }
15
+ let(:model) { double('model', :primary_key => :id) }
16
16
  let(:search) { double('search', :options => {}) }
17
17
 
18
18
  def raw_result(id, model_name)
@@ -50,11 +50,11 @@ describe ThinkingSphinx::Middlewares::ActiveRecordTranslator do
50
50
  end
51
51
 
52
52
  it "handles multiple models" do
53
- article_model = double('article model')
53
+ article_model = double('article model', :primary_key => :id)
54
54
  article_name = double('article name', :constantize => article_model)
55
55
  article = double('article instance', :id => 24)
56
56
 
57
- user_model = double('user model')
57
+ user_model = double('user model', :primary_key => :id)
58
58
  user_name = double('user name', :constantize => user_model)
59
59
  user = double('user instance', :id => 12)
60
60
 
@@ -17,10 +17,11 @@ describe ThinkingSphinx::Middlewares::SphinxQL do
17
17
  let(:context) { {} }
18
18
  let(:search) { double('search', :query => '', :options => {},
19
19
  :offset => 0, :per_page => 5) }
20
- let(:index_set) { [] }
20
+ let(:index_set) { [double(:name => 'article_core', :options => {})] }
21
21
  let(:sphinx_sql) { double('sphinx_sql', :from => true, :offset => true,
22
22
  :limit => true, :where => true, :matching => true) }
23
23
  let(:query) { double('query') }
24
+ let(:configuration) { double('configuration', :settings => {}) }
24
25
 
25
26
  before :each do
26
27
  stub_const 'Riddle::Query::Select', double(:new => sphinx_sql)
@@ -28,14 +29,14 @@ describe ThinkingSphinx::Middlewares::SphinxQL do
28
29
  stub_const 'ThinkingSphinx::Masks::GroupEnumeratorsMask', double
29
30
  stub_const 'ThinkingSphinx::IndexSet', double(:new => index_set)
30
31
 
31
- context.stub :search => search
32
+ context.stub :search => search, :configuration => configuration
32
33
  end
33
34
 
34
35
  describe '#call' do
35
36
  it "uses the indexes for the FROM clause" do
36
37
  index_set.replace [
37
- double('index', :name => 'article_core'),
38
- double('index', :name => 'user_core')
38
+ double('index', :name => 'article_core', :options => {}),
39
+ double('index', :name => 'user_core', :options => {})
39
40
  ]
40
41
 
41
42
  sphinx_sql.should_receive(:from).with('`article_core`', '`user_core`').
@@ -282,6 +283,36 @@ describe ThinkingSphinx::Middlewares::SphinxQL do
282
283
  middleware.call [context]
283
284
  end
284
285
 
286
+ it "uses index-defined field weights if they're available" do
287
+ index_set.first.options[:field_weights] = {:title => 3}
288
+
289
+ sphinx_sql.should_receive(:with_options).with(
290
+ hash_including(:field_weights => {:title => 3})
291
+ ).and_return(sphinx_sql)
292
+
293
+ middleware.call [context]
294
+ end
295
+
296
+ it "uses index-defined max matches if it's available" do
297
+ index_set.first.options[:max_matches] = 100
298
+
299
+ sphinx_sql.should_receive(:with_options).with(
300
+ hash_including(:max_matches => 100)
301
+ ).and_return(sphinx_sql)
302
+
303
+ middleware.call [context]
304
+ end
305
+
306
+ it "uses configuration-level max matches if set" do
307
+ configuration.settings['max_matches'] = 120
308
+
309
+ sphinx_sql.should_receive(:with_options).with(
310
+ hash_including(:max_matches => 120)
311
+ ).and_return(sphinx_sql)
312
+
313
+ middleware.call [context]
314
+ end
315
+
285
316
  it "uses any given ranker option" do
286
317
  search.options[:ranker] = 'proximity'
287
318
 
@@ -2,6 +2,7 @@ module ThinkingSphinx
2
2
  class Search; end
3
3
  end
4
4
 
5
+ require 'active_support/core_ext/object/blank'
5
6
  require './lib/thinking_sphinx/search/query'
6
7
 
7
8
  describe ThinkingSphinx::Search::Query do
@@ -42,5 +43,24 @@ describe ThinkingSphinx::Search::Query do
42
43
 
43
44
  query.to_s.should == '@sphinx_internal_class article'
44
45
  end
46
+
47
+ it "handles null values by removing them from the conditions hash" do
48
+ query = ThinkingSphinx::Search::Query.new '', :title => nil
49
+
50
+ query.to_s.should == ''
51
+ end
52
+
53
+ it "handles empty string values by removing them from the conditions hash" do
54
+ query = ThinkingSphinx::Search::Query.new '', :title => ''
55
+
56
+ query.to_s.should == ''
57
+ end
58
+
59
+ it "allows mixing of blank and non-blank conditions" do
60
+ query = ThinkingSphinx::Search::Query.new 'tasty', :title => 'pancakes',
61
+ :ingredients => nil
62
+
63
+ query.to_s.should == 'tasty @title pancakes'
64
+ end
45
65
  end
46
66
  end
@@ -111,6 +111,11 @@ describe ThinkingSphinx::Search do
111
111
  it "should allow for string arguments" do
112
112
  ThinkingSphinx::Search.new(:per_page => '10').per_page.should == 10
113
113
  end
114
+
115
+ it "allows setting of the per_page value" do
116
+ search.per_page(24)
117
+ search.per_page.should == 24
118
+ end
114
119
  end
115
120
 
116
121
  describe '#populate' do
@@ -3,7 +3,7 @@ $:.push File.expand_path('../lib', __FILE__)
3
3
 
4
4
  Gem::Specification.new do |s|
5
5
  s.name = 'thinking-sphinx'
6
- s.version = '3.0.1'
6
+ s.version = '3.0.2'
7
7
  s.platform = Gem::Platform::RUBY
8
8
  s.authors = ["Pat Allan"]
9
9
  s.email = ["pat@freelancing-gods.com"]
@@ -27,7 +27,7 @@ Gem::Specification.new do |s|
27
27
  s.add_runtime_dependency 'riddle', '>= 1.5.4'
28
28
 
29
29
  s.add_development_dependency 'appraisal', '~> 0.4.0'
30
- s.add_development_dependency 'combustion', '~> 0.3.3'
30
+ s.add_development_dependency 'combustion', '~> 0.4.0'
31
31
  s.add_development_dependency 'database_cleaner', '~> 0.7.1'
32
- s.add_development_dependency 'rspec', '~> 2.11.0'
32
+ s.add_development_dependency 'rspec', '~> 2.13.0'
33
33
  end
metadata CHANGED
@@ -1,100 +1,88 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: thinking-sphinx
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.0.1
5
- prerelease:
4
+ version: 3.0.2
6
5
  platform: ruby
7
6
  authors:
8
7
  - Pat Allan
9
8
  autorequire:
10
9
  bindir: bin
11
10
  cert_chain: []
12
- date: 2013-02-04 00:00:00.000000000 Z
11
+ date: 2013-03-23 00:00:00.000000000 Z
13
12
  dependencies:
14
13
  - !ruby/object:Gem::Dependency
15
14
  name: activerecord
16
15
  requirement: !ruby/object:Gem::Requirement
17
- none: false
18
16
  requirements:
19
- - - ! '>='
17
+ - - '>='
20
18
  - !ruby/object:Gem::Version
21
19
  version: 3.1.0
22
20
  type: :runtime
23
21
  prerelease: false
24
22
  version_requirements: !ruby/object:Gem::Requirement
25
- none: false
26
23
  requirements:
27
- - - ! '>='
24
+ - - '>='
28
25
  - !ruby/object:Gem::Version
29
26
  version: 3.1.0
30
27
  - !ruby/object:Gem::Dependency
31
28
  name: builder
32
29
  requirement: !ruby/object:Gem::Requirement
33
- none: false
34
30
  requirements:
35
- - - ! '>='
31
+ - - '>='
36
32
  - !ruby/object:Gem::Version
37
33
  version: 2.1.2
38
34
  type: :runtime
39
35
  prerelease: false
40
36
  version_requirements: !ruby/object:Gem::Requirement
41
- none: false
42
37
  requirements:
43
- - - ! '>='
38
+ - - '>='
44
39
  - !ruby/object:Gem::Version
45
40
  version: 2.1.2
46
41
  - !ruby/object:Gem::Dependency
47
42
  name: middleware
48
43
  requirement: !ruby/object:Gem::Requirement
49
- none: false
50
44
  requirements:
51
- - - ! '>='
45
+ - - '>='
52
46
  - !ruby/object:Gem::Version
53
47
  version: 0.1.0
54
48
  type: :runtime
55
49
  prerelease: false
56
50
  version_requirements: !ruby/object:Gem::Requirement
57
- none: false
58
51
  requirements:
59
- - - ! '>='
52
+ - - '>='
60
53
  - !ruby/object:Gem::Version
61
54
  version: 0.1.0
62
55
  - !ruby/object:Gem::Dependency
63
56
  name: innertube
64
57
  requirement: !ruby/object:Gem::Requirement
65
- none: false
66
58
  requirements:
67
- - - ! '>='
59
+ - - '>='
68
60
  - !ruby/object:Gem::Version
69
61
  version: 1.0.2
70
62
  type: :runtime
71
63
  prerelease: false
72
64
  version_requirements: !ruby/object:Gem::Requirement
73
- none: false
74
65
  requirements:
75
- - - ! '>='
66
+ - - '>='
76
67
  - !ruby/object:Gem::Version
77
68
  version: 1.0.2
78
69
  - !ruby/object:Gem::Dependency
79
70
  name: riddle
80
71
  requirement: !ruby/object:Gem::Requirement
81
- none: false
82
72
  requirements:
83
- - - ! '>='
73
+ - - '>='
84
74
  - !ruby/object:Gem::Version
85
75
  version: 1.5.4
86
76
  type: :runtime
87
77
  prerelease: false
88
78
  version_requirements: !ruby/object:Gem::Requirement
89
- none: false
90
79
  requirements:
91
- - - ! '>='
80
+ - - '>='
92
81
  - !ruby/object:Gem::Version
93
82
  version: 1.5.4
94
83
  - !ruby/object:Gem::Dependency
95
84
  name: appraisal
96
85
  requirement: !ruby/object:Gem::Requirement
97
- none: false
98
86
  requirements:
99
87
  - - ~>
100
88
  - !ruby/object:Gem::Version
@@ -102,7 +90,6 @@ dependencies:
102
90
  type: :development
103
91
  prerelease: false
104
92
  version_requirements: !ruby/object:Gem::Requirement
105
- none: false
106
93
  requirements:
107
94
  - - ~>
108
95
  - !ruby/object:Gem::Version
@@ -110,23 +97,20 @@ dependencies:
110
97
  - !ruby/object:Gem::Dependency
111
98
  name: combustion
112
99
  requirement: !ruby/object:Gem::Requirement
113
- none: false
114
100
  requirements:
115
101
  - - ~>
116
102
  - !ruby/object:Gem::Version
117
- version: 0.3.3
103
+ version: 0.4.0
118
104
  type: :development
119
105
  prerelease: false
120
106
  version_requirements: !ruby/object:Gem::Requirement
121
- none: false
122
107
  requirements:
123
108
  - - ~>
124
109
  - !ruby/object:Gem::Version
125
- version: 0.3.3
110
+ version: 0.4.0
126
111
  - !ruby/object:Gem::Dependency
127
112
  name: database_cleaner
128
113
  requirement: !ruby/object:Gem::Requirement
129
- none: false
130
114
  requirements:
131
115
  - - ~>
132
116
  - !ruby/object:Gem::Version
@@ -134,7 +118,6 @@ dependencies:
134
118
  type: :development
135
119
  prerelease: false
136
120
  version_requirements: !ruby/object:Gem::Requirement
137
- none: false
138
121
  requirements:
139
122
  - - ~>
140
123
  - !ruby/object:Gem::Version
@@ -142,19 +125,17 @@ dependencies:
142
125
  - !ruby/object:Gem::Dependency
143
126
  name: rspec
144
127
  requirement: !ruby/object:Gem::Requirement
145
- none: false
146
128
  requirements:
147
129
  - - ~>
148
130
  - !ruby/object:Gem::Version
149
- version: 2.11.0
131
+ version: 2.13.0
150
132
  type: :development
151
133
  prerelease: false
152
134
  version_requirements: !ruby/object:Gem::Requirement
153
- none: false
154
135
  requirements:
155
136
  - - ~>
156
137
  - !ruby/object:Gem::Version
157
- version: 2.11.0
138
+ version: 2.13.0
158
139
  description: An intelligent layer for ActiveRecord (via Rails and Sinatra) for the
159
140
  Sphinx full-text search tool.
160
141
  email:
@@ -174,6 +155,7 @@ files:
174
155
  - gemfiles/.gitignore
175
156
  - gemfiles/rails_3_1.gemfile
176
157
  - gemfiles/rails_3_2.gemfile
158
+ - gemfiles/rails_4_0.gemfile
177
159
  - lib/thinking-sphinx.rb
178
160
  - lib/thinking/sphinx.rb
179
161
  - lib/thinking_sphinx.rb
@@ -268,7 +250,6 @@ files:
268
250
  - lib/thinking_sphinx/search/merger.rb
269
251
  - lib/thinking_sphinx/search/query.rb
270
252
  - lib/thinking_sphinx/search/stale_ids_exception.rb
271
- - lib/thinking_sphinx/search/translator.rb
272
253
  - lib/thinking_sphinx/sinatra.rb
273
254
  - lib/thinking_sphinx/tasks.rb
274
255
  - lib/thinking_sphinx/test.rb
@@ -390,33 +371,26 @@ files:
390
371
  - thinking-sphinx.gemspec
391
372
  homepage: http://pat.github.com/ts/en
392
373
  licenses: []
374
+ metadata: {}
393
375
  post_install_message:
394
376
  rdoc_options: []
395
377
  require_paths:
396
378
  - lib
397
379
  required_ruby_version: !ruby/object:Gem::Requirement
398
- none: false
399
380
  requirements:
400
- - - ! '>='
381
+ - - '>='
401
382
  - !ruby/object:Gem::Version
402
383
  version: '0'
403
- segments:
404
- - 0
405
- hash: -122546704410277760
406
384
  required_rubygems_version: !ruby/object:Gem::Requirement
407
- none: false
408
385
  requirements:
409
- - - ! '>='
386
+ - - '>='
410
387
  - !ruby/object:Gem::Version
411
388
  version: '0'
412
- segments:
413
- - 0
414
- hash: -122546704410277760
415
389
  requirements: []
416
390
  rubyforge_project: thinking-sphinx
417
- rubygems_version: 1.8.23
391
+ rubygems_version: 2.0.0
418
392
  signing_key:
419
- specification_version: 3
393
+ specification_version: 4
420
394
  summary: A smart wrapper over Sphinx for ActiveRecord
421
395
  test_files:
422
396
  - spec/acceptance/association_scoping_spec.rb
@@ -1,50 +0,0 @@
1
- class ThinkingSphinx::Search::Translator
2
- attr_reader :raw, :excerpter
3
-
4
- def initialize(raw, excerpter)
5
- @raw, @excerpter = raw, excerpter
6
- end
7
-
8
- def to_active_record
9
- results_for_models # load now to avoid segfaults
10
- raw.collect { |row|
11
- ThinkingSphinx::Search::Glaze.new result_for(row), excerpter, row
12
- }
13
- end
14
-
15
- private
16
-
17
- def ids_for_model(model_name)
18
- raw.select { |row|
19
- row['sphinx_internal_class'] == model_name
20
- }.collect { |row|
21
- row['sphinx_internal_id']
22
- }
23
- end
24
-
25
- def model_names
26
- @model_names ||= raw.collect { |row|
27
- row['sphinx_internal_class']
28
- }.uniq
29
- end
30
-
31
- def result_for(row)
32
- results_for_models[row['sphinx_internal_class']].detect { |record|
33
- record.id == row['sphinx_internal_id']
34
- }
35
- end
36
-
37
- def results_for_models
38
- @results_for_models ||= model_names.inject({}) { |hash, name|
39
- ids = ids_for_model(name)
40
- hash[name] = name.constantize.where(:id => ids)
41
-
42
- stale_ids = ids - hash[name].collect(&:id)
43
- if stale_ids.any?
44
- raise ThinkingSphinx::Search::StaleIdsException, stale_ids
45
- end
46
-
47
- hash
48
- }
49
- end
50
- end