2019-07-31 22:56:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
describe SearchService do
|
2014-09-02 18:07:02 +05:30
|
|
|
let(:user) { create(:user) }
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
let(:accessible_group) { create(:group, :private) }
|
|
|
|
let(:inaccessible_group) { create(:group, :private) }
|
|
|
|
let!(:group_member) { create(:group_member, group: accessible_group, user: user) }
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
let!(:accessible_project) { create(:project, :private, name: 'accessible_project') }
|
2017-08-17 22:00:37 +05:30
|
|
|
let(:note) { create(:note_on_issue, project: accessible_project) }
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
let!(:inaccessible_project) { create(:project, :private, name: 'inaccessible_project') }
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
let(:snippet) { create(:snippet, author: user) }
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:group_project) { create(:project, group: accessible_group, name: 'group_project') }
|
|
|
|
let(:public_project) { create(:project, :public, name: 'public_project') }
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
subject(:search_service) { described_class.new(user, search: search, scope: scope, page: 1) }
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
before do
|
2018-11-18 11:00:15 +05:30
|
|
|
accessible_project.add_maintainer(user)
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '#project' do
|
|
|
|
context 'when the project is accessible' do
|
|
|
|
it 'returns the project' do
|
2017-09-10 17:25:29 +05:30
|
|
|
project = described_class.new(user, project_id: accessible_project.id).project
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(project).to eq accessible_project
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
|
|
|
|
it 'returns the project for guests' do
|
|
|
|
search_project = create :project
|
|
|
|
search_project.add_guest(user)
|
|
|
|
|
|
|
|
project = described_class.new(user, project_id: search_project.id).project
|
|
|
|
|
|
|
|
expect(project).to eq search_project
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the project is not accessible' do
|
|
|
|
it 'returns nil' do
|
2017-09-10 17:25:29 +05:30
|
|
|
project = described_class.new(user, project_id: inaccessible_project.id).project
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(project).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when there is no project_id' do
|
|
|
|
it 'returns nil' do
|
2017-09-10 17:25:29 +05:30
|
|
|
project = described_class.new(user).project
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(project).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#group' do
|
|
|
|
context 'when the group is accessible' do
|
|
|
|
it 'returns the group' do
|
2017-09-10 17:25:29 +05:30
|
|
|
group = described_class.new(user, group_id: accessible_group.id).group
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(group).to eq accessible_group
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the group is not accessible' do
|
|
|
|
it 'returns nil' do
|
2017-09-10 17:25:29 +05:30
|
|
|
group = described_class.new(user, group_id: inaccessible_group.id).group
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(group).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when there is no group_id' do
|
|
|
|
it 'returns nil' do
|
2017-09-10 17:25:29 +05:30
|
|
|
group = described_class.new(user).group
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(group).to be_nil
|
|
|
|
end
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
describe '#show_snippets?' do
|
|
|
|
context 'when :snippets is \'true\'' do
|
|
|
|
it 'returns true' do
|
2017-09-10 17:25:29 +05:30
|
|
|
show_snippets = described_class.new(user, snippets: 'true').show_snippets?
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(show_snippets).to be_truthy
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
context 'when :snippets is not \'true\'' do
|
|
|
|
it 'returns false' do
|
2017-09-10 17:25:29 +05:30
|
|
|
show_snippets = described_class.new(user, snippets: 'tru').show_snippets?
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(show_snippets).to be_falsey
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'when :snippets is missing' do
|
|
|
|
it 'returns false' do
|
2017-09-10 17:25:29 +05:30
|
|
|
show_snippets = described_class.new(user).show_snippets?
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(show_snippets).to be_falsey
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#scope' do
|
|
|
|
context 'with accessible project_id' do
|
|
|
|
context 'and allowed scope' do
|
|
|
|
it 'returns the specified scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, project_id: accessible_project.id, scope: 'notes').scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'notes'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and disallowed scope' do
|
|
|
|
it 'returns the default scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, project_id: accessible_project.id, scope: 'projects').scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'blobs'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and no scope' do
|
|
|
|
it 'returns the default scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, project_id: accessible_project.id).scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'blobs'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with \'true\' snippets' do
|
|
|
|
context 'and allowed scope' do
|
|
|
|
it 'returns the specified scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, snippets: 'true', scope: 'snippet_titles').scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'snippet_titles'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and disallowed scope' do
|
|
|
|
it 'returns the default scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, snippets: 'true', scope: 'projects').scope
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(scope).to eq 'snippet_blobs'
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
context 'and no scope' do
|
|
|
|
it 'returns the default scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, snippets: 'true').scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'snippet_blobs'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with no project_id, no snippets' do
|
|
|
|
context 'and allowed scope' do
|
|
|
|
it 'returns the specified scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, scope: 'issues').scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'issues'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and disallowed scope' do
|
|
|
|
it 'returns the default scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user, scope: 'blobs').scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'projects'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and no scope' do
|
|
|
|
it 'returns the default scope' do
|
2017-09-10 17:25:29 +05:30
|
|
|
scope = described_class.new(user).scope
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(scope).to eq 'projects'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#search_results' do
|
|
|
|
context 'with accessible project_id' do
|
|
|
|
it 'returns an instance of Gitlab::ProjectSearchResults' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_results = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
project_id: accessible_project.id,
|
|
|
|
scope: 'notes',
|
|
|
|
search: note.note).search_results
|
|
|
|
|
|
|
|
expect(search_results).to be_a Gitlab::ProjectSearchResults
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with accessible project_id and \'true\' snippets' do
|
|
|
|
it 'returns an instance of Gitlab::ProjectSearchResults' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_results = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
project_id: accessible_project.id,
|
|
|
|
snippets: 'true',
|
|
|
|
scope: 'notes',
|
|
|
|
search: note.note).search_results
|
|
|
|
|
|
|
|
expect(search_results).to be_a Gitlab::ProjectSearchResults
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with \'true\' snippets' do
|
|
|
|
it 'returns an instance of Gitlab::SnippetSearchResults' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_results = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
snippets: 'true',
|
|
|
|
search: snippet.content).search_results
|
|
|
|
|
|
|
|
expect(search_results).to be_a Gitlab::SnippetSearchResults
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with no project_id and no snippets' do
|
|
|
|
it 'returns an instance of Gitlab::SearchResults' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_results = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
search: public_project.name).search_results
|
|
|
|
|
|
|
|
expect(search_results).to be_a Gitlab::SearchResults
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#search_objects' do
|
|
|
|
context 'with accessible project_id' do
|
|
|
|
it 'returns objects in the project' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_objects = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
project_id: accessible_project.id,
|
|
|
|
scope: 'notes',
|
|
|
|
search: note.note).search_objects
|
|
|
|
|
|
|
|
expect(search_objects.first).to eq note
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with accessible project_id and \'true\' snippets' do
|
|
|
|
it 'returns objects in the project' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_objects = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
project_id: accessible_project.id,
|
|
|
|
snippets: 'true',
|
|
|
|
scope: 'notes',
|
|
|
|
search: note.note).search_objects
|
|
|
|
|
|
|
|
expect(search_objects.first).to eq note
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with \'true\' snippets' do
|
|
|
|
it 'returns objects in snippets' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_objects = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
snippets: 'true',
|
|
|
|
search: snippet.content).search_objects
|
|
|
|
|
|
|
|
expect(search_objects.first).to eq snippet
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with accessible group_id' do
|
|
|
|
it 'returns objects in the group' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_objects = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
group_id: accessible_group.id,
|
|
|
|
search: group_project.name).search_objects
|
|
|
|
|
|
|
|
expect(search_objects.first).to eq group_project
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with no project_id, group_id or snippets' do
|
|
|
|
it 'returns objects in global' do
|
2017-09-10 17:25:29 +05:30
|
|
|
search_objects = described_class.new(
|
2017-08-17 22:00:37 +05:30
|
|
|
user,
|
|
|
|
search: public_project.name).search_objects
|
|
|
|
|
|
|
|
expect(search_objects.first).to eq public_project
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|
2020-04-08 14:13:33 +05:30
|
|
|
|
|
|
|
context 'redacting search results' do
|
|
|
|
shared_examples 'it redacts incorrect results' do
|
|
|
|
before do
|
|
|
|
allow(Ability).to receive(:allowed?).and_return(allowed)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when allowed' do
|
|
|
|
let(:allowed) { true }
|
|
|
|
|
|
|
|
it 'does nothing' do
|
|
|
|
expect(results).not_to be_empty
|
|
|
|
expect(results).to all(be_an(model_class))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when disallowed' do
|
|
|
|
let(:allowed) { false }
|
|
|
|
|
|
|
|
it 'does nothing' do
|
|
|
|
expect(results).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'issues' do
|
|
|
|
let(:issue) { create(:issue, project: accessible_project) }
|
|
|
|
let(:scope) { 'issues' }
|
|
|
|
let(:model_class) { Issue }
|
|
|
|
let(:ability) { :read_issue }
|
|
|
|
let(:search) { issue.title }
|
|
|
|
let(:results) { subject.search_objects }
|
|
|
|
|
|
|
|
it_behaves_like 'it redacts incorrect results'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'notes' do
|
|
|
|
let(:note) { create(:note_on_commit, project: accessible_project) }
|
|
|
|
let(:scope) { 'notes' }
|
|
|
|
let(:model_class) { Note }
|
|
|
|
let(:ability) { :read_note }
|
|
|
|
let(:search) { note.note }
|
|
|
|
let(:results) do
|
|
|
|
described_class.new(
|
|
|
|
user,
|
|
|
|
project_id: accessible_project.id,
|
|
|
|
scope: scope,
|
|
|
|
search: note.note
|
|
|
|
).search_objects
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'it redacts incorrect results'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'merge_requests' do
|
|
|
|
let(:scope) { 'merge_requests' }
|
|
|
|
let(:model_class) { MergeRequest }
|
|
|
|
let(:ability) { :read_merge_request }
|
|
|
|
let(:merge_request) { create(:merge_request, source_project: accessible_project, author: user) }
|
|
|
|
let(:search) { merge_request.title }
|
|
|
|
let(:results) { subject.search_objects }
|
|
|
|
|
|
|
|
it_behaves_like 'it redacts incorrect results'
|
|
|
|
end
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|