2012-04-08 17:28:58 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2015-12-09 04:50:51 -05:00
|
|
|
describe Milestone, models: true do
|
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
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to validate_presence_of(:title) }
|
|
|
|
it { is_expected.to validate_presence_of(:project) }
|
2016-11-15 12:48:30 -05:00
|
|
|
|
|
|
|
describe 'start_date' do
|
|
|
|
it 'adds an error when start_date is greated then due_date' do
|
|
|
|
milestone = build(:milestone, start_date: Date.tomorrow, due_date: Date.yesterday)
|
|
|
|
|
|
|
|
expect(milestone).not_to be_valid
|
|
|
|
expect(milestone.errors[:start_date]).to include("Can't be greater than due date")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "Associations" do
|
|
|
|
it { is_expected.to belong_to(:project) }
|
|
|
|
it { is_expected.to have_many(:issues) }
|
2012-04-08 18:01:42 -04:00
|
|
|
end
|
|
|
|
|
2017-01-26 17:44:58 -05:00
|
|
|
let(:project) { create(:empty_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
|
|
|
|
|
2016-02-02 04:44:30 -05:00
|
|
|
describe "unique milestone title per project" do
|
2016-08-01 11:00:44 -04:00
|
|
|
it "does not accept the same title in a project twice" do
|
2016-02-02 04:44:30 -05:00
|
|
|
new_milestone = Milestone.new(project: milestone.project, title: milestone.title)
|
|
|
|
expect(new_milestone).not_to be_valid
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it "accepts the same title in another project" do
|
2017-01-26 17:44:58 -05:00
|
|
|
project = build(:empty_project)
|
2016-02-02 04:44:30 -05:00
|
|
|
new_milestone = Milestone.new(project: project, title: milestone.title)
|
|
|
|
|
|
|
|
expect(new_milestone).to be_valid
|
|
|
|
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
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { expect(milestone.expired?).to be_truthy }
|
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
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { expect(milestone.expired?).to be_falsey }
|
2013-01-03 12:11:14 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-11-15 12:48:30 -05:00
|
|
|
describe '#upcoming?' do
|
|
|
|
it 'returns true' do
|
|
|
|
milestone = build(:milestone, start_date: Time.now + 1.month)
|
|
|
|
expect(milestone.upcoming?).to be_truthy
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false' do
|
|
|
|
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 '#percent_complete' do
|
2013-01-03 12:11:14 -05:00
|
|
|
before do
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(milestone).to receive_messages(
|
2013-01-03 12:11:14 -05:00
|
|
|
closed_items_count: 3,
|
|
|
|
total_items_count: 4
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2016-03-17 16:59:30 -04:00
|
|
|
it { expect(milestone.percent_complete(user)).to eq(75) }
|
2013-01-03 12:11:14 -05:00
|
|
|
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 '#total_items_count' do
|
2013-02-18 08:52:39 -05:00
|
|
|
before do
|
2016-12-12 03:43:56 -05:00
|
|
|
create :closed_issue, milestone: milestone, project: project
|
2015-10-03 02:48:54 -04:00
|
|
|
create :merge_request, milestone: milestone
|
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 total count of issues and merge requests assigned to milestone' do
|
2016-03-17 16:59:30 -04:00
|
|
|
expect(milestone.total_items_count(user)).to eq 2
|
2013-02-18 04:38:29 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-07-11 18:12:31 -04:00
|
|
|
describe '#can_be_closed?' do
|
2013-02-18 08:52:39 -05:00
|
|
|
before do
|
2013-02-18 04:38:29 -05:00
|
|
|
milestone = create :milestone
|
2013-02-18 08:52:39 -05:00
|
|
|
create :closed_issue, milestone: milestone
|
|
|
|
|
2015-10-03 02:48:54 -04:00
|
|
|
create :issue
|
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
|
|
|
|
|
2015-10-15 12:10:35 -04:00
|
|
|
describe '#sort_issues' do
|
|
|
|
let(:milestone) { create(:milestone) }
|
|
|
|
|
|
|
|
let(:issue1) { create(:issue, milestone: milestone, position: 1) }
|
|
|
|
let(:issue2) { create(:issue, milestone: milestone, position: 2) }
|
|
|
|
let(:issue3) { create(:issue, milestone: milestone, position: 3) }
|
|
|
|
let(:issue4) { create(:issue, position: 42) }
|
|
|
|
|
|
|
|
it 'sorts the given issues' do
|
|
|
|
milestone.sort_issues([issue3.id, issue2.id, issue1.id])
|
|
|
|
|
|
|
|
issue1.reload
|
|
|
|
issue2.reload
|
|
|
|
issue3.reload
|
|
|
|
|
|
|
|
expect(issue1.position).to eq(3)
|
|
|
|
expect(issue2.position).to eq(2)
|
|
|
|
expect(issue3.position).to eq(1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'ignores issues not part of the milestone' do
|
|
|
|
milestone.sort_issues([issue3.id, issue2.id, issue1.id, issue4.id])
|
|
|
|
|
|
|
|
issue4.reload
|
|
|
|
|
|
|
|
expect(issue4.position).to eq(42)
|
|
|
|
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
|
|
|
|
expect(described_class.search(milestone.description[0..2])).
|
|
|
|
to eq([milestone])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns milestones with a matching description regardless of the casing' do
|
|
|
|
expect(described_class.search(milestone.description.upcase)).
|
|
|
|
to eq([milestone])
|
|
|
|
end
|
|
|
|
end
|
2016-05-11 12:38:34 -04:00
|
|
|
|
|
|
|
describe '.upcoming_ids_by_projects' do
|
|
|
|
let(:project_1) { create(:empty_project) }
|
|
|
|
let(:project_2) { create(:empty_project) }
|
|
|
|
let(:project_3) { create(:empty_project) }
|
|
|
|
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) }
|
|
|
|
|
2016-05-16 05:23:21 -04:00
|
|
|
# The call to `#try` is because this returns a relation with a Postgres DB,
|
|
|
|
# and an array of IDs with a MySQL DB.
|
|
|
|
let(:milestone_ids) { Milestone.upcoming_ids_by_projects(projects).map { |id| id.try(:id) || id } }
|
2016-05-11 12:38:34 -04:00
|
|
|
|
|
|
|
it 'returns the next upcoming open milestone ID for each project' do
|
|
|
|
expect(milestone_ids).to contain_exactly(current_milestone_project_1.id, current_milestone_project_2.id)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the projects have no open upcoming milestones' do
|
|
|
|
let(:projects) { [project_3] }
|
|
|
|
|
|
|
|
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
|
|
|
|
let(:project) { build(:empty_project, name: 'sample-project') }
|
|
|
|
let(:milestone) { build(:milestone, iid: 1, project: project) }
|
|
|
|
|
|
|
|
it 'returns a String reference to the object' do
|
|
|
|
expect(milestone.to_reference).to eq "%1"
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'supports a cross-project reference' do
|
2017-01-26 17:44:58 -05:00
|
|
|
another_project = build(:empty_project, name: 'another-project', namespace: project.namespace)
|
2016-11-02 19:49:13 -04:00
|
|
|
expect(milestone.to_reference(another_project)).to eq "sample-project%1"
|
|
|
|
end
|
|
|
|
end
|
2017-05-29 05:20:41 -04:00
|
|
|
|
|
|
|
describe '#participants' do
|
|
|
|
let(:project) { build(:empty_project, name: 'sample-project') }
|
|
|
|
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
|
2012-04-08 17:28:58 -04:00
|
|
|
end
|