require 'spec_helper' describe Milestone, models: true do describe "Validation" do before do allow(subject).to receive(:set_iid).and_return(false) end it { is_expected.to validate_presence_of(:title) } it { is_expected.to validate_presence_of(:project) } 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) } end let(:project) { create(:empty_project, :public) } let(:milestone) { create(:milestone, project: project) } let(:issue) { create(:issue, project: project) } let(:user) { create(:user) } describe "#title" do let(:milestone) { create(:milestone, title: "foo & bar -> 2.2") } it "sanitizes title" do expect(milestone.title).to eq("foo & bar -> 2.2") end end describe "unique milestone title per project" do it "does not accept the same title in a project twice" do new_milestone = Milestone.new(project: milestone.project, title: milestone.title) expect(new_milestone).not_to be_valid end it "accepts the same title in another project" do project = build(:empty_project) new_milestone = Milestone.new(project: project, title: milestone.title) expect(new_milestone).to be_valid end end describe "#percent_complete" do it "does not count open issues" do milestone.issues << issue expect(milestone.percent_complete(user)).to eq(0) end it "counts closed issues" do issue.close milestone.issues << issue expect(milestone.percent_complete(user)).to eq(100) end it "recovers from dividing by zero" do expect(milestone.percent_complete(user)).to eq(0) end end describe '#expired?' do context "expired" do before do allow(milestone).to receive(:due_date).and_return(Date.today.prev_year) end it { expect(milestone.expired?).to be_truthy } end context "not expired" do before do allow(milestone).to receive(:due_date).and_return(Date.today.next_year) end it { expect(milestone.expired?).to be_falsey } end end 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 describe '#percent_complete' do before do allow(milestone).to receive_messages( closed_items_count: 3, total_items_count: 4 ) end it { expect(milestone.percent_complete(user)).to eq(75) } end describe '#can_be_closed?' do it { expect(milestone.can_be_closed?).to be_truthy } end describe '#total_items_count' do before do create :closed_issue, milestone: milestone, project: project create :merge_request, milestone: milestone end it 'returns total count of issues and merge requests assigned to milestone' do expect(milestone.total_items_count(user)).to eq 2 end end describe '#can_be_closed?' do before do milestone = create :milestone create :closed_issue, milestone: milestone create :issue end it 'returns true if milestone active and all nested issues closed' do expect(milestone.can_be_closed?).to be_truthy end it 'returns false if milestone active and not all nested issues closed' do issue.milestone = milestone issue.save expect(milestone.can_be_closed?).to be_falsey end end 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 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 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) } # 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 } } 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 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 another_project = build(:empty_project, name: 'another-project', namespace: project.namespace) expect(milestone.to_reference(another_project)).to eq "sample-project%1" end end 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 end