comparison test/unit/project_test.rb @ 1526:404aa68d4227

Merge from live branch
author Chris Cannam
date Thu, 11 Sep 2014 12:46:20 +0100
parents dffacf8a6908
children
comparison
equal deleted inserted replaced
1493:a5f2bdf3b486 1526:404aa68d4227
1 # Redmine - project management software 1 # Redmine - project management software
2 # Copyright (C) 2006-2012 Jean-Philippe Lang 2 # Copyright (C) 2006-2014 Jean-Philippe Lang
3 # 3 #
4 # This program is free software; you can redistribute it and/or 4 # This program is free software; you can redistribute it and/or
5 # modify it under the terms of the GNU General Public License 5 # modify it under the terms of the GNU General Public License
6 # as published by the Free Software Foundation; either version 2 6 # as published by the Free Software Foundation; either version 2
7 # of the License, or (at your option) any later version. 7 # of the License, or (at your option) any later version.
28 :custom_values, 28 :custom_values,
29 :roles, 29 :roles,
30 :member_roles, 30 :member_roles,
31 :members, 31 :members,
32 :enabled_modules, 32 :enabled_modules,
33 :workflows,
34 :versions, 33 :versions,
35 :wikis, :wiki_pages, :wiki_contents, :wiki_content_versions, 34 :wikis, :wiki_pages, :wiki_contents, :wiki_content_versions,
36 :groups_users, 35 :groups_users,
37 :boards, :messages, 36 :boards, :messages,
38 :repositories, 37 :repositories,
73 end 72 end
74 73
75 with_settings :default_projects_modules => ['issue_tracking', 'repository'] do 74 with_settings :default_projects_modules => ['issue_tracking', 'repository'] do
76 assert_equal ['issue_tracking', 'repository'], Project.new.enabled_module_names 75 assert_equal ['issue_tracking', 'repository'], Project.new.enabled_module_names
77 end 76 end
78 77 end
79 assert_equal Tracker.all.sort, Project.new.trackers.sort 78
80 assert_equal Tracker.find(1, 3).sort, Project.new(:tracker_ids => [1, 3]).trackers.sort 79 def test_default_trackers_should_match_default_tracker_ids_setting
80 with_settings :default_projects_tracker_ids => ['1', '3'] do
81 assert_equal Tracker.find(1, 3).sort, Project.new.trackers.sort
82 end
83 end
84
85 def test_default_trackers_should_be_all_trackers_with_blank_setting
86 with_settings :default_projects_tracker_ids => nil do
87 assert_equal Tracker.all.sort, Project.new.trackers.sort
88 end
89 end
90
91 def test_default_trackers_should_be_empty_with_empty_setting
92 with_settings :default_projects_tracker_ids => [] do
93 assert_equal [], Project.new.trackers
94 end
95 end
96
97 def test_default_trackers_should_not_replace_initialized_trackers
98 with_settings :default_projects_tracker_ids => ['1', '3'] do
99 assert_equal Tracker.find(1, 2).sort, Project.new(:tracker_ids => [1, 2]).trackers.sort
100 end
81 end 101 end
82 102
83 def test_update 103 def test_update
84 assert_equal "eCookbook", @ecookbook.name 104 assert_equal "eCookbook", @ecookbook.name
85 @ecookbook.name = "eCook" 105 @ecookbook.name = "eCook"
111 def test_identifier_should_not_be_frozen_for_a_new_project 131 def test_identifier_should_not_be_frozen_for_a_new_project
112 assert_equal false, Project.new.identifier_frozen? 132 assert_equal false, Project.new.identifier_frozen?
113 end 133 end
114 134
115 def test_identifier_should_not_be_frozen_for_a_saved_project_with_blank_identifier 135 def test_identifier_should_not_be_frozen_for_a_saved_project_with_blank_identifier
116 Project.update_all(["identifier = ''"], "id = 1") 136 Project.where(:id => 1).update_all(["identifier = ''"])
117
118 assert_equal false, Project.find(1).identifier_frozen? 137 assert_equal false, Project.find(1).identifier_frozen?
119 end 138 end
120 139
121 def test_identifier_should_be_frozen_for_a_saved_project_with_valid_identifier 140 def test_identifier_should_be_frozen_for_a_saved_project_with_valid_identifier
122 assert_equal true, Project.find(1).identifier_frozen? 141 assert_equal true, Project.find(1).identifier_frozen?
153 172
154 def test_archive_should_fail_if_versions_are_used_by_non_descendant_projects 173 def test_archive_should_fail_if_versions_are_used_by_non_descendant_projects
155 # Assign an issue of a project to a version of a child project 174 # Assign an issue of a project to a version of a child project
156 Issue.find(4).update_attribute :fixed_version_id, 4 175 Issue.find(4).update_attribute :fixed_version_id, 4
157 176
158 assert_no_difference "Project.count(:all, :conditions => 'status = #{Project::STATUS_ARCHIVED}')" do 177 assert_no_difference "Project.where(:status => Project::STATUS_ARCHIVED).count" do
159 assert_equal false, @ecookbook.archive 178 assert_equal false, @ecookbook.archive
160 end 179 end
161 @ecookbook.reload 180 @ecookbook.reload
162 assert @ecookbook.active? 181 assert @ecookbook.active?
163 end 182 end
181 200
182 def test_destroy 201 def test_destroy
183 # 2 active members 202 # 2 active members
184 assert_equal 2, @ecookbook.members.size 203 assert_equal 2, @ecookbook.members.size
185 # and 1 is locked 204 # and 1 is locked
186 assert_equal 3, Member.find(:all, :conditions => ['project_id = ?', @ecookbook.id]).size 205 assert_equal 3, Member.where(:project_id => @ecookbook.id).count
187 # some boards 206 # some boards
188 assert @ecookbook.boards.any? 207 assert @ecookbook.boards.any?
189 208
190 @ecookbook.destroy 209 @ecookbook.destroy
191 # make sure that the project non longer exists 210 # make sure that the project non longer exists
192 assert_raise(ActiveRecord::RecordNotFound) { Project.find(@ecookbook.id) } 211 assert_raise(ActiveRecord::RecordNotFound) { Project.find(@ecookbook.id) }
193 # make sure related data was removed 212 # make sure related data was removed
194 assert_nil Member.first(:conditions => {:project_id => @ecookbook.id}) 213 assert_nil Member.where(:project_id => @ecookbook.id).first
195 assert_nil Board.first(:conditions => {:project_id => @ecookbook.id}) 214 assert_nil Board.where(:project_id => @ecookbook.id).first
196 assert_nil Issue.first(:conditions => {:project_id => @ecookbook.id}) 215 assert_nil Issue.where(:project_id => @ecookbook.id).first
197 end 216 end
198 217
199 def test_destroy_should_destroy_subtasks 218 def test_destroy_should_destroy_subtasks
200 issues = (0..2).to_a.map {Issue.create!(:project_id => 1, :tracker_id => 1, :author_id => 1, :subject => 'test')} 219 issues = (0..2).to_a.map {Issue.create!(:project_id => 1, :tracker_id => 1, :author_id => 1, :subject => 'test')}
201 issues[0].update_attribute :parent_issue_id, issues[1].id 220 issues[0].update_attribute :parent_issue_id, issues[1].id
203 assert_equal 2, issues[1].children.count 222 assert_equal 2, issues[1].children.count
204 223
205 assert_nothing_raised do 224 assert_nothing_raised do
206 Project.find(1).destroy 225 Project.find(1).destroy
207 end 226 end
208 assert Issue.find_all_by_id(issues.map(&:id)).empty? 227 assert_equal 0, Issue.where(:id => issues.map(&:id)).count
209 end 228 end
210 229
211 def test_destroying_root_projects_should_clear_data 230 def test_destroying_root_projects_should_clear_data
212 Project.roots.each do |root| 231 Project.roots.each do |root|
213 root.destroy 232 root.destroy
224 assert_equal 0, IssueCategory.count 243 assert_equal 0, IssueCategory.count
225 assert_equal 0, IssueRelation.count 244 assert_equal 0, IssueRelation.count
226 assert_equal 0, Board.count 245 assert_equal 0, Board.count
227 assert_equal 0, Message.count 246 assert_equal 0, Message.count
228 assert_equal 0, News.count 247 assert_equal 0, News.count
229 assert_equal 0, Query.count(:conditions => "project_id IS NOT NULL") 248 assert_equal 0, Query.where("project_id IS NOT NULL").count
230 assert_equal 0, Repository.count 249 assert_equal 0, Repository.count
231 assert_equal 0, Changeset.count 250 assert_equal 0, Changeset.count
232 assert_equal 0, Change.count 251 assert_equal 0, Change.count
233 assert_equal 0, Comment.count 252 assert_equal 0, Comment.count
234 assert_equal 0, TimeEntry.count 253 assert_equal 0, TimeEntry.count
236 assert_equal 0, Watcher.count 255 assert_equal 0, Watcher.count
237 assert_equal 0, Wiki.count 256 assert_equal 0, Wiki.count
238 assert_equal 0, WikiPage.count 257 assert_equal 0, WikiPage.count
239 assert_equal 0, WikiContent.count 258 assert_equal 0, WikiContent.count
240 assert_equal 0, WikiContent::Version.count 259 assert_equal 0, WikiContent::Version.count
241 assert_equal 0, Project.connection.select_all("SELECT * FROM projects_trackers").size 260 assert_equal 0, Project.connection.select_all("SELECT * FROM projects_trackers").count
242 assert_equal 0, Project.connection.select_all("SELECT * FROM custom_fields_projects").size 261 assert_equal 0, Project.connection.select_all("SELECT * FROM custom_fields_projects").count
243 assert_equal 0, CustomValue.count(:conditions => {:customized_type => ['Project', 'Issue', 'TimeEntry', 'Version']}) 262 assert_equal 0, CustomValue.where(:customized_type => ['Project', 'Issue', 'TimeEntry', 'Version']).count
263 end
264
265 def test_destroy_should_delete_time_entries_custom_values
266 project = Project.generate!
267 time_entry = TimeEntry.generate!(:project => project, :custom_field_values => {10 => '1'})
268
269 assert_difference 'CustomValue.where(:customized_type => "TimeEntry").count', -1 do
270 assert project.destroy
271 end
244 end 272 end
245 273
246 def test_move_an_orphan_project_to_a_root_project 274 def test_move_an_orphan_project_to_a_root_project
247 sub = Project.find(2) 275 sub = Project.find(2)
248 sub.set_parent! @ecookbook 276 sub.set_parent! @ecookbook
433 parent.children.each(&:archive) 461 parent.children.each(&:archive)
434 462
435 assert_equal [1,2], parent.rolled_up_trackers.collect(&:id) 463 assert_equal [1,2], parent.rolled_up_trackers.collect(&:id)
436 end 464 end
437 465
438 context "#rolled_up_versions" do 466 test "#rolled_up_trackers should ignore projects with issue_tracking module disabled" do
439 setup do 467 parent = Project.generate!
440 @project = Project.generate! 468 parent.trackers = Tracker.find([1, 2])
441 @parent_version_1 = Version.generate!(:project => @project) 469 child = Project.generate_with_parent!(parent)
442 @parent_version_2 = Version.generate!(:project => @project) 470 child.trackers = Tracker.find([2, 3])
443 end 471
444 472 assert_equal [1, 2, 3], parent.rolled_up_trackers.collect(&:id).sort
445 should "include the versions for the current project" do 473
446 assert_same_elements [@parent_version_1, @parent_version_2], @project.rolled_up_versions 474 assert child.disable_module!(:issue_tracking)
447 end 475 parent.reload
448 476 assert_equal [1, 2], parent.rolled_up_trackers.collect(&:id).sort
449 should "include versions for a subproject" do 477 end
450 @subproject = Project.generate! 478
451 @subproject.set_parent!(@project) 479 test "#rolled_up_versions should include the versions for the current project" do
452 @subproject_version = Version.generate!(:project => @subproject) 480 project = Project.generate!
453 481 parent_version_1 = Version.generate!(:project => project)
454 assert_same_elements [ 482 parent_version_2 = Version.generate!(:project => project)
455 @parent_version_1, 483 assert_same_elements [parent_version_1, parent_version_2], project.rolled_up_versions
456 @parent_version_2, 484 end
457 @subproject_version 485
458 ], @project.rolled_up_versions 486 test "#rolled_up_versions should include versions for a subproject" do
459 end 487 project = Project.generate!
460 488 parent_version_1 = Version.generate!(:project => project)
461 should "include versions for a sub-subproject" do 489 parent_version_2 = Version.generate!(:project => project)
462 @subproject = Project.generate! 490 subproject = Project.generate_with_parent!(project)
463 @subproject.set_parent!(@project) 491 subproject_version = Version.generate!(:project => subproject)
464 @sub_subproject = Project.generate! 492
465 @sub_subproject.set_parent!(@subproject) 493 assert_same_elements [
466 @sub_subproject_version = Version.generate!(:project => @sub_subproject) 494 parent_version_1,
467 495 parent_version_2,
468 @project.reload 496 subproject_version
469 497 ], project.rolled_up_versions
470 assert_same_elements [ 498 end
471 @parent_version_1, 499
472 @parent_version_2, 500 test "#rolled_up_versions should include versions for a sub-subproject" do
473 @sub_subproject_version 501 project = Project.generate!
474 ], @project.rolled_up_versions 502 parent_version_1 = Version.generate!(:project => project)
475 end 503 parent_version_2 = Version.generate!(:project => project)
476 504 subproject = Project.generate_with_parent!(project)
477 should "only check active projects" do 505 sub_subproject = Project.generate_with_parent!(subproject)
478 @subproject = Project.generate! 506 sub_subproject_version = Version.generate!(:project => sub_subproject)
479 @subproject.set_parent!(@project) 507 project.reload
480 @subproject_version = Version.generate!(:project => @subproject) 508
481 assert @subproject.archive 509 assert_same_elements [
482 510 parent_version_1,
483 @project.reload 511 parent_version_2,
484 512 sub_subproject_version
485 assert !@subproject.active? 513 ], project.rolled_up_versions
486 assert_same_elements [@parent_version_1, @parent_version_2], @project.rolled_up_versions 514 end
487 end 515
516 test "#rolled_up_versions should only check active projects" do
517 project = Project.generate!
518 parent_version_1 = Version.generate!(:project => project)
519 parent_version_2 = Version.generate!(:project => project)
520 subproject = Project.generate_with_parent!(project)
521 subproject_version = Version.generate!(:project => subproject)
522 assert subproject.archive
523 project.reload
524
525 assert !subproject.active?
526 assert_same_elements [parent_version_1, parent_version_2], project.rolled_up_versions
488 end 527 end
489 528
490 def test_shared_versions_none_sharing 529 def test_shared_versions_none_sharing
491 p = Project.find(5) 530 p = Project.find(5)
492 v = Version.create!(:name => 'none_sharing', :project => p, :sharing => 'none') 531 v = Version.create!(:name => 'none_sharing', :project => p, :sharing => 'none')
543 private_child = parent.children.find(5) 582 private_child = parent.children.find(5)
544 583
545 assert_equal [1,2,3], parent.version_ids.sort 584 assert_equal [1,2,3], parent.version_ids.sort
546 assert_equal [4], child.version_ids 585 assert_equal [4], child.version_ids
547 assert_equal [6], private_child.version_ids 586 assert_equal [6], private_child.version_ids
548 assert_equal [7], Version.find_all_by_sharing('system').collect(&:id) 587 assert_equal [7], Version.where(:sharing => 'system').collect(&:id)
549 588
550 assert_equal 6, parent.shared_versions.size 589 assert_equal 6, parent.shared_versions.size
551 parent.shared_versions.each do |version| 590 parent.shared_versions.each do |version|
552 assert_kind_of Version, version 591 assert_kind_of Version, version
553 end 592 end
609 project.enabled_module_names = %w(issue_tracking news) 648 project.enabled_module_names = %w(issue_tracking news)
610 assert_equal %w(issue_tracking news), project.enabled_module_names.sort 649 assert_equal %w(issue_tracking news), project.enabled_module_names.sort
611 end 650 end
612 end 651 end
613 652
614 context "enabled_modules" do 653 test "enabled_modules should define module by names and preserve ids" do
615 setup do 654 @project = Project.find(1)
616 @project = Project.find(1) 655 # Remove one module
617 end 656 modules = @project.enabled_modules.slice(0..-2)
618 657 assert modules.any?
619 should "define module by names and preserve ids" do 658 assert_difference 'EnabledModule.count', -1 do
620 # Remove one module 659 @project.enabled_module_names = modules.collect(&:name)
621 modules = @project.enabled_modules.slice(0..-2) 660 end
622 assert modules.any? 661 @project.reload
623 assert_difference 'EnabledModule.count', -1 do 662 # Ids should be preserved
624 @project.enabled_module_names = modules.collect(&:name) 663 assert_equal @project.enabled_module_ids.sort, modules.collect(&:id).sort
625 end 664 end
626 @project.reload 665
627 # Ids should be preserved 666 test "enabled_modules should enable a module" do
628 assert_equal @project.enabled_module_ids.sort, modules.collect(&:id).sort 667 @project = Project.find(1)
629 end 668 @project.enabled_module_names = []
630 669 @project.reload
631 should "enable a module" do 670 assert_equal [], @project.enabled_module_names
632 @project.enabled_module_names = [] 671 #with string
633 @project.reload 672 @project.enable_module!("issue_tracking")
634 assert_equal [], @project.enabled_module_names 673 assert_equal ["issue_tracking"], @project.enabled_module_names
635 #with string 674 #with symbol
636 @project.enable_module!("issue_tracking") 675 @project.enable_module!(:gantt)
637 assert_equal ["issue_tracking"], @project.enabled_module_names 676 assert_equal ["issue_tracking", "gantt"], @project.enabled_module_names
638 #with symbol 677 #don't add a module twice
639 @project.enable_module!(:gantt) 678 @project.enable_module!("issue_tracking")
640 assert_equal ["issue_tracking", "gantt"], @project.enabled_module_names 679 assert_equal ["issue_tracking", "gantt"], @project.enabled_module_names
641 #don't add a module twice 680 end
642 @project.enable_module!("issue_tracking") 681
643 assert_equal ["issue_tracking", "gantt"], @project.enabled_module_names 682 test "enabled_modules should disable a module" do
644 end 683 @project = Project.find(1)
645 684 #with string
646 should "disable a module" do 685 assert @project.enabled_module_names.include?("issue_tracking")
647 #with string 686 @project.disable_module!("issue_tracking")
648 assert @project.enabled_module_names.include?("issue_tracking") 687 assert ! @project.reload.enabled_module_names.include?("issue_tracking")
649 @project.disable_module!("issue_tracking") 688 #with symbol
650 assert ! @project.reload.enabled_module_names.include?("issue_tracking") 689 assert @project.enabled_module_names.include?("gantt")
651 #with symbol 690 @project.disable_module!(:gantt)
652 assert @project.enabled_module_names.include?("gantt") 691 assert ! @project.reload.enabled_module_names.include?("gantt")
653 @project.disable_module!(:gantt) 692 #with EnabledModule object
654 assert ! @project.reload.enabled_module_names.include?("gantt") 693 first_module = @project.enabled_modules.first
655 #with EnabledModule object 694 @project.disable_module!(first_module)
656 first_module = @project.enabled_modules.first 695 assert ! @project.reload.enabled_module_names.include?(first_module.name)
657 @project.disable_module!(first_module)
658 assert ! @project.reload.enabled_module_names.include?(first_module.name)
659 end
660 end 696 end
661 697
662 def test_enabled_module_names_should_not_recreate_enabled_modules 698 def test_enabled_module_names_should_not_recreate_enabled_modules
663 project = Project.find(1) 699 project = Project.find(1)
664 # Remove one module 700 # Remove one module
691 assert_equal 1, copied_project.status 727 assert_equal 1, copied_project.status
692 end 728 end
693 729
694 def test_activities_should_use_the_system_activities 730 def test_activities_should_use_the_system_activities
695 project = Project.find(1) 731 project = Project.find(1)
696 assert_equal project.activities, TimeEntryActivity.find(:all, :conditions => {:active => true} ) 732 assert_equal project.activities, TimeEntryActivity.where(:active => true).all
733 assert_kind_of ActiveRecord::Relation, project.activities
697 end 734 end
698 735
699 736
700 def test_activities_should_use_the_project_specific_activities 737 def test_activities_should_use_the_project_specific_activities
701 project = Project.find(1) 738 project = Project.find(1)
702 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project}) 739 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project})
703 assert overridden_activity.save! 740 assert overridden_activity.save!
704 741
705 assert project.activities.include?(overridden_activity), "Project specific Activity not found" 742 assert project.activities.include?(overridden_activity), "Project specific Activity not found"
743 assert_kind_of ActiveRecord::Relation, project.activities
706 end 744 end
707 745
708 def test_activities_should_not_include_the_inactive_project_specific_activities 746 def test_activities_should_not_include_the_inactive_project_specific_activities
709 project = Project.find(1) 747 project = Project.find(1)
710 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project, :parent => TimeEntryActivity.find(:first), :active => false}) 748 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project, :parent => TimeEntryActivity.first, :active => false})
711 assert overridden_activity.save! 749 assert overridden_activity.save!
712 750
713 assert !project.activities.include?(overridden_activity), "Inactive Project specific Activity found" 751 assert !project.activities.include?(overridden_activity), "Inactive Project specific Activity found"
714 end 752 end
715 753
720 758
721 assert !project.activities.include?(overridden_activity), "Project specific Activity found on a different project" 759 assert !project.activities.include?(overridden_activity), "Project specific Activity found on a different project"
722 end 760 end
723 761
724 def test_activities_should_handle_nils 762 def test_activities_should_handle_nils
725 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => Project.find(1), :parent => TimeEntryActivity.find(:first)}) 763 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => Project.find(1), :parent => TimeEntryActivity.first})
726 TimeEntryActivity.delete_all 764 TimeEntryActivity.delete_all
727 765
728 # No activities 766 # No activities
729 project = Project.find(1) 767 project = Project.find(1)
730 assert project.activities.empty? 768 assert project.activities.empty?
735 assert_equal [overridden_activity], project.activities 773 assert_equal [overridden_activity], project.activities
736 end 774 end
737 775
738 def test_activities_should_override_system_activities_with_project_activities 776 def test_activities_should_override_system_activities_with_project_activities
739 project = Project.find(1) 777 project = Project.find(1)
740 parent_activity = TimeEntryActivity.find(:first) 778 parent_activity = TimeEntryActivity.first
741 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project, :parent => parent_activity}) 779 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project, :parent => parent_activity})
742 assert overridden_activity.save! 780 assert overridden_activity.save!
743 781
744 assert project.activities.include?(overridden_activity), "Project specific Activity not found" 782 assert project.activities.include?(overridden_activity), "Project specific Activity not found"
745 assert !project.activities.include?(parent_activity), "System Activity found when it should have been overridden" 783 assert !project.activities.include?(parent_activity), "System Activity found when it should have been overridden"
746 end 784 end
747 785
748 def test_activities_should_include_inactive_activities_if_specified 786 def test_activities_should_include_inactive_activities_if_specified
749 project = Project.find(1) 787 project = Project.find(1)
750 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project, :parent => TimeEntryActivity.find(:first), :active => false}) 788 overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project, :parent => TimeEntryActivity.first, :active => false})
751 assert overridden_activity.save! 789 assert overridden_activity.save!
752 790
753 assert project.activities(true).include?(overridden_activity), "Inactive Project specific Activity not found" 791 assert project.activities(true).include?(overridden_activity), "Inactive Project specific Activity not found"
754 end 792 end
755 793
773 project.reload 811 project.reload
774 assert_nil project.versions.detect {|v| v.completed? && v.status != 'closed'} 812 assert_nil project.versions.detect {|v| v.completed? && v.status != 'closed'}
775 assert_not_nil project.versions.detect {|v| !v.completed? && v.status == 'open'} 813 assert_not_nil project.versions.detect {|v| !v.completed? && v.status == 'open'}
776 end 814 end
777 815
778 context "Project#copy" do 816 test "#start_date should be nil if there are no issues on the project" do
779 setup do 817 project = Project.generate!
780 ProjectCustomField.destroy_all # Custom values are a mess to isolate in tests 818 assert_nil project.start_date
781 Project.destroy_all :identifier => "copy-test" 819 end
782 @source_project = Project.find(2) 820
783 @project = Project.new(:name => 'Copy Test', :identifier => 'copy-test') 821 test "#start_date should be nil when issues have no start date" do
784 @project.trackers = @source_project.trackers 822 project = Project.generate!
785 @project.enabled_module_names = @source_project.enabled_modules.collect(&:name) 823 project.trackers << Tracker.generate!
786 end 824 early = 7.days.ago.to_date
787 825 Issue.generate!(:project => project, :start_date => nil)
788 should "copy issues" do 826
789 @source_project.issues << Issue.generate!(:status => IssueStatus.find_by_name('Closed'), 827 assert_nil project.start_date
790 :subject => "copy issue status", 828 end
791 :tracker_id => 1, 829
792 :assigned_to_id => 2, 830 test "#start_date should be the earliest start date of it's issues" do
793 :project_id => @source_project.id) 831 project = Project.generate!
794 assert @project.valid? 832 project.trackers << Tracker.generate!
795 assert @project.issues.empty? 833 early = 7.days.ago.to_date
796 assert @project.copy(@source_project) 834 Issue.generate!(:project => project, :start_date => Date.today)
797 835 Issue.generate!(:project => project, :start_date => early)
798 assert_equal @source_project.issues.size, @project.issues.size 836
799 @project.issues.each do |issue| 837 assert_equal early, project.start_date
800 assert issue.valid? 838 end
801 assert ! issue.assigned_to.blank? 839
802 assert_equal @project, issue.project 840 test "#due_date should be nil if there are no issues on the project" do
803 end 841 project = Project.generate!
804 842 assert_nil project.due_date
805 copied_issue = @project.issues.first(:conditions => {:subject => "copy issue status"}) 843 end
806 assert copied_issue 844
807 assert copied_issue.status 845 test "#due_date should be nil if there are no issues with due dates" do
808 assert_equal "Closed", copied_issue.status.name 846 project = Project.generate!
809 end 847 project.trackers << Tracker.generate!
810 848 Issue.generate!(:project => project, :due_date => nil)
811 should "copy issues assigned to a locked version" do 849
812 User.current = User.find(1) 850 assert_nil project.due_date
813 assigned_version = Version.generate!(:name => "Assigned Issues") 851 end
814 @source_project.versions << assigned_version 852
815 Issue.generate!(:project => @source_project, 853 test "#due_date should be the latest due date of it's issues" do
816 :fixed_version_id => assigned_version.id, 854 project = Project.generate!
817 :subject => "copy issues assigned to a locked version") 855 project.trackers << Tracker.generate!
818 assigned_version.update_attribute :status, 'locked' 856 future = 7.days.from_now.to_date
819 857 Issue.generate!(:project => project, :due_date => future)
820 assert @project.copy(@source_project) 858 Issue.generate!(:project => project, :due_date => Date.today)
821 @project.reload 859
822 copied_issue = @project.issues.first(:conditions => {:subject => "copy issues assigned to a locked version"}) 860 assert_equal future, project.due_date
823 861 end
824 assert copied_issue 862
825 assert copied_issue.fixed_version 863 test "#due_date should be the latest due date of it's versions" do
826 assert_equal "Assigned Issues", copied_issue.fixed_version.name # Same name 864 project = Project.generate!
827 assert_equal 'locked', copied_issue.fixed_version.status 865 future = 7.days.from_now.to_date
828 end 866 project.versions << Version.generate!(:effective_date => future)
829 867 project.versions << Version.generate!(:effective_date => Date.today)
830 should "change the new issues to use the copied version" do 868
831 User.current = User.find(1) 869 assert_equal future, project.due_date
832 assigned_version = Version.generate!(:name => "Assigned Issues", :status => 'open') 870 end
833 @source_project.versions << assigned_version 871
834 assert_equal 3, @source_project.versions.size 872 test "#due_date should pick the latest date from it's issues and versions" do
835 Issue.generate!(:project => @source_project, 873 project = Project.generate!
836 :fixed_version_id => assigned_version.id, 874 project.trackers << Tracker.generate!
837 :subject => "change the new issues to use the copied version") 875 future = 7.days.from_now.to_date
838 876 far_future = 14.days.from_now.to_date
839 assert @project.copy(@source_project) 877 Issue.generate!(:project => project, :due_date => far_future)
840 @project.reload 878 project.versions << Version.generate!(:effective_date => future)
841 copied_issue = @project.issues.first(:conditions => {:subject => "change the new issues to use the copied version"}) 879
842 880 assert_equal far_future, project.due_date
843 assert copied_issue 881 end
844 assert copied_issue.fixed_version 882
845 assert_equal "Assigned Issues", copied_issue.fixed_version.name # Same name 883 test "#completed_percent with no versions should be 100" do
846 assert_not_equal assigned_version.id, copied_issue.fixed_version.id # Different record 884 project = Project.generate!
847 end 885 assert_equal 100, project.completed_percent
848 886 end
849 should "keep target shared versions from other project" do 887
850 assigned_version = Version.generate!(:name => "Assigned Issues", :status => 'open', :project_id => 1, :sharing => 'system') 888 test "#completed_percent with versions should return 0 if the versions have no issues" do
851 issue = Issue.generate!(:project => @source_project, 889 project = Project.generate!
852 :fixed_version => assigned_version, 890 Version.generate!(:project => project)
853 :subject => "keep target shared versions") 891 Version.generate!(:project => project)
854 892
855 assert @project.copy(@source_project) 893 assert_equal 0, project.completed_percent
856 @project.reload 894 end
857 copied_issue = @project.issues.first(:conditions => {:subject => "keep target shared versions"}) 895
858 896 test "#completed_percent with versions should return 100 if the version has only closed issues" do
859 assert copied_issue 897 project = Project.generate!
860 assert_equal assigned_version, copied_issue.fixed_version 898 project.trackers << Tracker.generate!
861 end 899 v1 = Version.generate!(:project => project)
862 900 Issue.generate!(:project => project, :status => IssueStatus.find_by_name('Closed'), :fixed_version => v1)
863 should "copy issue relations" do 901 v2 = Version.generate!(:project => project)
864 Setting.cross_project_issue_relations = '1' 902 Issue.generate!(:project => project, :status => IssueStatus.find_by_name('Closed'), :fixed_version => v2)
865 903
866 second_issue = Issue.generate!(:status_id => 5, 904 assert_equal 100, project.completed_percent
867 :subject => "copy issue relation", 905 end
868 :tracker_id => 1, 906
869 :assigned_to_id => 2, 907 test "#completed_percent with versions should return the averaged completed percent of the versions (not weighted)" do
870 :project_id => @source_project.id) 908 project = Project.generate!
871 source_relation = IssueRelation.create!(:issue_from => Issue.find(4), 909 project.trackers << Tracker.generate!
872 :issue_to => second_issue, 910 v1 = Version.generate!(:project => project)
873 :relation_type => "relates") 911 Issue.generate!(:project => project, :status => IssueStatus.find_by_name('New'), :estimated_hours => 10, :done_ratio => 50, :fixed_version => v1)
874 source_relation_cross_project = IssueRelation.create!(:issue_from => Issue.find(1), 912 v2 = Version.generate!(:project => project)
875 :issue_to => second_issue, 913 Issue.generate!(:project => project, :status => IssueStatus.find_by_name('New'), :estimated_hours => 10, :done_ratio => 50, :fixed_version => v2)
876 :relation_type => "duplicates") 914
877 915 assert_equal 50, project.completed_percent
878 assert @project.copy(@source_project) 916 end
879 assert_equal @source_project.issues.count, @project.issues.count 917
880 copied_issue = @project.issues.find_by_subject("Issue on project 2") # Was #4 918 test "#notified_users" do
881 copied_second_issue = @project.issues.find_by_subject("copy issue relation") 919 project = Project.generate!
882 920 role = Role.generate!
883 # First issue with a relation on project 921
884 assert_equal 1, copied_issue.relations.size, "Relation not copied" 922 user_with_membership_notification = User.generate!(:mail_notification => 'selected')
885 copied_relation = copied_issue.relations.first 923 Member.create!(:project => project, :roles => [role], :principal => user_with_membership_notification, :mail_notification => true)
886 assert_equal "relates", copied_relation.relation_type 924
887 assert_equal copied_second_issue.id, copied_relation.issue_to_id 925 all_events_user = User.generate!(:mail_notification => 'all')
888 assert_not_equal source_relation.id, copied_relation.id 926 Member.create!(:project => project, :roles => [role], :principal => all_events_user)
889 927
890 # Second issue with a cross project relation 928 no_events_user = User.generate!(:mail_notification => 'none')
891 assert_equal 2, copied_second_issue.relations.size, "Relation not copied" 929 Member.create!(:project => project, :roles => [role], :principal => no_events_user)
892 copied_relation = copied_second_issue.relations.select {|r| r.relation_type == 'duplicates'}.first 930
893 assert_equal "duplicates", copied_relation.relation_type 931 only_my_events_user = User.generate!(:mail_notification => 'only_my_events')
894 assert_equal 1, copied_relation.issue_from_id, "Cross project relation not kept" 932 Member.create!(:project => project, :roles => [role], :principal => only_my_events_user)
895 assert_not_equal source_relation_cross_project.id, copied_relation.id 933
896 end 934 only_assigned_user = User.generate!(:mail_notification => 'only_assigned')
897 935 Member.create!(:project => project, :roles => [role], :principal => only_assigned_user)
898 should "copy issue attachments" do 936
899 issue = Issue.generate!(:subject => "copy with attachment", :tracker_id => 1, :project_id => @source_project.id) 937 only_owned_user = User.generate!(:mail_notification => 'only_owner')
900 Attachment.create!(:container => issue, :file => uploaded_test_file("testfile.txt", "text/plain"), :author_id => 1) 938 Member.create!(:project => project, :roles => [role], :principal => only_owned_user)
901 @source_project.issues << issue 939
902 assert @project.copy(@source_project) 940 assert project.notified_users.include?(user_with_membership_notification), "should include members with a mail notification"
903 941 assert project.notified_users.include?(all_events_user), "should include users with the 'all' notification option"
904 copied_issue = @project.issues.first(:conditions => {:subject => "copy with attachment"}) 942 assert !project.notified_users.include?(no_events_user), "should not include users with the 'none' notification option"
905 assert_not_nil copied_issue 943 assert !project.notified_users.include?(only_my_events_user), "should not include users with the 'only_my_events' notification option"
906 assert_equal 1, copied_issue.attachments.count, "Attachment not copied" 944 assert !project.notified_users.include?(only_assigned_user), "should not include users with the 'only_assigned' notification option"
907 assert_equal "testfile.txt", copied_issue.attachments.first.filename 945 assert !project.notified_users.include?(only_owned_user), "should not include users with the 'only_owner' notification option"
908 end 946 end
909
910 should "copy memberships" do
911 assert @project.valid?
912 assert @project.members.empty?
913 assert @project.copy(@source_project)
914
915 assert_equal @source_project.memberships.size, @project.memberships.size
916 @project.memberships.each do |membership|
917 assert membership
918 assert_equal @project, membership.project
919 end
920 end
921
922 should "copy memberships with groups and additional roles" do
923 group = Group.create!(:lastname => "Copy group")
924 user = User.find(7)
925 group.users << user
926 # group role
927 Member.create!(:project_id => @source_project.id, :principal => group, :role_ids => [2])
928 member = Member.find_by_user_id_and_project_id(user.id, @source_project.id)
929 # additional role
930 member.role_ids = [1]
931
932 assert @project.copy(@source_project)
933 member = Member.find_by_user_id_and_project_id(user.id, @project.id)
934 assert_not_nil member
935 assert_equal [1, 2], member.role_ids.sort
936 end
937
938 should "copy project specific queries" do
939 assert @project.valid?
940 assert @project.queries.empty?
941 assert @project.copy(@source_project)
942
943 assert_equal @source_project.queries.size, @project.queries.size
944 @project.queries.each do |query|
945 assert query
946 assert_equal @project, query.project
947 end
948 assert_equal @source_project.queries.map(&:user_id).sort, @project.queries.map(&:user_id).sort
949 end
950
951 should "copy versions" do
952 @source_project.versions << Version.generate!
953 @source_project.versions << Version.generate!
954
955 assert @project.versions.empty?
956 assert @project.copy(@source_project)
957
958 assert_equal @source_project.versions.size, @project.versions.size
959 @project.versions.each do |version|
960 assert version
961 assert_equal @project, version.project
962 end
963 end
964
965 should "copy wiki" do
966 assert_difference 'Wiki.count' do
967 assert @project.copy(@source_project)
968 end
969
970 assert @project.wiki
971 assert_not_equal @source_project.wiki, @project.wiki
972 assert_equal "Start page", @project.wiki.start_page
973 end
974
975 should "copy wiki pages and content with hierarchy" do
976 assert_difference 'WikiPage.count', @source_project.wiki.pages.size do
977 assert @project.copy(@source_project)
978 end
979
980 assert @project.wiki
981 assert_equal @source_project.wiki.pages.size, @project.wiki.pages.size
982
983 @project.wiki.pages.each do |wiki_page|
984 assert wiki_page.content
985 assert !@source_project.wiki.pages.include?(wiki_page)
986 end
987
988 parent = @project.wiki.find_page('Parent_page')
989 child1 = @project.wiki.find_page('Child_page_1')
990 child2 = @project.wiki.find_page('Child_page_2')
991 assert_equal parent, child1.parent
992 assert_equal parent, child2.parent
993 end
994
995 should "copy issue categories" do
996 assert @project.copy(@source_project)
997
998 assert_equal 2, @project.issue_categories.size
999 @project.issue_categories.each do |issue_category|
1000 assert !@source_project.issue_categories.include?(issue_category)
1001 end
1002 end
1003
1004 should "copy boards" do
1005 assert @project.copy(@source_project)
1006
1007 assert_equal 1, @project.boards.size
1008 @project.boards.each do |board|
1009 assert !@source_project.boards.include?(board)
1010 end
1011 end
1012
1013 should "change the new issues to use the copied issue categories" do
1014 issue = Issue.find(4)
1015 issue.update_attribute(:category_id, 3)
1016
1017 assert @project.copy(@source_project)
1018
1019 @project.issues.each do |issue|
1020 assert issue.category
1021 assert_equal "Stock management", issue.category.name # Same name
1022 assert_not_equal IssueCategory.find(3), issue.category # Different record
1023 end
1024 end
1025
1026 should "limit copy with :only option" do
1027 assert @project.members.empty?
1028 assert @project.issue_categories.empty?
1029 assert @source_project.issues.any?
1030
1031 assert @project.copy(@source_project, :only => ['members', 'issue_categories'])
1032
1033 assert @project.members.any?
1034 assert @project.issue_categories.any?
1035 assert @project.issues.empty?
1036 end
1037 end
1038
1039 def test_copy_should_copy_subtasks
1040 source = Project.generate!(:tracker_ids => [1])
1041 issue = Issue.generate_with_descendants!(:project => source)
1042 project = Project.new(:name => 'Copy', :identifier => 'copy', :tracker_ids => [1])
1043
1044 assert_difference 'Project.count' do
1045 assert_difference 'Issue.count', 1+issue.descendants.count do
1046 assert project.copy(source.reload)
1047 end
1048 end
1049 copy = Issue.where(:parent_id => nil).order("id DESC").first
1050 assert_equal project, copy.project
1051 assert_equal issue.descendants.count, copy.descendants.count
1052 child_copy = copy.children.detect {|c| c.subject == 'Child1'}
1053 assert child_copy.descendants.any?
1054 end
1055
1056 context "#start_date" do
1057 setup do
1058 ProjectCustomField.destroy_all # Custom values are a mess to isolate in tests
1059 @project = Project.generate!(:identifier => 'test0')
1060 @project.trackers << Tracker.generate!
1061 end
1062
1063 should "be nil if there are no issues on the project" do
1064 assert_nil @project.start_date
1065 end
1066
1067 should "be tested when issues have no start date"
1068
1069 should "be the earliest start date of it's issues" do
1070 early = 7.days.ago.to_date
1071 Issue.generate!(:project => @project, :start_date => Date.today)
1072 Issue.generate!(:project => @project, :start_date => early)
1073
1074 assert_equal early, @project.start_date
1075 end
1076
1077 end
1078
1079 context "#due_date" do
1080 setup do
1081 ProjectCustomField.destroy_all # Custom values are a mess to isolate in tests
1082 @project = Project.generate!(:identifier => 'test0')
1083 @project.trackers << Tracker.generate!
1084 end
1085
1086 should "be nil if there are no issues on the project" do
1087 assert_nil @project.due_date
1088 end
1089
1090 should "be tested when issues have no due date"
1091
1092 should "be the latest due date of it's issues" do
1093 future = 7.days.from_now.to_date
1094 Issue.generate!(:project => @project, :due_date => future)
1095 Issue.generate!(:project => @project, :due_date => Date.today)
1096
1097 assert_equal future, @project.due_date
1098 end
1099
1100 should "be the latest due date of it's versions" do
1101 future = 7.days.from_now.to_date
1102 @project.versions << Version.generate!(:effective_date => future)
1103 @project.versions << Version.generate!(:effective_date => Date.today)
1104
1105
1106 assert_equal future, @project.due_date
1107
1108 end
1109
1110 should "pick the latest date from it's issues and versions" do
1111 future = 7.days.from_now.to_date
1112 far_future = 14.days.from_now.to_date
1113 Issue.generate!(:project => @project, :due_date => far_future)
1114 @project.versions << Version.generate!(:effective_date => future)
1115
1116 assert_equal far_future, @project.due_date
1117 end
1118
1119 end
1120
1121 context "Project#completed_percent" do
1122 setup do
1123 ProjectCustomField.destroy_all # Custom values are a mess to isolate in tests
1124 @project = Project.generate!(:identifier => 'test0')
1125 @project.trackers << Tracker.generate!
1126 end
1127
1128 context "no versions" do
1129 should "be 100" do
1130 assert_equal 100, @project.completed_percent
1131 end
1132 end
1133
1134 context "with versions" do
1135 should "return 0 if the versions have no issues" do
1136 Version.generate!(:project => @project)
1137 Version.generate!(:project => @project)
1138
1139 assert_equal 0, @project.completed_percent
1140 end
1141
1142 should "return 100 if the version has only closed issues" do
1143 v1 = Version.generate!(:project => @project)
1144 Issue.generate!(:project => @project, :status => IssueStatus.find_by_name('Closed'), :fixed_version => v1)
1145 v2 = Version.generate!(:project => @project)
1146 Issue.generate!(:project => @project, :status => IssueStatus.find_by_name('Closed'), :fixed_version => v2)
1147
1148 assert_equal 100, @project.completed_percent
1149 end
1150
1151 should "return the averaged completed percent of the versions (not weighted)" do
1152 v1 = Version.generate!(:project => @project)
1153 Issue.generate!(:project => @project, :status => IssueStatus.find_by_name('New'), :estimated_hours => 10, :done_ratio => 50, :fixed_version => v1)
1154 v2 = Version.generate!(:project => @project)
1155 Issue.generate!(:project => @project, :status => IssueStatus.find_by_name('New'), :estimated_hours => 10, :done_ratio => 50, :fixed_version => v2)
1156
1157 assert_equal 50, @project.completed_percent
1158 end
1159
1160 end
1161 end
1162
1163 context "#notified_users" do
1164 setup do
1165 @project = Project.generate!
1166 @role = Role.generate!
1167
1168 @user_with_membership_notification = User.generate!(:mail_notification => 'selected')
1169 Member.create!(:project => @project, :roles => [@role], :principal => @user_with_membership_notification, :mail_notification => true)
1170
1171 @all_events_user = User.generate!(:mail_notification => 'all')
1172 Member.create!(:project => @project, :roles => [@role], :principal => @all_events_user)
1173
1174 @no_events_user = User.generate!(:mail_notification => 'none')
1175 Member.create!(:project => @project, :roles => [@role], :principal => @no_events_user)
1176
1177 @only_my_events_user = User.generate!(:mail_notification => 'only_my_events')
1178 Member.create!(:project => @project, :roles => [@role], :principal => @only_my_events_user)
1179
1180 @only_assigned_user = User.generate!(:mail_notification => 'only_assigned')
1181 Member.create!(:project => @project, :roles => [@role], :principal => @only_assigned_user)
1182
1183 @only_owned_user = User.generate!(:mail_notification => 'only_owner')
1184 Member.create!(:project => @project, :roles => [@role], :principal => @only_owned_user)
1185 end
1186
1187 should "include members with a mail notification" do
1188 assert @project.notified_users.include?(@user_with_membership_notification)
1189 end
1190
1191 should "include users with the 'all' notification option" do
1192 assert @project.notified_users.include?(@all_events_user)
1193 end
1194
1195 should "not include users with the 'none' notification option" do
1196 assert !@project.notified_users.include?(@no_events_user)
1197 end
1198
1199 should "not include users with the 'only_my_events' notification option" do
1200 assert !@project.notified_users.include?(@only_my_events_user)
1201 end
1202
1203 should "not include users with the 'only_assigned' notification option" do
1204 assert !@project.notified_users.include?(@only_assigned_user)
1205 end
1206
1207 should "not include users with the 'only_owner' notification option" do
1208 assert !@project.notified_users.include?(@only_owned_user)
1209 end
1210 end
1211
1212 end 947 end