Chris@1494: # Redmine - project management software Chris@1494: # Copyright (C) 2006-2014 Jean-Philippe Lang Chris@1494: # Chris@1494: # This program is free software; you can redistribute it and/or Chris@1494: # modify it under the terms of the GNU General Public License Chris@1494: # as published by the Free Software Foundation; either version 2 Chris@1494: # of the License, or (at your option) any later version. Chris@1494: # Chris@1494: # This program is distributed in the hope that it will be useful, Chris@1494: # but WITHOUT ANY WARRANTY; without even the implied warranty of Chris@1494: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the Chris@1494: # GNU General Public License for more details. Chris@1494: # Chris@1494: # You should have received a copy of the GNU General Public License Chris@1494: # along with this program; if not, write to the Free Software Chris@1494: # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. Chris@1494: Chris@1494: require File.expand_path('../../test_helper', __FILE__) Chris@1494: Chris@1494: class IssueRelationTest < ActiveSupport::TestCase Chris@1494: fixtures :projects, Chris@1494: :users, Chris@1494: :roles, Chris@1494: :members, Chris@1494: :member_roles, Chris@1494: :issues, Chris@1494: :issue_statuses, Chris@1494: :issue_relations, Chris@1494: :enabled_modules, Chris@1494: :enumerations, Chris@1494: :trackers, Chris@1494: :projects_trackers Chris@1494: Chris@1494: include Redmine::I18n Chris@1494: Chris@1494: def test_create Chris@1494: from = Issue.find(1) Chris@1494: to = Issue.find(2) Chris@1494: Chris@1494: relation = IssueRelation.new :issue_from => from, :issue_to => to, Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: assert relation.save Chris@1494: relation.reload Chris@1494: assert_equal IssueRelation::TYPE_PRECEDES, relation.relation_type Chris@1494: assert_equal from, relation.issue_from Chris@1494: assert_equal to, relation.issue_to Chris@1494: end Chris@1494: Chris@1494: def test_create_minimum Chris@1494: relation = IssueRelation.new :issue_from => Issue.find(1), :issue_to => Issue.find(2) Chris@1494: assert relation.save Chris@1494: assert_equal IssueRelation::TYPE_RELATES, relation.relation_type Chris@1494: end Chris@1494: Chris@1494: def test_follows_relation_should_be_reversed Chris@1494: from = Issue.find(1) Chris@1494: to = Issue.find(2) Chris@1494: Chris@1494: relation = IssueRelation.new :issue_from => from, :issue_to => to, Chris@1494: :relation_type => IssueRelation::TYPE_FOLLOWS Chris@1494: assert relation.save Chris@1494: relation.reload Chris@1494: assert_equal IssueRelation::TYPE_PRECEDES, relation.relation_type Chris@1494: assert_equal to, relation.issue_from Chris@1494: assert_equal from, relation.issue_to Chris@1494: end Chris@1494: Chris@1494: def test_follows_relation_should_not_be_reversed_if_validation_fails Chris@1494: from = Issue.find(1) Chris@1494: to = Issue.find(2) Chris@1494: Chris@1494: relation = IssueRelation.new :issue_from => from, :issue_to => to, Chris@1494: :relation_type => IssueRelation::TYPE_FOLLOWS, Chris@1494: :delay => 'xx' Chris@1494: assert !relation.save Chris@1494: assert_equal IssueRelation::TYPE_FOLLOWS, relation.relation_type Chris@1494: assert_equal from, relation.issue_from Chris@1494: assert_equal to, relation.issue_to Chris@1494: end Chris@1494: Chris@1494: def test_relation_type_for Chris@1494: from = Issue.find(1) Chris@1494: to = Issue.find(2) Chris@1494: Chris@1494: relation = IssueRelation.new :issue_from => from, :issue_to => to, Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: assert_equal IssueRelation::TYPE_PRECEDES, relation.relation_type_for(from) Chris@1494: assert_equal IssueRelation::TYPE_FOLLOWS, relation.relation_type_for(to) Chris@1494: end Chris@1494: Chris@1494: def test_set_issue_to_dates_without_issue_to Chris@1494: r = IssueRelation.new(:issue_from => Issue.new(:start_date => Date.today), Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES, Chris@1494: :delay => 1) Chris@1494: assert_nil r.set_issue_to_dates Chris@1494: end Chris@1494: Chris@1494: def test_set_issue_to_dates_without_issues Chris@1494: r = IssueRelation.new(:relation_type => IssueRelation::TYPE_PRECEDES, :delay => 1) Chris@1494: assert_nil r.set_issue_to_dates Chris@1494: end Chris@1494: Chris@1494: def test_validates_circular_dependency Chris@1494: IssueRelation.delete_all Chris@1494: assert IssueRelation.create!( Chris@1494: :issue_from => Issue.find(1), :issue_to => Issue.find(2), Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: ) Chris@1494: assert IssueRelation.create!( Chris@1494: :issue_from => Issue.find(2), :issue_to => Issue.find(3), Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: ) Chris@1494: r = IssueRelation.new( Chris@1494: :issue_from => Issue.find(3), :issue_to => Issue.find(1), Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: ) Chris@1494: assert !r.save Chris@1494: assert_not_equal [], r.errors[:base] Chris@1494: end Chris@1494: Chris@1494: def test_validates_circular_dependency_of_subtask Chris@1494: set_language_if_valid 'en' Chris@1494: issue1 = Issue.generate! Chris@1494: issue2 = Issue.generate! Chris@1494: IssueRelation.create!( Chris@1494: :issue_from => issue1, :issue_to => issue2, Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: ) Chris@1494: child = Issue.generate!(:parent_issue_id => issue2.id) Chris@1494: issue1.reload Chris@1494: child.reload Chris@1494: Chris@1494: r = IssueRelation.new( Chris@1494: :issue_from => child, :issue_to => issue1, Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: ) Chris@1494: assert !r.save Chris@1494: assert_include 'This relation would create a circular dependency', r.errors.full_messages Chris@1494: end Chris@1494: Chris@1494: def test_subtasks_should_allow_precedes_relation Chris@1494: parent = Issue.generate! Chris@1494: child1 = Issue.generate!(:parent_issue_id => parent.id) Chris@1494: child2 = Issue.generate!(:parent_issue_id => parent.id) Chris@1494: Chris@1494: r = IssueRelation.new( Chris@1494: :issue_from => child1, :issue_to => child2, Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES Chris@1494: ) Chris@1494: assert r.valid? Chris@1494: assert r.save Chris@1494: end Chris@1494: Chris@1494: def test_validates_circular_dependency_on_reverse_relations Chris@1494: IssueRelation.delete_all Chris@1494: assert IssueRelation.create!( Chris@1494: :issue_from => Issue.find(1), :issue_to => Issue.find(3), Chris@1494: :relation_type => IssueRelation::TYPE_BLOCKS Chris@1494: ) Chris@1494: assert IssueRelation.create!( Chris@1494: :issue_from => Issue.find(1), :issue_to => Issue.find(2), Chris@1494: :relation_type => IssueRelation::TYPE_BLOCKED Chris@1494: ) Chris@1494: r = IssueRelation.new( Chris@1494: :issue_from => Issue.find(2), :issue_to => Issue.find(1), Chris@1494: :relation_type => IssueRelation::TYPE_BLOCKED Chris@1494: ) Chris@1494: assert !r.save Chris@1494: assert_not_equal [], r.errors[:base] Chris@1494: end Chris@1494: Chris@1494: def test_create_should_make_journal_entry Chris@1494: from = Issue.find(1) Chris@1494: to = Issue.find(2) Chris@1494: from_journals = from.journals.size Chris@1494: to_journals = to.journals.size Chris@1494: relation = IssueRelation.new(:issue_from => from, :issue_to => to, Chris@1494: :relation_type => IssueRelation::TYPE_PRECEDES) Chris@1494: assert relation.save Chris@1494: from.reload Chris@1494: to.reload Chris@1494: relation.reload Chris@1494: assert_equal from.journals.size, (from_journals + 1) Chris@1494: assert_equal to.journals.size, (to_journals + 1) Chris@1494: assert_equal 'relation', from.journals.last.details.last.property Chris@1494: assert_equal 'label_precedes', from.journals.last.details.last.prop_key Chris@1494: assert_equal '2', from.journals.last.details.last.value Chris@1494: assert_nil from.journals.last.details.last.old_value Chris@1494: assert_equal 'relation', to.journals.last.details.last.property Chris@1494: assert_equal 'label_follows', to.journals.last.details.last.prop_key Chris@1494: assert_equal '1', to.journals.last.details.last.value Chris@1494: assert_nil to.journals.last.details.last.old_value Chris@1494: end Chris@1494: Chris@1494: def test_delete_should_make_journal_entry Chris@1494: relation = IssueRelation.find(1) Chris@1494: id = relation.id Chris@1494: from = relation.issue_from Chris@1494: to = relation.issue_to Chris@1494: from_journals = from.journals.size Chris@1494: to_journals = to.journals.size Chris@1494: assert relation.destroy Chris@1494: from.reload Chris@1494: to.reload Chris@1494: assert_equal from.journals.size, (from_journals + 1) Chris@1494: assert_equal to.journals.size, (to_journals + 1) Chris@1494: assert_equal 'relation', from.journals.last.details.last.property Chris@1494: assert_equal 'label_blocks', from.journals.last.details.last.prop_key Chris@1494: assert_equal '9', from.journals.last.details.last.old_value Chris@1494: assert_nil from.journals.last.details.last.value Chris@1494: assert_equal 'relation', to.journals.last.details.last.property Chris@1494: assert_equal 'label_blocked_by', to.journals.last.details.last.prop_key Chris@1494: assert_equal '10', to.journals.last.details.last.old_value Chris@1494: assert_nil to.journals.last.details.last.value Chris@1494: end Chris@1494: end