require "spec_helper" module Pose describe Search do let(:subject) { Search.new [PosableOne, [PosableTwo]], 'query string' } let(:arel) { double() } let(:arel_2) { double() } let(:arel_3) { double() } describe :add_join do it 'returns the new arel' do arel.should_receive(:joins).with('foo').and_return(arel_2) expect(subject.add_join arel, 'foo').to eql arel_2 end context 'given a class' do it 'adds a join from PoseAssignment to the given class' do arel.should_receive(:joins) .with "INNER JOIN posable_ones ON pose_assignments.posable_id=posable_ones.id AND pose_assignments.posable_type='PosableOne'" subject.add_join arel, PosableOne end end context 'given a string' do it 'applies the join as given' do arel.should_receive(:joins).with "join string" subject.add_join arel, 'join string' end end context 'given a symbol' do it 'applies the join as given' do arel.should_receive(:joins).with :association subject.add_join arel, :association end end end describe :add_joins do it 'adds all joins to the given arel' do arel.should_receive(:joins).with('one').and_return(arel_2) arel_2.should_receive(:joins).with('two').and_return(arel_3) search = Search.new nil, nil, joins: ['one', 'two'] search.add_joins arel end it 'returns the given arel' do arel.should_receive(:joins).and_return(arel_2) arel_2.should_receive(:joins).and_return(arel_3) search = Search.new nil, nil, joins: ['one', 'two'] expect(search.add_joins arel).to eql arel_3 end end describe :add_wheres do it 'adds all joins to the given arel' do arel.should_receive(:where).with(['one = ?', true]).and_return(arel_2) arel_2.should_receive(:where).with(['two = ?', false]).and_return(arel_3) search = Search.new nil, nil, where: [['one = ?', true], ['two = ?', false]] search.add_wheres arel end it 'returns the given arel' do arel.should_receive(:where).and_return(arel_2) arel_2.should_receive(:where).and_return(arel_3) search = Search.new nil, nil, where: [['one'], ['two']] expect(search.add_wheres arel).to eql arel_3 end end describe :empty_result do it 'returns a hash with classes and empty arrays for each class in the search query' do search = Search.new [PosableOne, PosableTwo], '' result = search.empty_result expect(result['PosableOne']).to eq [] expect(result['PosableTwo']).to eq [] expect(result).to_not have_key 'User' end end describe :limit_ids do before :each do @search = Search.new nil, nil, limit: 2 end context 'with empty id set' do it 'does nothing' do result = { PosableOne => [] } @search.limit_ids result expect(result[PosableOne]).to eql [] end end context 'with id set less than the given limit' do it 'does nothing' do result = { PosableOne => [1, 2] } @search.limit_ids result expect(result[PosableOne]).to eql [1, 2] end end context 'with id set longer than the given limit' do it 'truncates the id set' do result = { PosableOne => [1, 2, 3] } @search.limit_ids result expect(result[PosableOne]).to eql [1, 2] end end context 'without limit in query' do it 'does nothing' do @search = Search.new nil, nil result = { PosableOne => [1, 2, 3] } @search.limit_ids result expect(result[PosableOne]).to eql [1, 2, 3] end end end describe :load_classes do context 'when the user wants ids' do it 'does nothing' do search = Search.new nil, nil, result_type: :ids result = { PosableOne => [1, 2] } search.load_classes @result expect(result[PosableOne]).to eql [1, 2] end end context 'when the user wants classes' do it 'loads the classes' do object_1 = create :posable_one object_2 = create :posable_one result = { PosableOne => [object_1.id, object_2.id] } search = Search.new nil, nil search.load_classes result expect(result[PosableOne]).to eq [object_1, object_2] end end context 'when the user specifies a select clause' do it 'loads only the specified attributes' do object_1 = create :posable_one object_2 = create :posable_one result = { PosableOne => [object_1.id, object_2.id] } search = Search.new nil, nil, select: 'id' search.load_classes result expect(result[PosableOne][0].id).to eq object_1.id expect(result[PosableOne][0].attribute_present? :text).to be_falsy expect(result[PosableOne][0].attribute_present? :id).to be_truthy expect(result[PosableOne][1].id).to eq object_2.id expect(result[PosableOne][1].attribute_present? :text).to be_falsy expect(result[PosableOne][1].attribute_present? :id).to be_truthy end end end describe :merge_search_result_word_matches do context 'given a new class name' do before :each do @result = {} end it 'sets the given ids as the ids for this class name' do subject.merge_search_result_word_matches @result, 'class1', [1, 2] expect(@result).to eq({ 'class1' => [1, 2] }) end end context 'given a class name with already existing ids from another word' do before :each do @result = { 'class1' => [1, 2] } end it 'only keeps the ids that are included in both sets' do subject.merge_search_result_word_matches @result, 'class1', [1, 3] expect(@result).to eq({ 'class1' => [1] }) end end context 'with an existing empty result set from a previous query' do before :each do @result = { 'class1' => [] } end it 'returns an empty result set' do subject.merge_search_result_word_matches @result, 'class1', [1, 3] @result.should eq({ 'class1' => [] }) end end context 'with a new empty result set' do before :each do @result = { 'class1' => [1, 2] } end it 'returns an empty result set' do subject.merge_search_result_word_matches @result, 'class1', [] @result.should eq({ 'class1' => [] }) end end context 'with a completely different result set' do before :each do @result = { 'class1' => [1, 2] } end it 'returns an empty result set' do subject.merge_search_result_word_matches @result, 'class1', [3, 4] @result.should eq({ 'class1' => [] }) end end end describe :search do it 'finds all matching instances of all classes' do posable_one_1 = create :posable_one, text: 'foo bar' posable_one_2 = create :posable_one, text: 'foo bar' posable_one_3 = create :posable_one, text: 'foo' posable_two_1 = create :posable_two, text: 'foo bar' posable_two_2 = create :posable_two, text: 'bar' search = Search.new [PosableOne, PosableTwo], 'foo bar' result = search.search expect(result[PosableOne]).to eq [posable_one_1, posable_one_2] end it 'searches through all given classes' do posable_one = create :posable_one, text: 'foo' posable_two = create :posable_two, text: 'foo' search = Search.new [PosableOne, PosableTwo], 'foo' result = search.search expect(result[PosableOne]).to eq [posable_one] expect(result[PosableTwo]).to eq [posable_two] end it 'searches through all given search words' do posable_one_1 = create :posable_one, text: 'foo bar' posable_one_2 = create :posable_one, text: 'foo' posable_one_3 = create :posable_one, text: 'bar' search = Search.new PosableOne, 'foo bar' result = search.search expect(result[PosableOne]).to eq [posable_one_1] end it 'limits the number of search results to the given limit' do posable_one_1 = create :posable_one, text: 'foo' posable_one_2 = create :posable_one, text: 'foo' search = Search.new PosableOne, 'foo', limit: 1 result = search.search expect(result[PosableOne]).to have(1).items end describe 'result types' do it 'loads classes by default' do posable_one = create :posable_one, text: 'foo' search = Search.new PosableOne, 'foo' result = search.search expect(result[PosableOne]).to eq [posable_one] end context 'result_type: :ids parameter' do it 'returns only the ids of the parameters' do posable_one = create :posable_one, text: 'foo' search = Search.new PosableOne, 'foo', result_type: :ids result = search.search expect(result[PosableOne]).to eql [posable_one.id] end end end context 'given joins and wheres' do it 'limits the search by the given joins and wheres' do user_1 = create :user, name: 'user one' user_2 = create :user, name: 'user two' posable_one_1 = create :posable_one, user: user_1, text: 'foo' posable_one_2 = create :posable_one, user: user_2, text: 'foo' search = Search.new PosableOne, 'foo', joins: [ PosableOne, 'INNER JOIN users ON posable_ones.user_id=users.id' ], where: ['users.name=?', 'user one'] result = search.search expect(result[PosableOne]).to eq [posable_one_1] end end end describe :search_word do context 'search results' do it 'returns the ids of the matching instances for this class' do posable_one_1 = create :posable_one, text: 'foo' posable_one_2 = create :posable_one, text: 'foo' search = Search.new PosableOne, nil result = search.search_word 'foo' expect(result['PosableOne']).to include posable_one_1.id, posable_one_2.id end end context 'no search results' do it 'returns an empty array for the class' do search = Search.new PosableOne, nil result = search.search_word 'foo' expect(result['PosableOne']).to eql [] end end context 'multiple classes to search over' do it 'returns results for all classes' do posable_one_1 = create :posable_one, text: 'foo' posable_two_1 = create :posable_two, text: 'foo' search = Search.new [PosableOne, PosableTwo], nil result = search.search_word 'foo' expect(result['PosableOne']).to include posable_one_1.id expect(result['PosableTwo']).to include posable_two_1.id end end end describe :search_words do context 'search results' do it 'returns the ids of all instances that match all query words' do posable_one = create :posable_one, text: 'foo bar' search = Search.new PosableOne, 'foo bar' result = search.search_words expect(result['PosableOne']).to eq [posable_one.id] end end context 'instance matches only one query word' do it 'does not return this instance' do posable_one_1 = create :posable_one, text: 'foo' posable_one_2 = create :posable_one, text: 'bar' search = Search.new PosableOne, 'foo bar' result = search.search_words expect(result['PosableOne']).to eq [] end end context 'multiple classes to search over' do it 'returns all matching instances from all classes' do posable_one = create :posable_one, text: 'foo bar' posable_two = create :posable_two, text: 'foo bar' search = Search.new [PosableOne, PosableTwo], 'foo bar' result = search.search_words expect(result['PosableOne']).to eq [posable_one.id] expect(result['PosableTwo']).to eq [posable_two.id] end end end end end