2019-07-07 11:18:12 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
RSpec.describe Milestone, 'Milestoneish' do
|
2016-06-02 11:05:42 +05:30
|
|
|
let(:author) { create(:user) }
|
|
|
|
let(:assignee) { create(:user) }
|
|
|
|
let(:non_member) { create(:user) }
|
|
|
|
let(:member) { create(:user) }
|
2016-06-16 23:09:34 +05:30
|
|
|
let(:guest) { create(:user) }
|
2016-06-02 11:05:42 +05:30
|
|
|
let(:admin) { create(:admin) }
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:project) { create(:project, :public) }
|
2016-06-02 11:05:42 +05:30
|
|
|
let(:milestone) { create(:milestone, project: project) }
|
2019-03-13 22:55:13 +05:30
|
|
|
let(:label1) { create(:label, project: project) }
|
|
|
|
let(:label2) { create(:label, project: project) }
|
|
|
|
let!(:issue) { create(:issue, project: project, milestone: milestone, assignees: [member], labels: [label1]) }
|
|
|
|
let!(:security_issue_1) { create(:issue, :confidential, project: project, author: author, milestone: milestone, labels: [label2]) }
|
2017-08-17 22:00:37 +05:30
|
|
|
let!(:security_issue_2) { create(:issue, :confidential, project: project, assignees: [assignee], milestone: milestone) }
|
2016-06-02 11:05:42 +05:30
|
|
|
let!(:closed_issue_1) { create(:issue, :closed, project: project, milestone: milestone) }
|
|
|
|
let!(:closed_issue_2) { create(:issue, :closed, project: project, milestone: milestone) }
|
|
|
|
let!(:closed_security_issue_1) { create(:issue, :confidential, :closed, project: project, author: author, milestone: milestone) }
|
2017-08-17 22:00:37 +05:30
|
|
|
let!(:closed_security_issue_2) { create(:issue, :confidential, :closed, project: project, assignees: [assignee], milestone: milestone) }
|
2016-06-02 11:05:42 +05:30
|
|
|
let!(:closed_security_issue_3) { create(:issue, :confidential, :closed, project: project, author: author, milestone: milestone) }
|
2017-08-17 22:00:37 +05:30
|
|
|
let!(:closed_security_issue_4) { create(:issue, :confidential, :closed, project: project, assignees: [assignee], milestone: milestone) }
|
2016-06-02 11:05:42 +05:30
|
|
|
let!(:merge_request) { create(:merge_request, source_project: project, target_project: project, milestone: milestone) }
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:label_1) { create(:label, title: 'label_1', project: project, priority: 1) }
|
|
|
|
let(:label_2) { create(:label, title: 'label_2', project: project, priority: 2) }
|
|
|
|
let(:label_3) { create(:label, title: 'label_3', project: project) }
|
2016-06-02 11:05:42 +05:30
|
|
|
|
|
|
|
before do
|
2018-03-17 18:26:18 +05:30
|
|
|
project.add_developer(member)
|
|
|
|
project.add_guest(guest)
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
describe '#sorted_issues' do
|
2020-04-22 19:07:51 +05:30
|
|
|
before do
|
2017-09-10 17:25:29 +05:30
|
|
|
issue.labels << label_1
|
|
|
|
security_issue_1.labels << label_2
|
|
|
|
closed_issue_1.labels << label_3
|
2020-04-22 19:07:51 +05:30
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
it 'sorts issues by label priority' do
|
2017-09-10 17:25:29 +05:30
|
|
|
issues = milestone.sorted_issues(member)
|
|
|
|
|
|
|
|
expect(issues.first).to eq(issue)
|
|
|
|
expect(issues.second).to eq(security_issue_1)
|
|
|
|
expect(issues.third).not_to eq(closed_issue_1)
|
|
|
|
end
|
2020-04-22 19:07:51 +05:30
|
|
|
|
|
|
|
it 'limits issue count and keeps the ordering' do
|
|
|
|
stub_const('Milestoneish::DISPLAY_ISSUES_LIMIT', 4)
|
|
|
|
|
|
|
|
issues = milestone.sorted_issues(member)
|
|
|
|
# Cannot use issues.count here because it is sorting
|
|
|
|
# by a virtual column 'highest_priority' and it will break
|
|
|
|
# the query.
|
|
|
|
total_issues_count = issues.opened.unassigned.length + issues.opened.assigned.length + issues.closed.length
|
|
|
|
expect(issues.length).to eq(4)
|
|
|
|
expect(total_issues_count).to eq(4)
|
|
|
|
expect(issues.first).to eq(issue)
|
|
|
|
expect(issues.second).to eq(security_issue_1)
|
|
|
|
expect(issues.third).not_to eq(closed_issue_1)
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2019-03-13 22:55:13 +05:30
|
|
|
context 'attributes visibility' do
|
|
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
|
|
|
|
let(:users) do
|
|
|
|
{
|
|
|
|
anonymous: nil,
|
|
|
|
non_member: non_member,
|
|
|
|
guest: guest,
|
|
|
|
member: member,
|
|
|
|
assignee: assignee
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:project_visibility_levels) do
|
|
|
|
{
|
|
|
|
public: Gitlab::VisibilityLevel::PUBLIC,
|
|
|
|
internal: Gitlab::VisibilityLevel::INTERNAL,
|
|
|
|
private: Gitlab::VisibilityLevel::PRIVATE
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#issue_participants_visible_by_user' do
|
|
|
|
where(:visibility, :user_role, :result) do
|
|
|
|
:public | nil | [:member]
|
|
|
|
:public | :non_member | [:member]
|
|
|
|
:public | :guest | [:member]
|
|
|
|
:public | :member | [:member, :assignee]
|
|
|
|
:internal | nil | []
|
|
|
|
:internal | :non_member | [:member]
|
|
|
|
:internal | :guest | [:member]
|
|
|
|
:internal | :member | [:member, :assignee]
|
|
|
|
:private | nil | []
|
|
|
|
:private | :non_member | []
|
|
|
|
:private | :guest | [:member]
|
|
|
|
:private | :member | [:member, :assignee]
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
before do
|
|
|
|
project.update(visibility_level: project_visibility_levels[visibility])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the proper participants' do
|
|
|
|
user = users[user_role]
|
|
|
|
participants = result.map { |role| users[role] }
|
|
|
|
|
|
|
|
expect(milestone.issue_participants_visible_by_user(user)).to match_array(participants)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#issue_labels_visible_by_user' do
|
|
|
|
let(:labels) do
|
|
|
|
{
|
|
|
|
label1: label1,
|
|
|
|
label2: label2
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
where(:visibility, :user_role, :result) do
|
|
|
|
:public | nil | [:label1]
|
|
|
|
:public | :non_member | [:label1]
|
|
|
|
:public | :guest | [:label1]
|
|
|
|
:public | :member | [:label1, :label2]
|
|
|
|
:internal | nil | []
|
|
|
|
:internal | :non_member | [:label1]
|
|
|
|
:internal | :guest | [:label1]
|
|
|
|
:internal | :member | [:label1, :label2]
|
|
|
|
:private | nil | []
|
|
|
|
:private | :non_member | []
|
|
|
|
:private | :guest | [:label1]
|
|
|
|
:private | :member | [:label1, :label2]
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
before do
|
|
|
|
project.update(visibility_level: project_visibility_levels[visibility])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the proper participants' do
|
|
|
|
user = users[user_role]
|
|
|
|
expected_labels = result.map { |label| labels[label] }
|
|
|
|
|
|
|
|
expect(milestone.issue_labels_visible_by_user(user)).to match_array(expected_labels)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
describe '#sorted_merge_requests' do
|
|
|
|
it 'sorts merge requests by label priority' do
|
|
|
|
merge_request_1 = create(:labeled_merge_request, labels: [label_2], source_project: project, source_branch: 'branch_1', milestone: milestone)
|
|
|
|
merge_request_2 = create(:labeled_merge_request, labels: [label_1], source_project: project, source_branch: 'branch_2', milestone: milestone)
|
|
|
|
merge_request_3 = create(:labeled_merge_request, labels: [label_3], source_project: project, source_branch: 'branch_3', milestone: milestone)
|
|
|
|
|
2019-03-13 22:55:13 +05:30
|
|
|
merge_requests = milestone.sorted_merge_requests(member)
|
2017-09-10 17:25:29 +05:30
|
|
|
|
|
|
|
expect(merge_requests.first).to eq(merge_request_2)
|
|
|
|
expect(merge_requests.second).to eq(merge_request_1)
|
|
|
|
expect(merge_requests.third).to eq(merge_request_3)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-13 22:55:13 +05:30
|
|
|
describe '#merge_requests_visible_to_user' do
|
|
|
|
let(:merge_request) { create(:merge_request, source_project: project, milestone: milestone) }
|
|
|
|
|
|
|
|
context 'when project is private' do
|
|
|
|
before do
|
|
|
|
project.update(visibility_level: Gitlab::VisibilityLevel::PRIVATE)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not return any merge request for a non member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(non_member)
|
|
|
|
expect(merge_requests).to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestone merge requests for a member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(member)
|
|
|
|
expect(merge_requests).to contain_exactly(merge_request)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when project is public' do
|
|
|
|
context 'when merge requests are available to anyone' do
|
|
|
|
it 'returns milestone merge requests for a non member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(non_member)
|
|
|
|
expect(merge_requests).to contain_exactly(merge_request)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when merge requests are available to project members' do
|
|
|
|
before do
|
|
|
|
project.project_feature.update(merge_requests_access_level: ProjectFeature::PRIVATE)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not return any merge request for a non member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(non_member)
|
|
|
|
expect(merge_requests).to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestone merge requests for a member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(member)
|
|
|
|
expect(merge_requests).to contain_exactly(merge_request)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2020-01-01 13:55:28 +05:30
|
|
|
|
|
|
|
context 'when milestone is at parent level group' do
|
|
|
|
let(:parent_group) { create(:group) }
|
|
|
|
let(:group) { create(:group, parent: parent_group) }
|
|
|
|
let(:project) { create(:project, namespace: group) }
|
|
|
|
let(:milestone) { create(:milestone, group: parent_group) }
|
|
|
|
|
|
|
|
it 'does not return any merge request for a non member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(non_member)
|
|
|
|
expect(merge_requests).to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestone merge requests for a member' do
|
|
|
|
merge_requests = milestone.merge_requests_visible_to_user(member)
|
|
|
|
expect(merge_requests).to contain_exactly(merge_request)
|
|
|
|
end
|
|
|
|
end
|
2019-03-13 22:55:13 +05:30
|
|
|
end
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
describe '#complete?', :use_clean_rails_memory_store_caching do
|
2016-06-02 11:05:42 +05:30
|
|
|
it 'returns false when has items opened' do
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(milestone.complete?).to eq false
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns true when all items are closed' do
|
|
|
|
issue.close
|
2020-04-08 14:13:33 +05:30
|
|
|
security_issue_1.close
|
|
|
|
security_issue_2.close
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(milestone.complete?).to eq true
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
describe '#percent_complete', :use_clean_rails_memory_store_caching do
|
2019-09-04 21:01:54 +05:30
|
|
|
context 'division by zero' do
|
|
|
|
let(:new_milestone) { build_stubbed(:milestone) }
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
it { expect(new_milestone.percent_complete).to eq(0) }
|
2019-09-04 21:01:54 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
describe '#closed_issues_count' do
|
|
|
|
it 'counts all closed issues including confidential' do
|
|
|
|
expect(milestone.closed_issues_count).to eq 6
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
2020-04-08 14:13:33 +05:30
|
|
|
end
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
describe '#total_issues_count' do
|
|
|
|
it 'counts all issues including confidential' do
|
|
|
|
expect(milestone.total_issues_count).to eq 9
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
describe '#remaining_days' do
|
|
|
|
it 'shows 0 if no due date' do
|
|
|
|
milestone = build_stubbed(:milestone)
|
|
|
|
|
|
|
|
expect(milestone.remaining_days).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows 0 if expired' do
|
|
|
|
milestone = build_stubbed(:milestone, due_date: 2.days.ago)
|
|
|
|
|
|
|
|
expect(milestone.remaining_days).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows correct remaining days' do
|
|
|
|
milestone = build_stubbed(:milestone, due_date: 2.days.from_now)
|
|
|
|
|
|
|
|
expect(milestone.remaining_days).to eq(2)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#elapsed_days' do
|
|
|
|
it 'shows 0 if no start_date set' do
|
|
|
|
milestone = build_stubbed(:milestone)
|
|
|
|
|
|
|
|
expect(milestone.elapsed_days).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows 0 if start_date is a future' do
|
2020-06-23 00:09:42 +05:30
|
|
|
milestone = build_stubbed(:milestone, start_date: Time.current + 2.days)
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(milestone.elapsed_days).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows correct amount of days' do
|
2020-06-23 00:09:42 +05:30
|
|
|
milestone = build_stubbed(:milestone, start_date: Time.current - 2.days)
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(milestone.elapsed_days).to eq(2)
|
|
|
|
end
|
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
describe '#total_time_spent' do
|
|
|
|
it 'calculates total time spent' do
|
2018-03-17 18:26:18 +05:30
|
|
|
closed_issue_1.spend_time(duration: 300, user_id: author.id)
|
|
|
|
closed_issue_1.save!
|
|
|
|
closed_issue_2.spend_time(duration: 600, user_id: assignee.id)
|
|
|
|
closed_issue_2.save!
|
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(milestone.total_time_spent).to eq(900)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'includes merge request time spent' do
|
|
|
|
closed_issue_1.spend_time(duration: 300, user_id: author.id)
|
|
|
|
closed_issue_1.save!
|
|
|
|
merge_request.spend_time(duration: 900, user_id: author.id)
|
|
|
|
merge_request.save!
|
|
|
|
|
|
|
|
expect(milestone.total_time_spent).to eq(1200)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#human_total_time_spent' do
|
|
|
|
it 'returns nil if no time has been spent' do
|
|
|
|
expect(milestone.human_total_time_spent).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#total_time_estimate' do
|
|
|
|
it 'calculates total estimate' do
|
|
|
|
closed_issue_1.time_estimate = 300
|
|
|
|
closed_issue_1.save!
|
|
|
|
closed_issue_2.time_estimate = 600
|
|
|
|
closed_issue_2.save!
|
|
|
|
|
|
|
|
expect(milestone.total_time_estimate).to eq(900)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'includes merge request time estimate' do
|
|
|
|
closed_issue_1.time_estimate = 300
|
|
|
|
closed_issue_1.save!
|
|
|
|
merge_request.time_estimate = 900
|
|
|
|
merge_request.save!
|
|
|
|
|
|
|
|
expect(milestone.total_time_estimate).to eq(1200)
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
describe '#human_total_time_estimate' do
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'returns nil if no time has been spent' do
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(milestone.human_total_time_estimate).to be_nil
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
end
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|