Commit 768d21654a135130a28278c46cf413dbe65f8c81

Authored by Dmitriy Zaporozhets
1 parent fc60c391

Better search login for global context. Improved tests

Signed-off-by: Dmitriy Zaporozhets <dmitriy.zaporozhets@gmail.com>
app/contexts/search/global_context.rb
@@ -11,22 +11,17 @@ module Search @@ -11,22 +11,17 @@ module Search
11 query = Shellwords.shellescape(query) if query.present? 11 query = Shellwords.shellescape(query) if query.present?
12 return result unless query.present? 12 return result unless query.present?
13 13
14 -  
15 - projects = current_user.authorized_projects  
16 -  
17 - if params[:group_id].present?  
18 - group = Group.find_by_id(params[:group_id])  
19 - projects = projects.where(namespace_id: group.id) if group  
20 - end  
21 - 14 + authorized_projects_ids = []
  15 + authorized_projects_ids += current_user.authorized_projects.pluck(:id) if current_user
  16 + authorized_projects_ids += Project.public_or_internal_only(current_user).pluck(:id)
  17 +
  18 + group = Group.find_by_id(params[:group_id]) if params[:group_id].present?
  19 + projects = Project.where(id: authorized_projects_ids)
  20 + projects = projects.where(namespace_id: group.id) if group
  21 + projects = projects.search(query)
22 project_ids = projects.pluck(:id) 22 project_ids = projects.pluck(:id)
23 23
24 - visibility_levels = if current_user  
25 - [Gitlab::VisibilityLevel::INTERNAL, Gitlab::VisibilityLevel::PUBLIC]  
26 - else  
27 - [Gitlab::VisibilityLevel::PUBLIC]  
28 - end  
29 - result[:projects] = Project.where("projects.id in (?) OR projects.visibility_level in (?)", project_ids, visibility_levels).search(query).limit(20) 24 + result[:projects] = projects.limit(20)
30 result[:merge_requests] = MergeRequest.in_projects(project_ids).search(query).order('updated_at DESC').limit(20) 25 result[:merge_requests] = MergeRequest.in_projects(project_ids).search(query).order('updated_at DESC').limit(20)
31 result[:issues] = Issue.where(project_id: project_ids).search(query).order('updated_at DESC').limit(20) 26 result[:issues] = Issue.where(project_id: project_ids).search(query).order('updated_at DESC').limit(20)
32 result[:total_results] = %w(projects issues merge_requests).sum { |items| result[items.to_sym].size } 27 result[:total_results] = %w(projects issues merge_requests).sum { |items| result[items.to_sym].size }
spec/contexts/search_context_spec.rb
1 require 'spec_helper' 1 require 'spec_helper'
2 2
3 -describe SearchContext do 3 +describe 'Search::GlobalContext' do
4 let(:found_namespace) { create(:namespace, name: 'searchable namespace', path:'another_thing') } 4 let(:found_namespace) { create(:namespace, name: 'searchable namespace', path:'another_thing') }
5 let(:user) { create(:user, namespace: found_namespace) } 5 let(:user) { create(:user, namespace: found_namespace) }
6 let!(:found_project) { create(:project, name: 'searchable_project', creator_id: user.id, namespace: found_namespace, visibility_level: Gitlab::VisibilityLevel::PRIVATE) } 6 let!(:found_project) { create(:project, name: 'searchable_project', creator_id: user.id, namespace: found_namespace, visibility_level: Gitlab::VisibilityLevel::PRIVATE) }
7 7
8 let(:unfound_namespace) { create(:namespace, name: 'unfound namespace', path: 'yet_something_else') } 8 let(:unfound_namespace) { create(:namespace, name: 'unfound namespace', path: 'yet_something_else') }
9 let!(:unfound_project) { create(:project, name: 'unfound_project', creator_id: user.id, namespace: unfound_namespace, visibility_level: Gitlab::VisibilityLevel::PRIVATE) } 9 let!(:unfound_project) { create(:project, name: 'unfound_project', creator_id: user.id, namespace: unfound_namespace, visibility_level: Gitlab::VisibilityLevel::PRIVATE) }
10 - 10 +
11 let(:internal_namespace) { create(:namespace, path: 'something_internal',name: 'searchable internal namespace') } 11 let(:internal_namespace) { create(:namespace, path: 'something_internal',name: 'searchable internal namespace') }
12 let(:internal_user) { create(:user, namespace: internal_namespace) } 12 let(:internal_user) { create(:user, namespace: internal_namespace) }
13 let!(:internal_project) { create(:project, name: 'searchable_internal_project', creator_id: internal_user.id, namespace: internal_namespace, visibility_level: Gitlab::VisibilityLevel::INTERNAL) } 13 let!(:internal_project) { create(:project, name: 'searchable_internal_project', creator_id: internal_user.id, namespace: internal_namespace, visibility_level: Gitlab::VisibilityLevel::INTERNAL) }
14 - 14 +
15 let(:public_namespace) { create(:namespace, path: 'something_public',name: 'searchable public namespace') } 15 let(:public_namespace) { create(:namespace, path: 'something_public',name: 'searchable public namespace') }
16 let(:public_user) { create(:user, namespace: public_namespace) } 16 let(:public_user) { create(:user, namespace: public_namespace) }
17 let!(:public_project) { create(:project, name: 'searchable_public_project', creator_id: public_user.id, namespace: public_namespace, visibility_level: Gitlab::VisibilityLevel::PUBLIC) } 17 let!(:public_project) { create(:project, name: 'searchable_public_project', creator_id: public_user.id, namespace: public_namespace, visibility_level: Gitlab::VisibilityLevel::PUBLIC) }
18 18
19 describe '#execute' do 19 describe '#execute' do
20 - it 'public projects should be searchable' do  
21 - context = SearchContext.new([found_project.id], nil, {search_code: false, search: "searchable"})  
22 - results = context.execute  
23 - results[:projects].should == [found_project, public_project] 20 + context 'unauthenticated' do
  21 + it 'should return public projects only' do
  22 + context = Search::GlobalContext.new(nil, search: "searchable")
  23 + results = context.execute
  24 + results[:projects].should have(1).items
  25 + results[:projects].should include(public_project)
  26 + end
24 end 27 end
25 28
26 - it 'internal projects should be searchable' do  
27 - context = SearchContext.new([found_project.id], user, {search_code: false, search: "searchable"})  
28 - results = context.execute  
29 - # can't seem to rely on the return order, so check this way  
30 - #subject { results[:projects] }  
31 - results[:projects].should have(3).items  
32 - results[:projects].should include(found_project)  
33 - results[:projects].should include(internal_project)  
34 - results[:projects].should include(public_project)  
35 - end 29 + context 'authenticated' do
  30 + it 'should return public, internal and private projects' do
  31 + context = Search::GlobalContext.new(user, search: "searchable")
  32 + results = context.execute
  33 + results[:projects].should have(3).items
  34 + results[:projects].should include(public_project)
  35 + results[:projects].should include(found_project)
  36 + results[:projects].should include(internal_project)
  37 + end
  38 +
  39 + it 'should return only public & internal projects' do
  40 + context = Search::GlobalContext.new(internal_user, search: "searchable")
  41 + results = context.execute
  42 + results[:projects].should have(2).items
  43 + results[:projects].should include(internal_project)
  44 + results[:projects].should include(public_project)
  45 + end
36 46
37 - it 'namespace name should be searchable' do  
38 - context = SearchContext.new([found_project.id], user, {search_code: false, search: "searchable namespace"})  
39 - results = context.execute  
40 - results[:projects].should == [found_project] 47 + it 'namespace name should be searchable' do
  48 + context = Search::GlobalContext.new(user, search: "searchable namespace")
  49 + results = context.execute
  50 + results[:projects].should == [found_project]
  51 + end
41 end 52 end
42 end 53 end
43 end 54 end