2019-12-21 20:55:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
describe TodosFinder do
|
|
|
|
describe '#execute' do
|
2018-03-27 19:54:05 +05:30
|
|
|
let(:user) { create(:user) }
|
|
|
|
let(:group) { create(:group) }
|
|
|
|
let(:project) { create(:project, namespace: group) }
|
2018-11-18 11:00:15 +05:30
|
|
|
let(:issue) { create(:issue, project: project) }
|
|
|
|
let(:merge_request) { create(:merge_request, source_project: project) }
|
2018-03-27 19:54:05 +05:30
|
|
|
let(:finder) { described_class }
|
2016-09-13 17:45:13 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
before do
|
2018-03-27 19:54:05 +05:30
|
|
|
group.add_developer(user)
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
2016-09-13 17:45:13 +05:30
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
describe '#execute' do
|
2019-12-21 20:55:43 +05:30
|
|
|
it 'returns no todos if user is nil' do
|
|
|
|
expect(described_class.new(nil, {}).execute).to be_empty
|
|
|
|
end
|
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
context 'filtering' do
|
|
|
|
let!(:todo1) { create(:todo, user: user, project: project, target: issue) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, group: group, target: merge_request) }
|
|
|
|
|
|
|
|
it 'returns correct todos when filtered by a project' do
|
|
|
|
todos = finder.new(user, { project_id: project.id }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns correct todos when filtered by a group' do
|
|
|
|
todos = finder.new(user, { group_id: group.id }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1, todo2])
|
|
|
|
end
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
context 'when filtering by type' do
|
|
|
|
it 'returns correct todos when filtered by a type' do
|
|
|
|
todos = finder.new(user, { type: 'Issue' }).execute
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
expect(todos).to match_array([todo1])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the correct todos when filtering for multiple types' do
|
|
|
|
todos = finder.new(user, { type: %w[Issue MergeRequest] }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1, todo2])
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
context 'when filtering for actions' do
|
|
|
|
let!(:todo1) { create(:todo, user: user, project: project, target: issue, action: Todo::ASSIGNED) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, group: group, target: merge_request, action: Todo::DIRECTLY_ADDRESSED) }
|
|
|
|
|
|
|
|
context 'by action ids' do
|
|
|
|
it 'returns the expected todos' do
|
|
|
|
todos = finder.new(user, { action_id: Todo::DIRECTLY_ADDRESSED }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo2])
|
|
|
|
end
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
it 'returns the expected todos when filtering for multiple action ids' do
|
|
|
|
todos = finder.new(user, { action_id: [Todo::DIRECTLY_ADDRESSED, Todo::ASSIGNED] }).execute
|
2019-12-21 20:55:43 +05:30
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
expect(todos).to match_array([todo2, todo1])
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'by action names' do
|
|
|
|
it 'returns the expected todos' do
|
|
|
|
todos = finder.new(user, { action: :directly_addressed }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo2])
|
|
|
|
end
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
it 'returns the expected todos when filtering for multiple action names' do
|
|
|
|
todos = finder.new(user, { action: [:directly_addressed, :assigned] }).execute
|
2019-12-21 20:55:43 +05:30
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
expect(todos).to match_array([todo2, todo1])
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when filtering by author' do
|
|
|
|
let(:author1) { create(:user) }
|
|
|
|
let(:author2) { create(:user) }
|
|
|
|
|
|
|
|
let!(:todo1) { create(:todo, user: user, author: author1) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, author: author2) }
|
|
|
|
|
|
|
|
it 'returns correct todos when filtering by an author' do
|
|
|
|
todos = finder.new(user, { author_id: author1.id }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1])
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'querying for multiple authors' do
|
|
|
|
it 'returns the correct todo items' do
|
|
|
|
todos = finder.new(user, { author_id: [author2.id, author1.id] }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo2, todo1])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'by groups' do
|
|
|
|
context 'with subgroups' do
|
|
|
|
let(:subgroup) { create(:group, parent: group) }
|
|
|
|
let!(:todo3) { create(:todo, user: user, group: subgroup, target: issue) }
|
|
|
|
|
|
|
|
it 'returns todos from subgroups when filtered by a group' do
|
|
|
|
todos = finder.new(user, { group_id: group.id }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1, todo2, todo3])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'filtering for multiple groups' do
|
|
|
|
let_it_be(:group2) { create(:group) }
|
|
|
|
let_it_be(:group3) { create(:group) }
|
|
|
|
|
|
|
|
let!(:todo1) { create(:todo, user: user, project: project, target: issue) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, group: group, target: merge_request) }
|
|
|
|
let!(:todo3) { create(:todo, user: user, group: group2, target: merge_request) }
|
|
|
|
|
|
|
|
let(:subgroup1) { create(:group, parent: group) }
|
|
|
|
let!(:todo4) { create(:todo, user: user, group: subgroup1, target: issue) }
|
|
|
|
|
|
|
|
let(:subgroup2) { create(:group, parent: group2) }
|
|
|
|
let!(:todo5) { create(:todo, user: user, group: subgroup2, target: issue) }
|
|
|
|
|
|
|
|
let!(:todo6) { create(:todo, user: user, group: group3, target: issue) }
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
it 'returns the expected groups' do
|
|
|
|
todos = finder.new(user, { group_id: [group.id, group2.id] }).execute
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(todos).to match_array([todo1, todo2, todo3, todo4, todo5])
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
context 'by state' do
|
|
|
|
let!(:todo1) { create(:todo, user: user, group: group, target: issue, state: :done) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, group: group, target: issue, state: :pending) }
|
|
|
|
|
|
|
|
it 'returns the expected items when no state is provided' do
|
|
|
|
todos = finder.new(user, {}).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo2])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the expected items when a state is provided' do
|
|
|
|
todos = finder.new(user, { state: :done }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the expected items when multiple states are provided' do
|
|
|
|
todos = finder.new(user, { state: [:pending, :done] }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo1, todo2])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'by project' do
|
|
|
|
let_it_be(:project1) { create(:project) }
|
|
|
|
let_it_be(:project2) { create(:project) }
|
|
|
|
let_it_be(:project3) { create(:project) }
|
|
|
|
|
|
|
|
let!(:todo1) { create(:todo, user: user, project: project1, state: :pending) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, project: project2, state: :pending) }
|
|
|
|
let!(:todo3) { create(:todo, user: user, project: project3, state: :pending) }
|
|
|
|
|
|
|
|
it 'returns the expected todos for one project' do
|
|
|
|
todos = finder.new(user, { project_id: project2.id }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo2])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the expected todos for many projects' do
|
|
|
|
todos = finder.new(user, { project_id: [project2.id, project1.id] }).execute
|
|
|
|
|
|
|
|
expect(todos).to match_array([todo2, todo1])
|
|
|
|
end
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
|
|
|
context 'external authorization' do
|
|
|
|
it_behaves_like 'a finder with external authorization service' do
|
|
|
|
let!(:subject) { create(:todo, project: project, user: user) }
|
|
|
|
let(:project_params) { { project_id: project.id } }
|
|
|
|
end
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
describe '#sort' do
|
|
|
|
context 'by date' do
|
|
|
|
let!(:todo1) { create(:todo, user: user, project: project) }
|
|
|
|
let!(:todo2) { create(:todo, user: user, project: project) }
|
|
|
|
let!(:todo3) { create(:todo, user: user, project: project) }
|
|
|
|
|
|
|
|
it 'sorts with oldest created first' do
|
|
|
|
todos = finder.new(user, { sort: 'id_asc' }).execute
|
|
|
|
|
|
|
|
expect(todos.first).to eq(todo1)
|
|
|
|
expect(todos.second).to eq(todo2)
|
|
|
|
expect(todos.third).to eq(todo3)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts with newest created first' do
|
|
|
|
todos = finder.new(user, { sort: 'id_desc' }).execute
|
|
|
|
|
|
|
|
expect(todos.first).to eq(todo3)
|
|
|
|
expect(todos.second).to eq(todo2)
|
|
|
|
expect(todos.third).to eq(todo1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it "sorts by priority" do
|
2020-03-13 15:44:24 +05:30
|
|
|
project_2 = create(:project)
|
2018-03-27 19:54:05 +05:30
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
label_1 = create(:label, title: 'label_1', project: project, priority: 1)
|
|
|
|
label_2 = create(:label, title: 'label_2', project: project, priority: 2)
|
|
|
|
label_3 = create(:label, title: 'label_3', project: project, priority: 3)
|
2018-03-27 19:54:05 +05:30
|
|
|
label_1_2 = create(:label, title: 'label_1', project: project_2, priority: 1)
|
2016-09-13 17:45:13 +05:30
|
|
|
|
|
|
|
issue_1 = create(:issue, title: 'issue_1', project: project)
|
|
|
|
issue_2 = create(:issue, title: 'issue_2', project: project)
|
|
|
|
issue_3 = create(:issue, title: 'issue_3', project: project)
|
|
|
|
issue_4 = create(:issue, title: 'issue_4', project: project)
|
2018-03-27 19:54:05 +05:30
|
|
|
merge_request_1 = create(:merge_request, source_project: project_2)
|
2016-09-13 17:45:13 +05:30
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
merge_request_1.labels << label_1_2
|
2016-09-13 17:45:13 +05:30
|
|
|
|
|
|
|
# Covers the case where Todo has more than one label
|
|
|
|
issue_3.labels << label_1
|
|
|
|
issue_3.labels << label_3
|
|
|
|
|
|
|
|
issue_2.labels << label_3
|
|
|
|
issue_1.labels << label_2
|
|
|
|
|
|
|
|
todo_1 = create(:todo, user: user, project: project, target: issue_4)
|
|
|
|
todo_2 = create(:todo, user: user, project: project, target: issue_2)
|
|
|
|
todo_3 = create(:todo, user: user, project: project, target: issue_3, created_at: 2.hours.ago)
|
|
|
|
todo_4 = create(:todo, user: user, project: project, target: issue_1)
|
2018-03-27 19:54:05 +05:30
|
|
|
todo_5 = create(:todo, user: user, project: project_2, target: merge_request_1, created_at: 1.hour.ago)
|
|
|
|
|
|
|
|
project_2.add_developer(user)
|
2016-09-13 17:45:13 +05:30
|
|
|
|
|
|
|
todos = finder.new(user, { sort: 'priority' }).execute
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
expect(todos).to eq([todo_3, todo_5, todo_4, todo_2, todo_1])
|
2016-09-13 17:45:13 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-12-13 13:39:08 +05:30
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
describe '.todo_types' do
|
|
|
|
it 'returns the expected types' do
|
|
|
|
expected_result =
|
|
|
|
if Gitlab.ee?
|
|
|
|
%w[Epic Issue MergeRequest]
|
|
|
|
else
|
|
|
|
%w[Issue MergeRequest]
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(described_class.todo_types).to contain_exactly(*expected_result)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
describe '#any_for_target?' do
|
|
|
|
it 'returns true if there are any todos for the given target' do
|
|
|
|
todo = create(:todo, :pending)
|
|
|
|
finder = described_class.new(todo.user)
|
|
|
|
|
|
|
|
expect(finder.any_for_target?(todo.target)).to eq(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false if there are no todos for the given target' do
|
|
|
|
issue = create(:issue)
|
|
|
|
finder = described_class.new(issue.author)
|
|
|
|
|
|
|
|
expect(finder.any_for_target?(issue)).to eq(false)
|
|
|
|
end
|
|
|
|
end
|
2016-09-13 17:45:13 +05:30
|
|
|
end
|