pixeltrix-thinking-sphinx 1.1.5 → 1.2.1
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.
- data/README.textile +147 -0
- data/lib/thinking_sphinx/active_record/attribute_updates.rb +48 -0
- data/lib/thinking_sphinx/active_record/delta.rb +14 -1
- data/lib/thinking_sphinx/active_record/scopes.rb +37 -0
- data/lib/thinking_sphinx/active_record.rb +46 -12
- data/lib/thinking_sphinx/adapters/abstract_adapter.rb +9 -1
- data/lib/thinking_sphinx/adapters/mysql_adapter.rb +3 -2
- data/lib/thinking_sphinx/adapters/postgresql_adapter.rb +12 -5
- data/lib/thinking_sphinx/association.rb +20 -0
- data/lib/thinking_sphinx/attribute.rb +187 -116
- data/lib/thinking_sphinx/class_facet.rb +15 -0
- data/lib/thinking_sphinx/configuration.rb +46 -14
- data/lib/thinking_sphinx/core/string.rb +3 -10
- data/lib/thinking_sphinx/deltas/datetime_delta.rb +3 -3
- data/lib/thinking_sphinx/deltas/default_delta.rb +9 -6
- data/lib/thinking_sphinx/deltas/delayed_delta/delta_job.rb +1 -1
- data/lib/thinking_sphinx/deltas/delayed_delta.rb +4 -2
- data/lib/thinking_sphinx/deltas.rb +14 -6
- data/lib/thinking_sphinx/deploy/capistrano.rb +98 -0
- data/lib/thinking_sphinx/excerpter.rb +22 -0
- data/lib/thinking_sphinx/facet.rb +68 -18
- data/lib/thinking_sphinx/facet_search.rb +134 -0
- data/lib/thinking_sphinx/field.rb +7 -97
- data/lib/thinking_sphinx/index/builder.rb +255 -201
- data/lib/thinking_sphinx/index.rb +28 -343
- data/lib/thinking_sphinx/property.rb +160 -0
- data/lib/thinking_sphinx/rails_additions.rb +7 -4
- data/lib/thinking_sphinx/search.rb +593 -587
- data/lib/thinking_sphinx/search_methods.rb +421 -0
- data/lib/thinking_sphinx/source/internal_properties.rb +46 -0
- data/lib/thinking_sphinx/source/sql.rb +128 -0
- data/lib/thinking_sphinx/source.rb +150 -0
- data/lib/thinking_sphinx/tasks.rb +45 -11
- data/lib/thinking_sphinx.rb +88 -14
- data/rails/init.rb +14 -0
- data/spec/{unit → lib}/thinking_sphinx/active_record/delta_spec.rb +7 -7
- data/spec/{unit → lib}/thinking_sphinx/active_record/has_many_association_spec.rb +0 -0
- data/spec/lib/thinking_sphinx/active_record/scopes_spec.rb +92 -0
- data/spec/{unit → lib}/thinking_sphinx/active_record_spec.rb +115 -42
- data/spec/{unit → lib}/thinking_sphinx/association_spec.rb +4 -5
- data/spec/lib/thinking_sphinx/attribute_spec.rb +465 -0
- data/spec/{unit → lib}/thinking_sphinx/configuration_spec.rb +118 -7
- data/spec/{unit → lib}/thinking_sphinx/core/string_spec.rb +0 -0
- data/spec/lib/thinking_sphinx/excerpter_spec.rb +49 -0
- data/spec/lib/thinking_sphinx/facet_search_spec.rb +176 -0
- data/spec/lib/thinking_sphinx/facet_spec.rb +302 -0
- data/spec/{unit → lib}/thinking_sphinx/field_spec.rb +26 -17
- data/spec/lib/thinking_sphinx/index/builder_spec.rb +355 -0
- data/spec/{unit → lib}/thinking_sphinx/index/faux_column_spec.rb +0 -0
- data/spec/{unit → lib}/thinking_sphinx/index_spec.rb +3 -12
- data/spec/lib/thinking_sphinx/rails_additions_spec.rb +191 -0
- data/spec/lib/thinking_sphinx/search_methods_spec.rb +152 -0
- data/spec/lib/thinking_sphinx/search_spec.rb +887 -0
- data/spec/lib/thinking_sphinx/source_spec.rb +217 -0
- data/spec/{unit → lib}/thinking_sphinx_spec.rb +30 -8
- data/tasks/distribution.rb +20 -1
- data/tasks/testing.rb +7 -15
- data/vendor/after_commit/init.rb +3 -0
- data/vendor/after_commit/lib/after_commit/active_record.rb +27 -4
- data/vendor/after_commit/lib/after_commit/connection_adapters.rb +1 -1
- data/vendor/after_commit/lib/after_commit.rb +4 -1
- data/vendor/riddle/lib/riddle/client/message.rb +4 -3
- data/vendor/riddle/lib/riddle/client.rb +3 -0
- data/vendor/riddle/lib/riddle/configuration/section.rb +8 -2
- data/vendor/riddle/lib/riddle/controller.rb +1 -1
- data/vendor/riddle/lib/riddle.rb +1 -1
- metadata +75 -39
- data/README +0 -107
- data/lib/thinking_sphinx/active_record/search.rb +0 -57
- data/lib/thinking_sphinx/collection.rb +0 -142
- data/lib/thinking_sphinx/facet_collection.rb +0 -44
- data/spec/unit/thinking_sphinx/active_record/search_spec.rb +0 -107
- data/spec/unit/thinking_sphinx/attribute_spec.rb +0 -212
- data/spec/unit/thinking_sphinx/collection_spec.rb +0 -14
- data/spec/unit/thinking_sphinx/index/builder_spec.rb +0 -5
- data/spec/unit/thinking_sphinx/search_spec.rb +0 -59
@@ -0,0 +1,176 @@
|
|
1
|
+
require 'spec/spec_helper'
|
2
|
+
|
3
|
+
describe ThinkingSphinx::FacetSearch do
|
4
|
+
describe 'populate' do
|
5
|
+
it "should make separate Sphinx queries for each facet" do
|
6
|
+
ThinkingSphinx.should_receive(:search).with(
|
7
|
+
hash_including(:group_by => 'city_facet')
|
8
|
+
).and_return([])
|
9
|
+
ThinkingSphinx.should_receive(:search).with(
|
10
|
+
hash_including(:group_by => 'state_facet')
|
11
|
+
).and_return([])
|
12
|
+
ThinkingSphinx.should_receive(:search).with(
|
13
|
+
hash_including(:group_by => 'birthday')
|
14
|
+
).and_return([])
|
15
|
+
|
16
|
+
ThinkingSphinx::FacetSearch.new(:classes => [Person])
|
17
|
+
end
|
18
|
+
|
19
|
+
it "should request all shared facets in a multi-model request by default" do
|
20
|
+
ThinkingSphinx.stub!(:search => [])
|
21
|
+
ThinkingSphinx::FacetSearch.new.facet_names.should == ['class_crc']
|
22
|
+
end
|
23
|
+
|
24
|
+
it "should request all facets in a multi-model request if specified" do
|
25
|
+
ThinkingSphinx.stub!(:search => [])
|
26
|
+
ThinkingSphinx::FacetSearch.new(
|
27
|
+
:all_facets => true
|
28
|
+
).facet_names.should == [
|
29
|
+
'class_crc', 'city_facet', 'state_facet', 'birthday'
|
30
|
+
]
|
31
|
+
end
|
32
|
+
|
33
|
+
describe ':facets option' do
|
34
|
+
it "should limit facets to the requested set" do
|
35
|
+
ThinkingSphinx.should_receive(:search).once.and_return([])
|
36
|
+
|
37
|
+
ThinkingSphinx::FacetSearch.new(
|
38
|
+
:classes => [Person], :facets => :state
|
39
|
+
)
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
describe "empty result set for attributes" do
|
44
|
+
before :each do
|
45
|
+
ThinkingSphinx.stub!(:search => [])
|
46
|
+
@facets = ThinkingSphinx::FacetSearch.new(
|
47
|
+
:classes => [Person], :facets => :state
|
48
|
+
)
|
49
|
+
end
|
50
|
+
|
51
|
+
it "should add key as attribute" do
|
52
|
+
@facets.should have_key(:state)
|
53
|
+
end
|
54
|
+
|
55
|
+
it "should return an empty hash for the facet results" do
|
56
|
+
@facets[:state].should be_empty
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
describe "non-empty result set" do
|
61
|
+
before :each do
|
62
|
+
@person = Person.find(:first)
|
63
|
+
@people = [@person]
|
64
|
+
@people.stub!(:each_with_groupby_and_count).
|
65
|
+
and_yield(@person, @person.city.to_crc32, 1)
|
66
|
+
ThinkingSphinx.stub!(:search => @people)
|
67
|
+
|
68
|
+
@facets = ThinkingSphinx::FacetSearch.new(
|
69
|
+
:classes => [Person], :facets => :city
|
70
|
+
)
|
71
|
+
end
|
72
|
+
|
73
|
+
it "should return a hash" do
|
74
|
+
@facets.should be_a_kind_of(Hash)
|
75
|
+
end
|
76
|
+
|
77
|
+
it "should add key as attribute" do
|
78
|
+
@facets.keys.should include(:city)
|
79
|
+
end
|
80
|
+
|
81
|
+
it "should return a hash" do
|
82
|
+
@facets[:city].should == {@person.city => 1}
|
83
|
+
end
|
84
|
+
end
|
85
|
+
|
86
|
+
before :each do
|
87
|
+
@config = ThinkingSphinx::Configuration.instance
|
88
|
+
@config.configuration.searchd.max_matches = 10_000
|
89
|
+
end
|
90
|
+
|
91
|
+
it "should use the system-set max_matches for limit on facet calls" do
|
92
|
+
ThinkingSphinx.should_receive(:search) do |options|
|
93
|
+
options[:max_matches].should == 10_000
|
94
|
+
options[:limit].should == 10_000
|
95
|
+
[]
|
96
|
+
end
|
97
|
+
|
98
|
+
ThinkingSphinx::FacetSearch.new
|
99
|
+
end
|
100
|
+
|
101
|
+
it "should use the default max-matches if there is no explicit setting" do
|
102
|
+
@config.configuration.searchd.max_matches = nil
|
103
|
+
ThinkingSphinx.should_receive(:search) do |options|
|
104
|
+
options[:max_matches].should == 1000
|
105
|
+
options[:limit].should == 1000
|
106
|
+
[]
|
107
|
+
end
|
108
|
+
|
109
|
+
ThinkingSphinx::FacetSearch.new
|
110
|
+
end
|
111
|
+
|
112
|
+
it "should ignore user-provided max_matches and limit on facet calls" do
|
113
|
+
ThinkingSphinx.should_receive(:search) do |options|
|
114
|
+
options[:max_matches].should == 10_000
|
115
|
+
options[:limit].should == 10_000
|
116
|
+
[]
|
117
|
+
end
|
118
|
+
|
119
|
+
ThinkingSphinx::FacetSearch.new(
|
120
|
+
:max_matches => 500,
|
121
|
+
:limit => 200
|
122
|
+
)
|
123
|
+
end
|
124
|
+
|
125
|
+
it "should not use an explicit :page" do
|
126
|
+
ThinkingSphinx.should_receive(:search) do |options|
|
127
|
+
options[:page].should == 1
|
128
|
+
[]
|
129
|
+
end
|
130
|
+
|
131
|
+
ThinkingSphinx::FacetSearch.new(:page => 3)
|
132
|
+
end
|
133
|
+
|
134
|
+
describe "conflicting facets" do
|
135
|
+
before :each do
|
136
|
+
@index = ThinkingSphinx::Index::Builder.generate(Alpha) do
|
137
|
+
indexes :name
|
138
|
+
has :value, :as => :city, :facet => true
|
139
|
+
end
|
140
|
+
end
|
141
|
+
|
142
|
+
after :each do
|
143
|
+
Alpha.sphinx_facets.delete_at(-1)
|
144
|
+
end
|
145
|
+
|
146
|
+
it "should raise an error if searching with facets of same name but different type" do
|
147
|
+
lambda {
|
148
|
+
facets = ThinkingSphinx.facets :all_facets => true
|
149
|
+
}.should raise_error
|
150
|
+
end
|
151
|
+
end
|
152
|
+
end
|
153
|
+
|
154
|
+
describe "#for" do
|
155
|
+
before do
|
156
|
+
@person = Person.find(:first)
|
157
|
+
@people = [@person]
|
158
|
+
@people.stub!(:each_with_groupby_and_count).
|
159
|
+
and_yield(@person, @person.city.to_crc32, 1)
|
160
|
+
ThinkingSphinx.stub!(:search => @people)
|
161
|
+
|
162
|
+
@facets = ThinkingSphinx::FacetSearch.new(
|
163
|
+
:classes => [Person], :facets => :city
|
164
|
+
)
|
165
|
+
end
|
166
|
+
|
167
|
+
it "should return the search results for the attribute and key pair" do
|
168
|
+
ThinkingSphinx.should_receive(:search) do |options|
|
169
|
+
options[:with].should have_key('city_facet')
|
170
|
+
options[:with]['city_facet'].should == @person.city.to_crc32
|
171
|
+
end
|
172
|
+
|
173
|
+
@facets.for(:city => @person.city)
|
174
|
+
end
|
175
|
+
end
|
176
|
+
end
|
@@ -0,0 +1,302 @@
|
|
1
|
+
require 'spec/spec_helper'
|
2
|
+
|
3
|
+
describe ThinkingSphinx::Facet do
|
4
|
+
describe ".name_for" do
|
5
|
+
it "should remove '_facet' from provided string and return a symbol" do
|
6
|
+
ThinkingSphinx::Facet.name_for('attribute_facet').should == :attribute
|
7
|
+
end
|
8
|
+
|
9
|
+
it "should remove '_facet' from provided symbol" do
|
10
|
+
ThinkingSphinx::Facet.name_for(:attribute_facet).should == :attribute
|
11
|
+
end
|
12
|
+
|
13
|
+
it "should return the name of the facet if a Facet is passed" do
|
14
|
+
facet = ThinkingSphinx::Facet.new(
|
15
|
+
ThinkingSphinx::Attribute.stub_instance(:unique_name => :attribute, :columns => ['attribute'])
|
16
|
+
)
|
17
|
+
ThinkingSphinx::Facet.name_for(facet).should == :attribute
|
18
|
+
end
|
19
|
+
|
20
|
+
it "should return 'class' for special case name 'class_crc'" do
|
21
|
+
ThinkingSphinx::Facet.name_for(:class_crc).should == :class
|
22
|
+
end
|
23
|
+
|
24
|
+
it "should cycle" do
|
25
|
+
ThinkingSphinx::Facet.name_for(ThinkingSphinx::Facet.attribute_name_for(:attribute)).should == :attribute
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
describe ".attribute_name_for" do
|
30
|
+
it "should append '_facet' to provided string" do
|
31
|
+
ThinkingSphinx::Facet.attribute_name_for('attribute').should == 'attribute_facet'
|
32
|
+
end
|
33
|
+
|
34
|
+
it "should append '_facet' to provided symbol and return a string" do
|
35
|
+
ThinkingSphinx::Facet.attribute_name_for(:attribute).should == 'attribute_facet'
|
36
|
+
end
|
37
|
+
|
38
|
+
it "should return 'class_crc' for special case attribute 'class'" do
|
39
|
+
ThinkingSphinx::Facet.attribute_name_for(:class).should == 'class_crc'
|
40
|
+
end
|
41
|
+
|
42
|
+
it "should cycle" do
|
43
|
+
ThinkingSphinx::Facet.attribute_name_for(ThinkingSphinx::Facet.name_for('attribute_facet')).should == 'attribute_facet'
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
describe ".attribute_name_from_value" do
|
48
|
+
it "should append _facet if the value is a string" do
|
49
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', 'string').
|
50
|
+
should == 'attribute_facet'
|
51
|
+
end
|
52
|
+
|
53
|
+
it "should not append _facet if the value isn't a string" do
|
54
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', 1).
|
55
|
+
should == 'attribute'
|
56
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', Time.now).
|
57
|
+
should == 'attribute'
|
58
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', true).
|
59
|
+
should == 'attribute'
|
60
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', 1.23).
|
61
|
+
should == 'attribute'
|
62
|
+
end
|
63
|
+
|
64
|
+
it "should append _facet is the value is an array of strings" do
|
65
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', ['a', 'b']).
|
66
|
+
should == 'attribute_facet'
|
67
|
+
end
|
68
|
+
|
69
|
+
it "should not append _facet if the value is an array of integers" do
|
70
|
+
ThinkingSphinx::Facet.attribute_name_from_value('attribute', [1, 2]).
|
71
|
+
should == 'attribute'
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
describe ".translate?" do
|
76
|
+
before :each do
|
77
|
+
@index = ThinkingSphinx::Index.new(Alpha)
|
78
|
+
@source = ThinkingSphinx::Source.new(@index)
|
79
|
+
@attribute = ThinkingSphinx::Attribute.new(
|
80
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
81
|
+
)
|
82
|
+
end
|
83
|
+
|
84
|
+
it "should return true if the property is a field" do
|
85
|
+
field = ThinkingSphinx::Field.new(
|
86
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
87
|
+
)
|
88
|
+
|
89
|
+
ThinkingSphinx::Facet.translate?(field).should be_true
|
90
|
+
end
|
91
|
+
|
92
|
+
it "should return true if the property is a string attribute" do
|
93
|
+
@attribute.stub_method(:type => :string)
|
94
|
+
|
95
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_true
|
96
|
+
end
|
97
|
+
|
98
|
+
it "should return false if the property is an integer attribute" do
|
99
|
+
@attribute.stub_method(:type => :integer)
|
100
|
+
|
101
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_false
|
102
|
+
end
|
103
|
+
|
104
|
+
it "should return false if the property is a boolean attribute" do
|
105
|
+
@attribute.stub_method(:type => :boolean)
|
106
|
+
|
107
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_false
|
108
|
+
end
|
109
|
+
|
110
|
+
it "should return false if the property is a timestamp attribute" do
|
111
|
+
@attribute.stub_method(:type => :datetime)
|
112
|
+
|
113
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_false
|
114
|
+
end
|
115
|
+
|
116
|
+
it "should return false if the property is a float attribute" do
|
117
|
+
@attribute.stub_method(:type => :float)
|
118
|
+
|
119
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_false
|
120
|
+
end
|
121
|
+
|
122
|
+
it "should return false if the property is an MVA of integer values" do
|
123
|
+
@attribute.stub_method(:type => :multi, :all_ints? => true)
|
124
|
+
|
125
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_false
|
126
|
+
end
|
127
|
+
|
128
|
+
it "should return true if the property is an MVA of string values" do
|
129
|
+
@attribute.stub_method(:type => :multi, :all_ints? => false)
|
130
|
+
|
131
|
+
ThinkingSphinx::Facet.translate?(@attribute).should be_true
|
132
|
+
end
|
133
|
+
end
|
134
|
+
|
135
|
+
describe "#translate?" do
|
136
|
+
before :each do
|
137
|
+
@index = ThinkingSphinx::Index.new(Alpha)
|
138
|
+
@source = ThinkingSphinx::Source.new(@index)
|
139
|
+
@attribute = ThinkingSphinx::Attribute.new(
|
140
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
141
|
+
)
|
142
|
+
end
|
143
|
+
|
144
|
+
it "should return true if the property is a field" do
|
145
|
+
field = ThinkingSphinx::Field.new(
|
146
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
147
|
+
)
|
148
|
+
|
149
|
+
ThinkingSphinx::Facet.new(field).translate?.should be_true
|
150
|
+
end
|
151
|
+
|
152
|
+
it "should return true if the property is a string attribute" do
|
153
|
+
@attribute.stub_method(:type => :string)
|
154
|
+
|
155
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_true
|
156
|
+
end
|
157
|
+
|
158
|
+
it "should return false if the property is an integer attribute" do
|
159
|
+
@attribute.stub_method(:type => :integer)
|
160
|
+
|
161
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_false
|
162
|
+
end
|
163
|
+
|
164
|
+
it "should return false if the property is a boolean attribute" do
|
165
|
+
@attribute.stub_method(:type => :boolean)
|
166
|
+
|
167
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_false
|
168
|
+
end
|
169
|
+
|
170
|
+
it "should return false if the property is a timestamp attribute" do
|
171
|
+
@attribute.stub_method(:type => :datetime)
|
172
|
+
|
173
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_false
|
174
|
+
end
|
175
|
+
|
176
|
+
it "should return false if the property is a float attribute" do
|
177
|
+
@attribute.stub_method(:type => :float)
|
178
|
+
|
179
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_false
|
180
|
+
end
|
181
|
+
|
182
|
+
it "should return false if the property is an MVA of integer values" do
|
183
|
+
@attribute.stub_method(:type => :multi, :all_ints? => true)
|
184
|
+
|
185
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_false
|
186
|
+
end
|
187
|
+
|
188
|
+
it "should return true if the property is an MVA of string values" do
|
189
|
+
@attribute.stub_method(:type => :multi, :all_ints? => false)
|
190
|
+
|
191
|
+
ThinkingSphinx::Facet.new(@attribute).translate?.should be_true
|
192
|
+
end
|
193
|
+
end
|
194
|
+
|
195
|
+
describe "#attribute_name" do
|
196
|
+
before :each do
|
197
|
+
@index = ThinkingSphinx::Index.new(Alpha)
|
198
|
+
@source = ThinkingSphinx::Source.new(@index)
|
199
|
+
@attribute = ThinkingSphinx::Attribute.new(
|
200
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
201
|
+
)
|
202
|
+
end
|
203
|
+
|
204
|
+
it "should return the attribute name if built off an integer attribute" do
|
205
|
+
@attribute.stub_method(:type => :integer)
|
206
|
+
|
207
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name"
|
208
|
+
end
|
209
|
+
|
210
|
+
it "should return the attribute name if built off a boolean attribute" do
|
211
|
+
@attribute.stub_method(:type => :boolean)
|
212
|
+
|
213
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name"
|
214
|
+
end
|
215
|
+
|
216
|
+
it "should return the attribute name if built off a float attribute" do
|
217
|
+
@attribute.stub_method(:type => :float)
|
218
|
+
|
219
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name"
|
220
|
+
end
|
221
|
+
|
222
|
+
it "should return the attribute name if built off a timestamp attribute" do
|
223
|
+
@attribute.stub_method(:type => :datetime)
|
224
|
+
|
225
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name"
|
226
|
+
end
|
227
|
+
|
228
|
+
it "should return the attribute name with _facet suffix if built off a string attribute" do
|
229
|
+
@attribute.stub_method(:type => :string)
|
230
|
+
|
231
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name_facet"
|
232
|
+
end
|
233
|
+
|
234
|
+
it "should return the attribute name with _facet suffix if built off a field" do
|
235
|
+
field = ThinkingSphinx::Field.new(
|
236
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
237
|
+
)
|
238
|
+
|
239
|
+
ThinkingSphinx::Facet.new(field).attribute_name.should == "name_facet"
|
240
|
+
end
|
241
|
+
|
242
|
+
it "should return the attribute name if build off an integer MVA" do
|
243
|
+
@attribute.stub_method(:type => :multi, :all_ints? => true)
|
244
|
+
|
245
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name"
|
246
|
+
end
|
247
|
+
|
248
|
+
it "should return the attribute name with the _facet suffix if build off an non-integer MVA" do
|
249
|
+
@attribute.stub_method(:type => :multi, :all_ints? => false)
|
250
|
+
|
251
|
+
ThinkingSphinx::Facet.new(@attribute).attribute_name.should == "name_facet"
|
252
|
+
end
|
253
|
+
end
|
254
|
+
|
255
|
+
describe "#type" do
|
256
|
+
before :each do
|
257
|
+
@index = ThinkingSphinx::Index.new(Alpha)
|
258
|
+
@source = ThinkingSphinx::Source.new(@index)
|
259
|
+
end
|
260
|
+
|
261
|
+
it "should return :string if the property is a field" do
|
262
|
+
field = ThinkingSphinx::Field.new(
|
263
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
264
|
+
)
|
265
|
+
|
266
|
+
ThinkingSphinx::Facet.new(field).type.should == :string
|
267
|
+
end
|
268
|
+
|
269
|
+
it "should return the attribute type if the property is an attribute" do
|
270
|
+
attribute = ThinkingSphinx::Attribute.new(
|
271
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:name)
|
272
|
+
)
|
273
|
+
attribute.stub_method(:type => :anything)
|
274
|
+
|
275
|
+
ThinkingSphinx::Facet.new(attribute).type.should == :anything
|
276
|
+
end
|
277
|
+
end
|
278
|
+
|
279
|
+
describe "#value" do
|
280
|
+
before :each do
|
281
|
+
@index = ThinkingSphinx::Index.new(Friendship)
|
282
|
+
@source = ThinkingSphinx::Source.new(@index)
|
283
|
+
@field = ThinkingSphinx::Field.new(
|
284
|
+
@source, ThinkingSphinx::Index::FauxColumn.new(:person, :first_name)
|
285
|
+
)
|
286
|
+
@facet = ThinkingSphinx::Facet.new(@field)
|
287
|
+
end
|
288
|
+
|
289
|
+
it "should return association values" do
|
290
|
+
person = Person.find(:first)
|
291
|
+
friendship = Friendship.new(:person => person)
|
292
|
+
|
293
|
+
@facet.value(friendship, 1).should == person.first_name
|
294
|
+
end
|
295
|
+
|
296
|
+
it "should return nil if the association is nil" do
|
297
|
+
friendship = Friendship.new(:person => nil)
|
298
|
+
|
299
|
+
@facet.value(friendship, 1).should be_nil
|
300
|
+
end
|
301
|
+
end
|
302
|
+
end
|
@@ -1,23 +1,28 @@
|
|
1
1
|
require 'spec/spec_helper'
|
2
2
|
|
3
3
|
describe ThinkingSphinx::Field do
|
4
|
+
before :each do
|
5
|
+
@index = ThinkingSphinx::Index.new(Alpha)
|
6
|
+
@source = ThinkingSphinx::Source.new(@index)
|
7
|
+
end
|
8
|
+
|
4
9
|
describe '#initialize' do
|
5
10
|
it 'raises if no columns are provided so that configuration errors are easier to track down' do
|
6
11
|
lambda {
|
7
|
-
ThinkingSphinx::Field.new([])
|
12
|
+
ThinkingSphinx::Field.new(@source, [])
|
8
13
|
}.should raise_error(RuntimeError)
|
9
14
|
end
|
10
15
|
|
11
16
|
it 'raises if an element of the columns param is an integer - as happens when you use id instead of :id - so that configuration errors are easier to track down' do
|
12
17
|
lambda {
|
13
|
-
ThinkingSphinx::Field.new([1234])
|
18
|
+
ThinkingSphinx::Field.new(@source, [1234])
|
14
19
|
}.should raise_error(RuntimeError)
|
15
20
|
end
|
16
21
|
end
|
17
22
|
|
18
23
|
describe "unique_name method" do
|
19
24
|
before :each do
|
20
|
-
@field = ThinkingSphinx::Field.new [
|
25
|
+
@field = ThinkingSphinx::Field.new @source, [
|
21
26
|
Object.stub_instance(:__stack => [], :__name => "col_name")
|
22
27
|
]
|
23
28
|
end
|
@@ -42,13 +47,15 @@ describe ThinkingSphinx::Field do
|
|
42
47
|
|
43
48
|
describe "prefixes method" do
|
44
49
|
it "should default to false" do
|
45
|
-
@field = ThinkingSphinx::Field.new(
|
50
|
+
@field = ThinkingSphinx::Field.new(
|
51
|
+
@source, [Object.stub_instance(:__stack => [])]
|
52
|
+
)
|
46
53
|
@field.prefixes.should be_false
|
47
54
|
end
|
48
55
|
|
49
56
|
it "should be true if the corresponding option is set" do
|
50
57
|
@field = ThinkingSphinx::Field.new(
|
51
|
-
[Object.stub_instance(:__stack => [])], :prefixes => true
|
58
|
+
@source, [Object.stub_instance(:__stack => [])], :prefixes => true
|
52
59
|
)
|
53
60
|
@field.prefixes.should be_true
|
54
61
|
end
|
@@ -56,13 +63,15 @@ describe ThinkingSphinx::Field do
|
|
56
63
|
|
57
64
|
describe "infixes method" do
|
58
65
|
it "should default to false" do
|
59
|
-
@field = ThinkingSphinx::Field.new(
|
66
|
+
@field = ThinkingSphinx::Field.new(
|
67
|
+
@source, [Object.stub_instance(:__stack => [])]
|
68
|
+
)
|
60
69
|
@field.infixes.should be_false
|
61
70
|
end
|
62
71
|
|
63
72
|
it "should be true if the corresponding option is set" do
|
64
73
|
@field = ThinkingSphinx::Field.new(
|
65
|
-
[Object.stub_instance(:__stack => [])], :infixes => true
|
74
|
+
@source, [Object.stub_instance(:__stack => [])], :infixes => true
|
66
75
|
)
|
67
76
|
@field.infixes.should be_true
|
68
77
|
end
|
@@ -70,21 +79,21 @@ describe ThinkingSphinx::Field do
|
|
70
79
|
|
71
80
|
describe "column_with_prefix method" do
|
72
81
|
before :each do
|
73
|
-
@field = ThinkingSphinx::Field.new [
|
82
|
+
@field = ThinkingSphinx::Field.new @source, [
|
74
83
|
ThinkingSphinx::Index::FauxColumn.new(:col_name)
|
75
84
|
]
|
76
85
|
@field.columns.each { |col| @field.associations[col] = [] }
|
77
86
|
@field.model = Person
|
78
87
|
|
79
|
-
@first_join = Object.
|
80
|
-
@
|
88
|
+
@first_join = Object.new
|
89
|
+
@first_join.stub!(:aliased_table_name => "tabular")
|
90
|
+
@second_join = Object.new
|
91
|
+
@second_join.stub!(:aliased_table_name => "data")
|
81
92
|
|
82
|
-
@first_assoc = ThinkingSphinx::Association.
|
83
|
-
|
84
|
-
|
85
|
-
@second_assoc
|
86
|
-
:join => @second_join, :has_column? => true
|
87
|
-
)
|
93
|
+
@first_assoc = ThinkingSphinx::Association.new nil, nil
|
94
|
+
@first_assoc.stub!(:join => @first_join, :has_column? => true)
|
95
|
+
@second_assoc = ThinkingSphinx::Association.new nil, nil
|
96
|
+
@second_assoc.stub!(:join => @second_join, :has_column? => true)
|
88
97
|
end
|
89
98
|
|
90
99
|
it "should return the column name if the column is a string" do
|
@@ -116,7 +125,7 @@ describe ThinkingSphinx::Field do
|
|
116
125
|
@assoc_c = Object.stub_instance(:is_many? => true)
|
117
126
|
|
118
127
|
@field = ThinkingSphinx::Field.new(
|
119
|
-
[ThinkingSphinx::Index::FauxColumn.new(:col_name)]
|
128
|
+
@source, [ThinkingSphinx::Index::FauxColumn.new(:col_name)]
|
120
129
|
)
|
121
130
|
@field.associations = {
|
122
131
|
:a => @assoc_a, :b => @assoc_b, :c => @assoc_c
|