OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
openproject/spec/models/issue_position_spec.rb

360 lines
13 KiB

require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
describe Issue do
describe 'Story positions' do
def build_issue(options)
FactoryGirl.build(:issue, options.reverse_merge(:fixed_version_id => sprint_1.id,
:priority_id => priority.id,
:project_id => project.id,
:status_id => status.id,
:tracker_id => story_tracker.id))
end
def create_issue(options)
build_issue(options).tap { |i| i.save! }
end
let(:status) { FactoryGirl.create(:issue_status) }
let(:priority) { FactoryGirl.create(:priority_normal) }
let(:project) { FactoryGirl.create(:project) }
let(:story_tracker) { FactoryGirl.create(:tracker, :name => 'Story') }
let(:epic_tracker) { FactoryGirl.create(:tracker, :name => 'Epic') }
let(:task_tracker) { FactoryGirl.create(:tracker, :name => 'Task') }
let(:other_tracker) { FactoryGirl.create(:tracker, :name => 'Feedback') }
let(:sprint_1) { FactoryGirl.create(:version, :project_id => project.id, :name => 'Sprint 1') }
let(:sprint_2) { FactoryGirl.create(:version, :project_id => project.id, :name => 'Sprint 2') }
let(:issue_1) { create_issue(:subject => 'Issue 1', :fixed_version_id => sprint_1.id) }
let(:issue_2) { create_issue(:subject => 'Issue 2', :fixed_version_id => sprint_1.id) }
let(:issue_3) { create_issue(:subject => 'Issue 3', :fixed_version_id => sprint_1.id) }
let(:issue_4) { create_issue(:subject => 'Issue 4', :fixed_version_id => sprint_1.id) }
let(:issue_5) { create_issue(:subject => 'Issue 5', :fixed_version_id => sprint_1.id) }
let(:issue_a) { create_issue(:subject => 'Issue a', :fixed_version_id => sprint_2.id) }
let(:issue_b) { create_issue(:subject => 'Issue b', :fixed_version_id => sprint_2.id) }
let(:issue_c) { create_issue(:subject => 'Issue c', :fixed_version_id => sprint_2.id) }
let(:feedback_1) { create_issue(:subject => 'Feedback 1', :fixed_version_id => sprint_1.id,
:tracker_id => other_tracker.id) }
let(:task_1) { create_issue(:subject => 'Task 1', :fixed_version_id => sprint_1.id,
:tracker_id => task_tracker.id) }
before do
# had problems while writing these specs, that some elements kept creaping
# around between tests. This should be fast enough to not harm anybody
# while adding an additional safety net to make sure, that everything runs
# in isolation.
Issue.delete_all
IssuePriority.delete_all
IssueStatus.delete_all
Project.delete_all
Tracker.delete_all
Version.delete_all
# enable and configure backlogs
project.enabled_module_names = project.enabled_module_names + ["backlogs"]
Setting.plugin_openproject_backlogs = {"story_trackers" => [story_tracker.id, epic_tracker.id],
"task_tracker" => task_tracker.id}
# otherwise the tracker id's from the previous test are still active
Issue.instance_variable_set(:@backlogs_trackers, nil)
project.trackers = [story_tracker, epic_tracker, task_tracker, other_tracker]
sprint_1
sprint_2
# create and order issues
issue_1.move_to_bottom
issue_2.move_to_bottom
issue_3.move_to_bottom
issue_4.move_to_bottom
issue_5.move_to_bottom
issue_a.move_to_bottom
issue_b.move_to_bottom
issue_c.move_to_bottom
end
describe '- Creating an issue in a sprint' do
it 'adds it to the top of the list' do
new_issue = create_issue(:subject => 'Newest Issue', :fixed_version_id => sprint_1.id)
new_issue.should_not be_new_record
new_issue.should be_first
end
it 'reorders the existing issues' do
new_issue = create_issue(:subject => 'Newest Issue', :fixed_version_id => sprint_1.id)
[issue_1, issue_2, issue_3, issue_4, issue_5].each(&:reload).map(&:position).should == [2, 3, 4, 5, 6]
end
end
describe '- Removing an issue from the sprint' do
it 'reorders the remaining issues' do
issue_2.fixed_version = sprint_2
issue_2.save!
sprint_1.fixed_issues.all(:order => 'id').should == [issue_1, issue_3, issue_4, issue_5]
sprint_1.fixed_issues.all(:order => 'id').each(&:reload).map(&:position).should == [1, 2, 3, 4]
end
end
describe '- Adding an issue to a sprint' do
it 'adds it to the top of the list' do
issue_a.fixed_version = sprint_1
issue_a.save!
issue_a.should be_first
end
it 'reorders the existing issues' do
issue_a.fixed_version = sprint_1
issue_a.save!
[issue_1, issue_2, issue_3, issue_4, issue_5].each(&:reload).map(&:position).should == [2, 3, 4, 5, 6]
end
end
13 years ago
describe '- Deleting an issue in a sprint' do
it 'reorders the existing issues' do
issue_3.destroy
[issue_1, issue_2, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4]
end
end
describe '- Changing the tracker' do
describe 'by moving a story to another story tracker' do
it 'keeps all positions in the sprint in tact' do
issue_3.tracker = epic_tracker
issue_3.save!
[issue_1, issue_2, issue_3, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4, 5]
end
end
describe 'by moving a story to a non-backlogs tracker' do
it 'removes it from any list' do
issue_3.tracker = other_tracker
issue_3.save!
issue_3.should_not be_in_list
end
it 'reorders the remaining stories' do
issue_3.tracker = other_tracker
issue_3.save!
[issue_1, issue_2, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4]
end
end
13 years ago
describe 'by moving a story to the task tracker' do
it 'removes it from any list' do
issue_3.tracker = task_tracker
issue_3.save!
issue_3.should_not be_in_list
end
it 'reorders the remaining stories' do
issue_3.tracker = task_tracker
issue_3.save!
[issue_1, issue_2, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4]
end
end
describe 'by moving a task to the story tracker' do
it 'adds it to the top of the list' do
task_1.tracker = story_tracker
task_1.save!
task_1.should be_first
end
it 'reorders the existing stories' do
task_1.tracker = story_tracker
task_1.save!
[task_1, issue_1, issue_2, issue_3, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4, 5, 6]
end
end
describe 'by moving a non-backlogs issue to a story tracker' do
it 'adds it to the top of the list' do
feedback_1.tracker = story_tracker
feedback_1.save!
feedback_1.should be_first
end
it 'reorders the existing stories' do
feedback_1.tracker = story_tracker
feedback_1.save!
[feedback_1, issue_1, issue_2, issue_3, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4, 5, 6]
end
end
end
describe '- Moving issues between projects' do
# N.B.: You cannot move a ticket to another project and change the
# fixed_version at the same time. OTOH chiliproject tries to keep
# the fixed_version if possible (e.g. within project hierarchies with
# shared versions)
let(:project_wo_backlogs) { FactoryGirl.create(:project) }
let(:sub_project_wo_backlogs) { FactoryGirl.create(:project) }
let(:shared_sprint) { FactoryGirl.create(:version,
:project_id => project.id,
:name => 'Shared Sprint',
:sharing => 'descendants') }
let(:version_go_live) { FactoryGirl.create(:version,
:project_id => project_wo_backlogs.id,
:name => 'Go-Live') }
before do
project_wo_backlogs.enabled_module_names = project_wo_backlogs.enabled_module_names - ["backlogs"]
sub_project_wo_backlogs.enabled_module_names = sub_project_wo_backlogs.enabled_module_names - ["backlogs"]
project_wo_backlogs.trackers = [story_tracker, task_tracker, other_tracker]
sub_project_wo_backlogs.trackers = [story_tracker, task_tracker, other_tracker]
sub_project_wo_backlogs.move_to_child_of(project)
shared_sprint
version_go_live
end
describe '- Moving an issue from a project without backlogs to a backlogs_enabled project' do
describe 'if the fixed_version may not be kept' do
let(:issue_i) { create_issue(:subject => 'Issue I',
:fixed_version_id => version_go_live.id,
:project_id => project_wo_backlogs.id) }
before do
issue_i
end
it 'sets the fixed_version_id to nil' do
result = issue_i.move_to_project(project)
result.should be_true
issue_i.fixed_version.should be_nil
end
it 'removes it from any list' do
result = issue_i.move_to_project(project)
result.should be_true
issue_i.should_not be_in_list
end
end
describe 'if the fixed_version may be kept' do
let(:issue_i) { create_issue(:subject => 'Issue I',
:fixed_version_id => shared_sprint.id,
:project_id => sub_project_wo_backlogs.id) }
before do
issue_i
end
it 'keeps the fixed_version_id' do
result = issue_i.move_to_project(project)
result.should be_true
issue_i.fixed_version.should == shared_sprint
end
it 'adds it to the top of the list' do
result = issue_i.move_to_project(project)
result.should be_true
issue_i.should be_first
end
end
end
describe '- Moving an issue away from backlogs_enabled project to a project without backlogs' do
describe 'if the fixed_version may not be kept' do
it 'sets the fixed_version_id to nil' do
result = issue_3.move_to_project(project_wo_backlogs)
result.should be_true
issue_3.fixed_version.should be_nil
end
it 'removes it from any list' do
result = issue_3.move_to_project(sub_project_wo_backlogs)
result.should be_true
issue_3.should_not be_in_list
end
it 'reorders the remaining issues' do
result = issue_3.move_to_project(sub_project_wo_backlogs)
result.should be_true
[issue_1, issue_2, issue_4, issue_5].each(&:reload).map(&:position).should == [1, 2, 3, 4]
end
end
describe 'if the fixed_version may be kept' do
let(:issue_i) { create_issue(:subject => 'Issue I',
:fixed_version_id => shared_sprint.id) }
let(:issue_ii) { create_issue(:subject => 'Issue II',
:fixed_version_id => shared_sprint.id) }
let(:issue_iii) { create_issue(:subject => 'Issue III',
:fixed_version_id => shared_sprint.id) }
before do
issue_i.move_to_bottom
issue_ii.move_to_bottom
issue_iii.move_to_bottom
[issue_i, issue_ii, issue_iii].map(&:position).should == [1, 2, 3]
end
it 'keeps the fixed_version_id' do
result = issue_ii.move_to_project(sub_project_wo_backlogs)
result.should be_true
issue_ii.fixed_version.should == shared_sprint
end
it 'removes it from any list' do
result = issue_ii.move_to_project(sub_project_wo_backlogs)
result.should be_true
issue_ii.should_not be_in_list
end
it 'reorders the remaining issues' do
result = issue_ii.move_to_project(sub_project_wo_backlogs)
result.should be_true
[issue_i, issue_iii].each(&:reload).map(&:position).should == [1, 2]
end
end
end
end
end
end