2019-03-30 03:23:56 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2012-04-08 17:28:58 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2017-07-10 10:24:02 -04:00
|
|
|
describe Milestone do
|
2018-04-20 10:00:15 -04:00
|
|
|
describe 'modules' do
|
|
|
|
context 'with a project' do
|
|
|
|
it_behaves_like 'AtomicInternalId' do
|
|
|
|
let(:internal_id_attribute) { :iid }
|
|
|
|
let(:instance) { build(:milestone, project: build(:project), group: nil) }
|
2018-05-14 04:41:56 -04:00
|
|
|
let(:scope) { :project }
|
2018-04-20 10:00:15 -04:00
|
|
|
let(:scope_attrs) { { project: instance.project } }
|
|
|
|
let(:usage) { :milestones }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with a group' do
|
|
|
|
it_behaves_like 'AtomicInternalId' do
|
|
|
|
let(:internal_id_attribute) { :iid }
|
|
|
|
let(:instance) { build(:milestone, project: nil, group: build(:group)) }
|
2018-05-14 04:41:56 -04:00
|
|
|
let(:scope) { :group }
|
2018-04-20 10:00:15 -04:00
|
|
|
let(:scope_attrs) { { namespace: instance.group } }
|
|
|
|
let(:usage) { :milestones }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-04-08 18:01:42 -04:00
|
|
|
describe "Validation" do
|
2015-05-21 17:49:06 -04:00
|
|
|
before do
|
|
|
|
allow(subject).to receive(:set_iid).and_return(false)
|
|
|
|
end
|
|
|
|
|
2016-11-15 12:48:30 -05:00
|
|
|
describe 'start_date' do
|
2019-05-31 06:31:47 -04:00
|
|
|
it 'adds an error when start_date is greater then due_date' do
|
2016-11-15 12:48:30 -05:00
|
|
|
milestone = build(:milestone, start_date: Date.tomorrow, due_date: Date.yesterday)
|
|
|
|
|
|
|
|
expect(milestone).not_to be_valid
|
2017-11-18 13:22:11 -05:00
|
|
|
expect(milestone.errors[:due_date]).to include("must be greater than start date")
|
2016-11-15 12:48:30 -05:00
|
|
|
end
|
2019-05-31 06:31:47 -04:00
|
|
|
|
|
|
|
it 'adds an error when start_date is greater than 9999-12-31' do
|
|
|
|
milestone = build(:milestone, start_date: Date.new(10000, 1, 1))
|
|
|
|
|
|
|
|
expect(milestone).not_to be_valid
|
|
|
|
expect(milestone.errors[:start_date]).to include("date must not be after 9999-12-31")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'due_date' do
|
|
|
|
it 'adds an error when due_date is greater than 9999-12-31' do
|
|
|
|
milestone = build(:milestone, due_date: Date.new(10000, 1, 1))
|
|
|
|
|
|
|
|
expect(milestone).not_to be_valid
|
|
|
|
expect(milestone.errors[:due_date]).to include("date must not be after 9999-12-31")
|
|
|
|
end
|
2016-11-15 12:48:30 -05:00
|
|
|
end
|
2019-09-03 05:38:59 -04:00
|
|
|
|
2019-11-14 19:06:05 -05:00
|
|
|
describe 'title' do
|
|
|
|
it { is_expected.to validate_presence_of(:title) }
|
|
|
|
|
|
|
|
it 'is invalid if title would be empty after sanitation' do
|
|
|
|
milestone = build(:milestone, project: project, title: '<img src=x onerror=prompt(1)>')
|
|
|
|
|
|
|
|
expect(milestone).not_to be_valid
|
|
|
|
expect(milestone.errors[:title]).to include("can't be blank")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-09-16 11:06:26 -04:00
|
|
|
describe 'milestone_releases' do
|
2019-09-03 05:38:59 -04:00
|
|
|
let(:milestone) { build(:milestone, project: project) }
|
|
|
|
|
|
|
|
context 'when it is tied to a release for another project' do
|
|
|
|
it 'creates a validation error' do
|
|
|
|
other_project = create(:project)
|
2019-09-16 11:06:26 -04:00
|
|
|
milestone.releases << build(:release, project: other_project)
|
2019-09-03 05:38:59 -04:00
|
|
|
expect(milestone).not_to be_valid
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when it is tied to a release for the same project' do
|
|
|
|
it 'is valid' do
|
2019-09-16 11:06:26 -04:00
|
|
|
milestone.releases << build(:release, project: project)
|
2019-09-03 05:38:59 -04:00
|
|
|
expect(milestone).to be_valid
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-11-15 12:48:30 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "Associations" do
|
|
|
|
it { is_expected.to belong_to(:project) }
|
|
|
|
it { is_expected.to have_many(:issues) }
|
2019-09-16 11:06:26 -04:00
|
|
|
it { is_expected.to have_many(:releases) }
|
|
|
|
it { is_expected.to have_many(:milestone_releases) }
|
2012-04-08 18:01:42 -04:00
|
|
|
end
|
|
|
|
|
2017-08-02 15:55:11 -04:00
|
|
|
let(:project) { create(:project, :public) }
|
2016-12-12 03:43:56 -05:00
|
|
|
let(:milestone) { create(:milestone, project: project) }
|
|
|
|
let(:issue) { create(:issue, project: project) }
|
2016-03-17 16:59:30 -04:00
|
|
|
let(:user) { create(:user) }
|
2012-04-08 18:01:42 -04:00
|
|
|
|
2016-05-04 17:21:57 -04:00
|
|
|
describe "#title" do
|
2016-09-26 19:47:34 -04:00
|
|
|
let(:milestone) { create(:milestone, title: "<b>foo & bar -> 2.2</b>") }
|
2016-05-04 17:21:57 -04:00
|
|
|
|
|
|
|
it "sanitizes title" do
|
2016-09-26 19:47:34 -04:00
|
|
|
expect(milestone.title).to eq("foo & bar -> 2.2")
|
2016-05-04 17:21:57 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-03 13:06:49 -05:00
|
|
|
describe '#merge_requests_enabled?' do
|
|
|
|
context "per project" do
|
|
|
|
it "is true for projects with MRs enabled" do
|
|
|
|
project = create(:project, :merge_requests_enabled)
|
|
|
|
milestone = create(:milestone, project: project)
|
|
|
|
|
|
|
|
expect(milestone.merge_requests_enabled?).to be(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "is false for projects with MRs disabled" do
|
|
|
|
project = create(:project, :repository_enabled, :merge_requests_disabled)
|
|
|
|
milestone = create(:milestone, project: project)
|
|
|
|
|
|
|
|
expect(milestone.merge_requests_enabled?).to be(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "is false for projects with repository disabled" do
|
|
|
|
project = create(:project, :repository_disabled)
|
|
|
|
milestone = create(:milestone, project: project)
|
|
|
|
|
|
|
|
expect(milestone.merge_requests_enabled?).to be(false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "per group" do
|
|
|
|
let(:group) { create(:group) }
|
|
|
|
let(:milestone) { create(:milestone, group: group) }
|
|
|
|
|
|
|
|
it "is always true for groups, for performance reasons" do
|
|
|
|
expect(milestone.merge_requests_enabled?).to be(true)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-07-07 11:08:49 -04:00
|
|
|
describe "unique milestone title" do
|
|
|
|
context "per project" do
|
|
|
|
it "does not accept the same title in a project twice" do
|
2017-07-25 13:09:00 -04:00
|
|
|
new_milestone = described_class.new(project: milestone.project, title: milestone.title)
|
2017-07-07 11:08:49 -04:00
|
|
|
expect(new_milestone).not_to be_valid
|
|
|
|
end
|
|
|
|
|
|
|
|
it "accepts the same title in another project" do
|
2017-08-02 15:55:11 -04:00
|
|
|
project = create(:project)
|
2017-07-25 13:09:00 -04:00
|
|
|
new_milestone = described_class.new(project: project, title: milestone.title)
|
2017-07-07 11:08:49 -04:00
|
|
|
|
|
|
|
expect(new_milestone).to be_valid
|
|
|
|
end
|
2016-02-02 04:44:30 -05:00
|
|
|
end
|
|
|
|
|
2017-07-07 11:08:49 -04:00
|
|
|
context "per group" do
|
|
|
|
let(:group) { create(:group) }
|
|
|
|
let(:milestone) { create(:milestone, group: group) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
project.update(group: group)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "does not accept the same title in a group twice" do
|
2017-07-25 13:09:00 -04:00
|
|
|
new_milestone = described_class.new(group: group, title: milestone.title)
|
2017-07-07 11:08:49 -04:00
|
|
|
|
|
|
|
expect(new_milestone).not_to be_valid
|
|
|
|
end
|
2016-02-02 04:44:30 -05:00
|
|
|
|
2017-07-07 11:08:49 -04:00
|
|
|
it "does not accept the same title of a child project milestone" do
|
|
|
|
create(:milestone, project: group.projects.first)
|
|
|
|
|
2017-07-25 13:09:00 -04:00
|
|
|
new_milestone = described_class.new(group: group, title: milestone.title)
|
2017-07-07 11:08:49 -04:00
|
|
|
|
|
|
|
expect(new_milestone).not_to be_valid
|
|
|
|
end
|
2016-02-02 04:44:30 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-09-17 12:35:39 -04:00
|
|
|
describe '.order_by_name_asc' do
|
|
|
|
it 'sorts by name ascending' do
|
|
|
|
milestone1 = create(:milestone, title: 'Foo')
|
|
|
|
milestone2 = create(:milestone, title: 'Bar')
|
|
|
|
|
|
|
|
expect(described_class.order_by_name_asc).to eq([milestone2, milestone1])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.reorder_by_due_date_asc' do
|
|
|
|
it 'reorders the input relation' do
|
|
|
|
milestone1 = create(:milestone, due_date: Date.new(2018, 9, 30))
|
|
|
|
milestone2 = create(:milestone, due_date: Date.new(2018, 10, 20))
|
|
|
|
|
|
|
|
expect(described_class.reorder_by_due_date_asc).to eq([milestone1, milestone2])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-08-25 13:54:38 -04:00
|
|
|
describe "#percent_complete" do
|
2016-08-01 11:00:44 -04:00
|
|
|
it "does not count open issues" do
|
2012-04-08 18:01:42 -04:00
|
|
|
milestone.issues << issue
|
2016-03-17 16:59:30 -04:00
|
|
|
expect(milestone.percent_complete(user)).to eq(0)
|
2012-04-08 18:01:42 -04:00
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it "counts closed issues" do
|
2013-02-18 04:10:09 -05:00
|
|
|
issue.close
|
2012-08-25 13:54:38 -04:00
|
|
|
milestone.issues << issue
|
2016-03-17 16:59:30 -04:00
|
|
|
expect(milestone.percent_complete(user)).to eq(100)
|
2012-08-25 13:54:38 -04:00
|
|
|
end
|
2012-04-08 18:01:42 -04:00
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it "recovers from dividing by zero" do
|
2016-03-17 16:59:30 -04:00
|
|
|
expect(milestone.percent_complete(user)).to eq(0)
|
2012-04-08 18:01:42 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-07-11 18:12:31 -04:00
|
|
|
describe '#expired?' do
|
2013-01-03 12:11:14 -05:00
|
|
|
context "expired" do
|
|
|
|
before do
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(milestone).to receive(:due_date).and_return(Date.today.prev_year)
|
2013-01-03 12:11:14 -05:00
|
|
|
end
|
|
|
|
|
2018-04-19 10:43:20 -04:00
|
|
|
it 'returns true when due_date is in the past' do
|
|
|
|
expect(milestone.expired?).to be_truthy
|
|
|
|
end
|
2013-01-03 12:11:14 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
context "not expired" do
|
|
|
|
before do
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(milestone).to receive(:due_date).and_return(Date.today.next_year)
|
2013-01-03 12:11:14 -05:00
|
|
|
end
|
|
|
|
|
2018-04-19 10:43:20 -04:00
|
|
|
it 'returns false when due_date is in the future' do
|
|
|
|
expect(milestone.expired?).to be_falsey
|
|
|
|
end
|
2013-01-03 12:11:14 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-11-15 12:48:30 -05:00
|
|
|
describe '#upcoming?' do
|
2018-04-19 10:43:20 -04:00
|
|
|
it 'returns true when start_date is in the future' do
|
2016-11-15 12:48:30 -05:00
|
|
|
milestone = build(:milestone, start_date: Time.now + 1.month)
|
|
|
|
expect(milestone.upcoming?).to be_truthy
|
|
|
|
end
|
|
|
|
|
2018-04-19 10:43:20 -04:00
|
|
|
it 'returns false when start_date is in the past' do
|
2016-11-15 12:48:30 -05:00
|
|
|
milestone = build(:milestone, start_date: Date.today.prev_year)
|
|
|
|
expect(milestone.upcoming?).to be_falsey
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-07-11 18:12:31 -04:00
|
|
|
describe '#can_be_closed?' do
|
2015-02-12 13:17:35 -05:00
|
|
|
it { expect(milestone.can_be_closed?).to be_truthy }
|
2013-01-03 12:11:14 -05:00
|
|
|
end
|
2013-02-18 04:38:29 -05:00
|
|
|
|
2016-07-11 18:12:31 -04:00
|
|
|
describe '#can_be_closed?' do
|
2013-02-18 08:52:39 -05:00
|
|
|
before do
|
2019-01-14 05:46:39 -05:00
|
|
|
milestone = create :milestone, project: project
|
|
|
|
create :closed_issue, milestone: milestone, project: project
|
2013-02-18 08:52:39 -05:00
|
|
|
|
2019-01-14 05:46:39 -05:00
|
|
|
create :issue, project: project
|
2013-02-18 08:52:39 -05:00
|
|
|
end
|
2013-02-18 04:38:29 -05:00
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'returns true if milestone active and all nested issues closed' do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(milestone.can_be_closed?).to be_truthy
|
2013-02-18 04:38:29 -05:00
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'returns false if milestone active and not all nested issues closed' do
|
2013-02-18 08:52:39 -05:00
|
|
|
issue.milestone = milestone
|
2013-03-20 17:46:30 -04:00
|
|
|
issue.save
|
2013-02-18 04:38:29 -05:00
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(milestone.can_be_closed?).to be_falsey
|
2013-02-18 04:38:29 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-10-17 01:15:07 -04:00
|
|
|
describe '#to_ability_name' do
|
|
|
|
it 'returns milestone' do
|
|
|
|
milestone = build(:milestone)
|
|
|
|
|
|
|
|
expect(milestone.to_ability_name).to eq('milestone')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-01 11:05:26 -05:00
|
|
|
describe '.search' do
|
|
|
|
let(:milestone) { create(:milestone, title: 'foo', description: 'bar') }
|
|
|
|
|
|
|
|
it 'returns milestones with a matching title' do
|
|
|
|
expect(described_class.search(milestone.title)).to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a partially matching title' do
|
|
|
|
expect(described_class.search(milestone.title[0..2])).to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a matching title regardless of the casing' do
|
|
|
|
expect(described_class.search(milestone.title.upcase)).to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a matching description' do
|
|
|
|
expect(described_class.search(milestone.description)).to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a partially matching description' do
|
2017-06-21 09:48:12 -04:00
|
|
|
expect(described_class.search(milestone.description[0..2]))
|
|
|
|
.to eq([milestone])
|
2016-03-01 11:05:26 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a matching description regardless of the casing' do
|
2017-06-21 09:48:12 -04:00
|
|
|
expect(described_class.search(milestone.description.upcase))
|
|
|
|
.to eq([milestone])
|
2016-03-01 11:05:26 -05:00
|
|
|
end
|
|
|
|
end
|
2016-05-11 12:38:34 -04:00
|
|
|
|
2019-01-17 09:35:23 -05:00
|
|
|
describe '#search_title' do
|
|
|
|
let(:milestone) { create(:milestone, title: 'foo', description: 'bar') }
|
|
|
|
|
|
|
|
it 'returns milestones with a matching title' do
|
|
|
|
expect(described_class.search_title(milestone.title)) .to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a partially matching title' do
|
|
|
|
expect(described_class.search_title(milestone.title[0..2])).to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a matching title regardless of the casing' do
|
|
|
|
expect(described_class.search_title(milestone.title.upcase))
|
|
|
|
.to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'searches only on the title and ignores milestones with a matching description' do
|
|
|
|
create(:milestone, title: 'bar', description: 'foo')
|
|
|
|
|
|
|
|
expect(described_class.search_title(milestone.title)) .to eq([milestone])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-17 07:59:23 -05:00
|
|
|
describe '#for_projects_and_groups' do
|
|
|
|
let(:project) { create(:project) }
|
|
|
|
let(:project_other) { create(:project) }
|
|
|
|
let(:group) { create(:group) }
|
|
|
|
let(:group_other) { create(:group) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create(:milestone, project: project)
|
|
|
|
create(:milestone, project: project_other)
|
|
|
|
create(:milestone, group: group)
|
|
|
|
create(:milestone, group: group_other)
|
|
|
|
end
|
|
|
|
|
|
|
|
subject { described_class.for_projects_and_groups(projects, groups) }
|
|
|
|
|
|
|
|
shared_examples 'filters by projects and groups' do
|
|
|
|
it 'returns milestones filtered by project' do
|
|
|
|
milestones = described_class.for_projects_and_groups(projects, [])
|
|
|
|
|
|
|
|
expect(milestones.count).to eq(1)
|
|
|
|
expect(milestones.first.project_id).to eq(project.id)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones filtered by group' do
|
|
|
|
milestones = described_class.for_projects_and_groups([], groups)
|
|
|
|
|
|
|
|
expect(milestones.count).to eq(1)
|
|
|
|
expect(milestones.first.group_id).to eq(group.id)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones filtered by both project and group' do
|
|
|
|
milestones = described_class.for_projects_and_groups(projects, groups)
|
|
|
|
|
|
|
|
expect(milestones.count).to eq(2)
|
|
|
|
expect(milestones).to contain_exactly(project.milestones.first, group.milestones.first)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'ids as params' do
|
|
|
|
let(:projects) { [project.id] }
|
|
|
|
let(:groups) { [group.id] }
|
|
|
|
|
|
|
|
it_behaves_like 'filters by projects and groups'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'relations as params' do
|
2019-01-11 09:58:18 -05:00
|
|
|
let(:projects) { Project.where(id: project.id).select(:id) }
|
|
|
|
let(:groups) { Group.where(id: group.id).select(:id) }
|
2018-12-17 07:59:23 -05:00
|
|
|
|
|
|
|
it_behaves_like 'filters by projects and groups'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'objects as params' do
|
|
|
|
let(:projects) { [project] }
|
|
|
|
let(:groups) { [group] }
|
|
|
|
|
|
|
|
it_behaves_like 'filters by projects and groups'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns no records if projects and groups are nil' do
|
|
|
|
milestones = described_class.for_projects_and_groups(nil, nil)
|
|
|
|
|
|
|
|
expect(milestones).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-15 01:56:51 -05:00
|
|
|
describe '.upcoming_ids' do
|
|
|
|
let(:group_1) { create(:group) }
|
|
|
|
let(:group_2) { create(:group) }
|
|
|
|
let(:group_3) { create(:group) }
|
|
|
|
let(:groups) { [group_1, group_2, group_3] }
|
|
|
|
|
|
|
|
let!(:past_milestone_group_1) { create(:milestone, group: group_1, due_date: Time.now - 1.day) }
|
|
|
|
let!(:current_milestone_group_1) { create(:milestone, group: group_1, due_date: Time.now + 1.day) }
|
|
|
|
let!(:future_milestone_group_1) { create(:milestone, group: group_1, due_date: Time.now + 2.days) }
|
|
|
|
|
|
|
|
let!(:past_milestone_group_2) { create(:milestone, group: group_2, due_date: Time.now - 1.day) }
|
|
|
|
let!(:closed_milestone_group_2) { create(:milestone, :closed, group: group_2, due_date: Time.now + 1.day) }
|
|
|
|
let!(:current_milestone_group_2) { create(:milestone, group: group_2, due_date: Time.now + 2.days) }
|
|
|
|
|
|
|
|
let!(:past_milestone_group_3) { create(:milestone, group: group_3, due_date: Time.now - 1.day) }
|
|
|
|
|
2017-08-02 15:55:11 -04:00
|
|
|
let(:project_1) { create(:project) }
|
|
|
|
let(:project_2) { create(:project) }
|
|
|
|
let(:project_3) { create(:project) }
|
2016-05-11 12:38:34 -04:00
|
|
|
let(:projects) { [project_1, project_2, project_3] }
|
|
|
|
|
|
|
|
let!(:past_milestone_project_1) { create(:milestone, project: project_1, due_date: Time.now - 1.day) }
|
|
|
|
let!(:current_milestone_project_1) { create(:milestone, project: project_1, due_date: Time.now + 1.day) }
|
|
|
|
let!(:future_milestone_project_1) { create(:milestone, project: project_1, due_date: Time.now + 2.days) }
|
|
|
|
|
|
|
|
let!(:past_milestone_project_2) { create(:milestone, project: project_2, due_date: Time.now - 1.day) }
|
|
|
|
let!(:closed_milestone_project_2) { create(:milestone, :closed, project: project_2, due_date: Time.now + 1.day) }
|
|
|
|
let!(:current_milestone_project_2) { create(:milestone, project: project_2, due_date: Time.now + 2.days) }
|
|
|
|
|
|
|
|
let!(:past_milestone_project_3) { create(:milestone, project: project_3, due_date: Time.now - 1.day) }
|
|
|
|
|
2018-12-17 07:59:23 -05:00
|
|
|
let(:milestone_ids) { described_class.upcoming_ids(projects, groups).map(&:id) }
|
2018-11-15 01:56:51 -05:00
|
|
|
|
|
|
|
it 'returns the next upcoming open milestone ID for each project and group' do
|
|
|
|
expect(milestone_ids).to contain_exactly(
|
|
|
|
current_milestone_project_1.id,
|
|
|
|
current_milestone_project_2.id,
|
|
|
|
current_milestone_group_1.id,
|
|
|
|
current_milestone_group_2.id
|
|
|
|
)
|
2016-05-11 12:38:34 -04:00
|
|
|
end
|
|
|
|
|
2018-11-15 01:56:51 -05:00
|
|
|
context 'when the projects and groups have no open upcoming milestones' do
|
2016-05-11 12:38:34 -04:00
|
|
|
let(:projects) { [project_3] }
|
2018-11-15 01:56:51 -05:00
|
|
|
let(:groups) { [group_3] }
|
2016-05-11 12:38:34 -04:00
|
|
|
|
|
|
|
it 'returns no results' do
|
|
|
|
expect(milestone_ids).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-11-02 19:49:13 -04:00
|
|
|
|
|
|
|
describe '#to_reference' do
|
2017-08-03 07:50:06 -04:00
|
|
|
let(:group) { build_stubbed(:group) }
|
|
|
|
let(:project) { build_stubbed(:project, name: 'sample-project') }
|
|
|
|
let(:another_project) { build_stubbed(:project, name: 'another-project', namespace: project.namespace) }
|
|
|
|
|
|
|
|
context 'for a project milestone' do
|
|
|
|
let(:milestone) { build_stubbed(:milestone, iid: 1, project: project, name: 'milestone') }
|
|
|
|
|
|
|
|
it 'returns a String reference to the object' do
|
2017-09-20 05:55:54 -04:00
|
|
|
expect(milestone.to_reference).to eq '%"milestone"'
|
2017-08-03 07:50:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns a reference by name when the format is set to :name' do
|
|
|
|
expect(milestone.to_reference(format: :name)).to eq '%"milestone"'
|
|
|
|
end
|
2016-11-02 19:49:13 -04:00
|
|
|
|
2017-08-03 07:50:06 -04:00
|
|
|
it 'supports a cross-project reference' do
|
2017-09-20 05:55:54 -04:00
|
|
|
expect(milestone.to_reference(another_project)).to eq 'sample-project%"milestone"'
|
2017-08-03 07:50:06 -04:00
|
|
|
end
|
2016-11-02 19:49:13 -04:00
|
|
|
end
|
|
|
|
|
2017-08-03 07:50:06 -04:00
|
|
|
context 'for a group milestone' do
|
|
|
|
let(:milestone) { build_stubbed(:milestone, iid: 1, group: group, name: 'milestone') }
|
|
|
|
|
2017-09-20 05:55:54 -04:00
|
|
|
it 'returns a group milestone reference with a default format' do
|
|
|
|
expect(milestone.to_reference).to eq '%"milestone"'
|
2017-08-03 07:50:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns a reference by name when the format is set to :name' do
|
|
|
|
expect(milestone.to_reference(format: :name)).to eq '%"milestone"'
|
|
|
|
end
|
|
|
|
|
2017-09-20 05:55:54 -04:00
|
|
|
it 'does supports cross-project references within a group' do
|
2017-08-03 07:50:06 -04:00
|
|
|
expect(milestone.to_reference(another_project, format: :name)).to eq '%"milestone"'
|
|
|
|
end
|
2017-09-20 05:55:54 -04:00
|
|
|
|
|
|
|
it 'raises an error when using iid format' do
|
|
|
|
expect { milestone.to_reference(format: :iid) }
|
|
|
|
.to raise_error(ArgumentError, 'Cannot refer to a group milestone by an internal id!')
|
|
|
|
end
|
2016-11-02 19:49:13 -04:00
|
|
|
end
|
|
|
|
end
|
2017-05-29 05:20:41 -04:00
|
|
|
|
2018-12-19 06:12:19 -05:00
|
|
|
describe '#reference_link_text' do
|
|
|
|
let(:project) { build_stubbed(:project, name: 'sample-project') }
|
|
|
|
let(:milestone) { build_stubbed(:milestone, iid: 1, project: project, name: 'milestone') }
|
|
|
|
|
|
|
|
it 'returns the title with the reference prefix' do
|
|
|
|
expect(milestone.reference_link_text).to eq '%milestone'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-29 05:20:41 -04:00
|
|
|
describe '#participants' do
|
2017-08-02 15:55:11 -04:00
|
|
|
let(:project) { build(:project, name: 'sample-project') }
|
2017-05-29 05:20:41 -04:00
|
|
|
let(:milestone) { build(:milestone, iid: 1, project: project) }
|
|
|
|
|
|
|
|
it 'returns participants without duplicates' do
|
|
|
|
user = create :user
|
|
|
|
create :issue, project: project, milestone: milestone, assignees: [user]
|
|
|
|
create :issue, project: project, milestone: milestone, assignees: [user]
|
|
|
|
|
|
|
|
expect(milestone.participants).to eq [user]
|
|
|
|
end
|
|
|
|
end
|
2018-07-27 13:02:54 -04:00
|
|
|
|
|
|
|
describe '.sort_by_attribute' do
|
|
|
|
set(:milestone_1) { create(:milestone, title: 'Foo') }
|
|
|
|
set(:milestone_2) { create(:milestone, title: 'Bar') }
|
|
|
|
set(:milestone_3) { create(:milestone, title: 'Zoo') }
|
|
|
|
|
|
|
|
context 'ordering by name ascending' do
|
|
|
|
it 'sorts by title ascending' do
|
|
|
|
expect(described_class.sort_by_attribute('name_asc'))
|
|
|
|
.to eq([milestone_2, milestone_1, milestone_3])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'ordering by name descending' do
|
|
|
|
it 'sorts by title descending' do
|
|
|
|
expect(described_class.sort_by_attribute('name_desc'))
|
|
|
|
.to eq([milestone_3, milestone_1, milestone_2])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-10-16 09:18:25 -04:00
|
|
|
|
|
|
|
describe '.states_count' do
|
|
|
|
context 'when the projects have milestones' do
|
|
|
|
before do
|
2018-10-26 07:43:28 -04:00
|
|
|
project_1 = create(:project)
|
|
|
|
project_2 = create(:project)
|
|
|
|
group_1 = create(:group)
|
|
|
|
group_2 = create(:group)
|
|
|
|
|
2018-10-16 09:18:25 -04:00
|
|
|
create(:active_milestone, title: 'Active Group Milestone', project: project_1)
|
|
|
|
create(:closed_milestone, title: 'Closed Group Milestone', project: project_1)
|
2018-10-22 07:56:08 -04:00
|
|
|
create(:active_milestone, title: 'Active Group Milestone', project: project_2)
|
2018-10-16 09:18:25 -04:00
|
|
|
create(:closed_milestone, title: 'Closed Group Milestone', project: project_2)
|
2018-10-22 07:56:08 -04:00
|
|
|
create(:closed_milestone, title: 'Active Group Milestone', group: group_1)
|
|
|
|
create(:closed_milestone, title: 'Closed Group Milestone', group: group_1)
|
|
|
|
create(:closed_milestone, title: 'Active Group Milestone', group: group_2)
|
|
|
|
create(:closed_milestone, title: 'Closed Group Milestone', group: group_2)
|
2018-10-16 09:18:25 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the quantity of milestones in each possible state' do
|
2018-10-22 07:56:08 -04:00
|
|
|
expected_count = { opened: 5, closed: 6, all: 11 }
|
|
|
|
|
|
|
|
count = described_class.states_count(Project.all, Group.all)
|
2018-10-16 09:18:25 -04:00
|
|
|
expect(count).to eq(expected_count)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the projects do not have milestones' do
|
|
|
|
it 'returns 0 as the quantity of global milestones in each state' do
|
|
|
|
expected_count = { opened: 0, closed: 0, all: 0 }
|
|
|
|
|
|
|
|
count = described_class.states_count([project])
|
|
|
|
|
|
|
|
expect(count).to eq(expected_count)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2019-05-29 07:49:48 -04:00
|
|
|
|
|
|
|
describe '.reference_pattern' do
|
|
|
|
subject { described_class.reference_pattern }
|
|
|
|
|
|
|
|
it { is_expected.to match('gitlab-org/gitlab-ce%123') }
|
|
|
|
it { is_expected.to match('gitlab-org/gitlab-ce%"my-milestone"') }
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.link_reference_pattern' do
|
|
|
|
subject { described_class.link_reference_pattern }
|
|
|
|
|
2019-09-18 10:02:45 -04:00
|
|
|
it { is_expected.to match("#{Gitlab.config.gitlab.url}/gitlab-org/gitlab-foss/milestones/123") }
|
|
|
|
it { is_expected.to match("#{Gitlab.config.gitlab.url}/gitlab-org/gitlab-foss/-/milestones/123") }
|
|
|
|
it { is_expected.not_to match("#{Gitlab.config.gitlab.url}/gitlab-org/gitlab-foss/issues/123") }
|
2019-05-29 07:49:48 -04:00
|
|
|
it { is_expected.not_to match("gitlab-org/gitlab-ce/milestones/123") }
|
|
|
|
end
|
2012-04-08 17:28:58 -04:00
|
|
|
end
|