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.

project_test.rb 41KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146
  1. # frozen_string_literal: true
  2. # Redmine - project management software
  3. # Copyright (C) 2006-2021 Jean-Philippe Lang
  4. #
  5. # This program is free software; you can redistribute it and/or
  6. # modify it under the terms of the GNU General Public License
  7. # as published by the Free Software Foundation; either version 2
  8. # of the License, or (at your option) any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with this program; if not, write to the Free Software
  17. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  18. require File.expand_path('../../test_helper', __FILE__)
  19. class ProjectTest < ActiveSupport::TestCase
  20. fixtures :projects, :trackers, :issue_statuses, :issues,
  21. :journals, :journal_details,
  22. :enumerations, :users, :issue_categories,
  23. :projects_trackers,
  24. :custom_fields,
  25. :custom_fields_projects,
  26. :custom_fields_trackers,
  27. :custom_values,
  28. :roles,
  29. :member_roles,
  30. :members,
  31. :enabled_modules,
  32. :versions,
  33. :wikis, :wiki_pages, :wiki_contents, :wiki_content_versions,
  34. :groups_users,
  35. :boards, :messages,
  36. :repositories,
  37. :news, :comments,
  38. :documents,
  39. :workflows,
  40. :attachments
  41. def setup
  42. @ecookbook = Project.find(1)
  43. @ecookbook_sub1 = Project.find(3)
  44. set_tmp_attachments_directory
  45. User.current = nil
  46. end
  47. def test_truth
  48. assert_kind_of Project, @ecookbook
  49. assert_equal "eCookbook", @ecookbook.name
  50. end
  51. def test_default_attributes
  52. with_settings :default_projects_public => '1' do
  53. assert_equal true, Project.new.is_public
  54. assert_equal false, Project.new(:is_public => false).is_public
  55. end
  56. with_settings :default_projects_public => '0' do
  57. assert_equal false, Project.new.is_public
  58. assert_equal true, Project.new(:is_public => true).is_public
  59. end
  60. with_settings :sequential_project_identifiers => '1' do
  61. assert !Project.new.identifier.blank?
  62. assert Project.new(:identifier => '').identifier.blank?
  63. end
  64. with_settings :sequential_project_identifiers => '0' do
  65. assert Project.new.identifier.blank?
  66. assert !Project.new(:identifier => 'test').blank?
  67. end
  68. with_settings :default_projects_modules => ['issue_tracking', 'repository'] do
  69. assert_equal ['issue_tracking', 'repository'], Project.new.enabled_module_names
  70. end
  71. end
  72. def test_default_trackers_should_match_default_tracker_ids_setting
  73. with_settings :default_projects_tracker_ids => ['1', '3'] do
  74. assert_equal Tracker.find(1, 3).sort, Project.new.trackers.sort
  75. end
  76. end
  77. def test_default_trackers_should_be_all_trackers_with_blank_setting
  78. with_settings :default_projects_tracker_ids => nil do
  79. assert_equal Tracker.all.sort, Project.new.trackers.sort
  80. end
  81. end
  82. def test_default_trackers_should_be_empty_with_empty_setting
  83. with_settings :default_projects_tracker_ids => [] do
  84. assert_equal [], Project.new.trackers
  85. end
  86. end
  87. def test_default_trackers_should_not_replace_initialized_trackers
  88. with_settings :default_projects_tracker_ids => ['1', '3'] do
  89. assert_equal Tracker.find(1, 2).sort, Project.new(:tracker_ids => [1, 2]).trackers.sort
  90. end
  91. end
  92. def test_update
  93. assert_equal "eCookbook", @ecookbook.name
  94. @ecookbook.name = "eCook"
  95. assert @ecookbook.save, @ecookbook.errors.full_messages.join("; ")
  96. @ecookbook.reload
  97. assert_equal "eCook", @ecookbook.name
  98. end
  99. def test_validate_identifier
  100. to_test = {"abc" => true,
  101. "ab12" => true,
  102. "ab-12" => true,
  103. "ab_12" => true,
  104. "12" => false,
  105. "new" => false}
  106. to_test.each do |identifier, valid|
  107. p = Project.new
  108. p.identifier = identifier
  109. p.valid?
  110. if valid
  111. assert p.errors['identifier'].blank?, "identifier #{identifier} was not valid"
  112. else
  113. assert p.errors['identifier'].present?, "identifier #{identifier} was valid"
  114. end
  115. end
  116. end
  117. def test_identifier_should_not_be_frozen_for_a_new_project
  118. assert_equal false, Project.new.identifier_frozen?
  119. end
  120. def test_identifier_should_not_be_frozen_for_a_saved_project_with_blank_identifier
  121. Project.where(:id => 1).update_all(["identifier = ''"])
  122. assert_equal false, Project.find(1).identifier_frozen?
  123. end
  124. def test_identifier_should_be_frozen_for_a_saved_project_with_valid_identifier
  125. assert_equal true, Project.find(1).identifier_frozen?
  126. end
  127. def test_to_param_should_be_nil_for_new_records
  128. project = Project.new
  129. project.identifier = "foo"
  130. assert_nil project.to_param
  131. end
  132. def test_members_should_be_active_users
  133. Project.all.each do |project|
  134. assert_nil project.members.detect {|m| !(m.user.is_a?(User) && m.user.active?)}
  135. end
  136. end
  137. def test_users_should_be_active_users
  138. Project.all.each do |project|
  139. assert_nil project.users.detect {|u| !(u.is_a?(User) && u.active?)}
  140. end
  141. end
  142. def test_open_scope_on_issues_association
  143. assert_kind_of Issue, Project.find(1).issues.open.first
  144. end
  145. def test_archive
  146. user = @ecookbook.members.first.user
  147. @ecookbook.archive
  148. @ecookbook.reload
  149. assert !@ecookbook.active?
  150. assert @ecookbook.archived?
  151. assert !user.projects.include?(@ecookbook)
  152. # Subproject are also archived
  153. assert !@ecookbook.children.empty?
  154. assert @ecookbook.descendants.active.empty?
  155. end
  156. def test_archive_should_fail_if_versions_are_used_by_non_descendant_projects
  157. # Assign an issue of a project to a version of a child project
  158. Issue.find(4).update_attribute :fixed_version_id, 4
  159. assert_no_difference "Project.where(:status => Project::STATUS_ARCHIVED).count" do
  160. assert_equal false, @ecookbook.archive
  161. end
  162. @ecookbook.reload
  163. assert @ecookbook.active?
  164. end
  165. def test_unarchive
  166. user = @ecookbook.members.first.user
  167. @ecookbook.archive
  168. # Unarchive project
  169. assert @ecookbook.unarchive
  170. assert @ecookbook.active?
  171. assert !@ecookbook.archived?
  172. assert user.projects.include?(@ecookbook)
  173. end
  174. def test_unarchive_child_project_should_unarchive_ancestors
  175. @ecookbook.archive
  176. @ecookbook_sub1.reload
  177. assert_equal Project::STATUS_ARCHIVED, @ecookbook_sub1.status
  178. @ecookbook_sub1.unarchive
  179. assert_equal Project::STATUS_ACTIVE, @ecookbook_sub1.status
  180. @ecookbook.reload
  181. assert_equal Project::STATUS_ACTIVE, @ecookbook.status
  182. end
  183. def test_unarchive_a_child_of_a_closed_project_should_set_status_to_closed
  184. Project.find(1).close
  185. child = Project.find(3)
  186. assert_equal Project::STATUS_CLOSED, child.status
  187. child.archive
  188. assert_equal Project::STATUS_ARCHIVED, child.status
  189. child.unarchive
  190. assert_equal Project::STATUS_CLOSED, child.status
  191. end
  192. def test_destroy
  193. # 2 active members
  194. assert_equal 2, @ecookbook.members.size
  195. # and 1 is locked
  196. assert_equal 3, Member.where(:project_id => @ecookbook.id).count
  197. # some boards
  198. assert @ecookbook.boards.any?
  199. @ecookbook.destroy
  200. # make sure that the project non longer exists
  201. assert_raise(ActiveRecord::RecordNotFound) {Project.find(@ecookbook.id)}
  202. # make sure related data was removed
  203. assert_not Member.where(:project_id => @ecookbook.id).exists?
  204. assert_not Board.where(:project_id => @ecookbook.id).exists?
  205. assert_not Issue.where(:project_id => @ecookbook.id).exists?
  206. end
  207. def test_destroy_should_destroy_subtasks
  208. issues =
  209. (0..2).to_a.map do
  210. Issue.generate!(:project_id => 1, :tracker_id => 1,
  211. :author_id => 1, :subject => 'test')
  212. end
  213. issues[0].update! :parent_issue_id => issues[1].id
  214. issues[2].update! :parent_issue_id => issues[1].id
  215. assert_equal 2, issues[1].children.count
  216. assert_nothing_raised do
  217. Project.find(1).destroy
  218. end
  219. assert_equal 0, Issue.where(:id => issues.map(&:id)).count
  220. end
  221. def test_destroying_root_projects_should_clear_data
  222. Project.roots.each do |root|
  223. root.destroy
  224. end
  225. assert_equal 0, Project.count, "Projects were not deleted: #{Project.all.inspect}"
  226. assert_equal 0, Member.count, "Members were not deleted: #{Member.all.inspect}"
  227. assert_equal 0, MemberRole.count
  228. assert_equal 0, Issue.count
  229. assert_equal 0, Journal.count
  230. assert_equal 0, JournalDetail.count
  231. assert_equal 0, Attachment.count, "Attachments were not deleted: #{Attachment.all.inspect}"
  232. assert_equal 0, EnabledModule.count
  233. assert_equal 0, IssueCategory.count
  234. assert_equal 0, IssueRelation.count
  235. assert_equal 0, Board.count
  236. assert_equal 0, Message.count
  237. assert_equal 0, News.count
  238. assert_equal 0, Query.where("project_id IS NOT NULL").count
  239. assert_equal 0, Repository.count
  240. assert_equal 0, Changeset.count
  241. assert_equal 0, Change.count
  242. assert_equal 0, Comment.count
  243. assert_equal 0, TimeEntry.count
  244. assert_equal 0, Version.count
  245. assert_equal 0, Watcher.count
  246. assert_equal 0, Wiki.count
  247. assert_equal 0, WikiPage.count
  248. assert_equal 0, WikiContent.count
  249. assert_equal 0, WikiContentVersion.count
  250. assert_equal 0, Project.connection.select_all("SELECT * FROM projects_trackers").count
  251. assert_equal 0, Project.connection.select_all("SELECT * FROM custom_fields_projects").count
  252. assert_equal 0, CustomValue.where(:customized_type => ['Project', 'Issue', 'TimeEntry', 'Version']).count
  253. end
  254. def test_destroy_should_delete_time_entries_custom_values
  255. project = Project.generate!
  256. time_entry = TimeEntry.generate!(:project => project, :custom_field_values => {10 => '1'})
  257. assert_difference 'CustomValue.where(:customized_type => "TimeEntry").count', -1 do
  258. assert project.destroy
  259. end
  260. end
  261. def test_move_an_orphan_project_to_a_root_project
  262. sub = Project.find(2)
  263. sub.set_parent! @ecookbook
  264. assert_equal @ecookbook.id, sub.parent.id
  265. @ecookbook.reload
  266. assert_equal 4, @ecookbook.children.size
  267. end
  268. def test_move_an_orphan_project_to_a_subproject
  269. sub = Project.find(2)
  270. assert sub.set_parent!(@ecookbook_sub1)
  271. end
  272. def test_move_a_root_project_to_a_project
  273. sub = @ecookbook
  274. assert sub.set_parent!(Project.find(2))
  275. end
  276. def test_should_not_move_a_project_to_its_children
  277. sub = @ecookbook
  278. assert !(sub.set_parent!(Project.find(3)))
  279. end
  280. def test_set_parent_should_add_roots_in_alphabetical_order
  281. projects = new_records(Project, 4) do
  282. Project.create!(:name => 'Project C', :identifier => 'project-c').set_parent!(nil)
  283. Project.create!(:name => 'Project B', :identifier => 'project-b').set_parent!(nil)
  284. Project.create!(:name => 'Project D', :identifier => 'project-d').set_parent!(nil)
  285. Project.create!(:name => 'Project A', :identifier => 'project-a').set_parent!(nil)
  286. end
  287. assert_equal projects.sort_by(&:name), projects.sort_by(&:lft)
  288. end
  289. def test_set_parent_should_add_children_in_alphabetical_order
  290. ProjectCustomField.delete_all
  291. parent = Project.create!(:name => 'Parent', :identifier => 'parent')
  292. Project.create!(:name => 'Project C', :identifier => 'project-c').set_parent!(parent)
  293. Project.create!(:name => 'Project B', :identifier => 'project-b').set_parent!(parent)
  294. Project.create!(:name => 'Project D', :identifier => 'project-d').set_parent!(parent)
  295. Project.create!(:name => 'Project A', :identifier => 'project-a').set_parent!(parent)
  296. parent.reload
  297. assert_equal 4, parent.children.size
  298. assert_equal parent.children.sort_by(&:name), parent.children.to_a
  299. end
  300. def test_set_parent_should_update_issue_fixed_version_associations_when_a_fixed_version_is_moved_out_of_the_hierarchy
  301. # Parent issue with a hierarchy project's fixed version
  302. parent_issue = Issue.find(1)
  303. parent_issue.update_attribute(:fixed_version_id, 4)
  304. parent_issue.reload
  305. assert_equal 4, parent_issue.fixed_version_id
  306. # Should keep fixed versions for the issues
  307. issue_with_local_fixed_version = Issue.find(5)
  308. issue_with_local_fixed_version.update_attribute(:fixed_version_id, 4)
  309. issue_with_local_fixed_version.reload
  310. assert_equal 4, issue_with_local_fixed_version.fixed_version_id
  311. # Local issue with hierarchy fixed_version
  312. issue_with_hierarchy_fixed_version = Issue.find(13)
  313. issue_with_hierarchy_fixed_version.update_attribute(:fixed_version_id, 6)
  314. issue_with_hierarchy_fixed_version.reload
  315. assert_equal 6, issue_with_hierarchy_fixed_version.fixed_version_id
  316. # Move project out of the issue's hierarchy
  317. moved_project = Project.find(3)
  318. moved_project.set_parent!(Project.find(2))
  319. parent_issue.reload
  320. issue_with_local_fixed_version.reload
  321. issue_with_hierarchy_fixed_version.reload
  322. assert_equal 4, issue_with_local_fixed_version.fixed_version_id,
  323. "Fixed version was not keep on an issue local to the moved project"
  324. assert_nil issue_with_hierarchy_fixed_version.fixed_version_id,
  325. "Fixed version is still set after moving the Project out of the hierarchy " \
  326. "where the version is defined in"
  327. assert_nil parent_issue.fixed_version_id,
  328. "Fixed version is still set after moving the Version out of the hierarchy for the issue."
  329. end
  330. def test_parent
  331. p = Project.find(6).parent
  332. assert p.is_a?(Project)
  333. assert_equal 5, p.id
  334. end
  335. def test_ancestors
  336. a = Project.find(6).ancestors
  337. assert a.first.is_a?(Project)
  338. assert_equal [1, 5], a.collect(&:id)
  339. end
  340. def test_root
  341. r = Project.find(6).root
  342. assert r.is_a?(Project)
  343. assert_equal 1, r.id
  344. end
  345. def test_children
  346. c = Project.find(1).children
  347. assert c.first.is_a?(Project)
  348. assert_equal [5, 3, 4], c.collect(&:id)
  349. end
  350. def test_descendants
  351. d = Project.find(1).descendants
  352. assert d.first.is_a?(Project)
  353. assert_equal [5, 6, 3, 4], d.collect(&:id)
  354. end
  355. def test_allowed_parents_should_be_empty_for_non_member_user
  356. Role.non_member.add_permission!(:add_project)
  357. user = User.find(9)
  358. assert user.memberships.empty?
  359. User.current = user
  360. assert Project.new.allowed_parents.compact.empty?
  361. end
  362. def test_allowed_parents_with_add_subprojects_permission
  363. Role.find(1).remove_permission!(:add_project)
  364. Role.find(1).add_permission!(:add_subprojects)
  365. User.current = User.find(2)
  366. # new project
  367. assert !Project.new.allowed_parents.include?(nil)
  368. assert Project.new.allowed_parents.include?(Project.find(1))
  369. # existing root project
  370. assert Project.find(1).allowed_parents.include?(nil)
  371. # existing child
  372. assert Project.find(3).allowed_parents.include?(Project.find(1))
  373. assert !Project.find(3).allowed_parents.include?(nil)
  374. end
  375. def test_allowed_parents_with_add_project_permission
  376. Role.find(1).add_permission!(:add_project)
  377. Role.find(1).remove_permission!(:add_subprojects)
  378. User.current = User.find(2)
  379. # new project
  380. assert Project.new.allowed_parents.include?(nil)
  381. assert !Project.new.allowed_parents.include?(Project.find(1))
  382. # existing root project
  383. assert Project.find(1).allowed_parents.include?(nil)
  384. # existing child
  385. assert Project.find(3).allowed_parents.include?(Project.find(1))
  386. assert Project.find(3).allowed_parents.include?(nil)
  387. end
  388. def test_allowed_parents_with_add_project_and_subprojects_permission
  389. Role.find(1).add_permission!(:add_project)
  390. Role.find(1).add_permission!(:add_subprojects)
  391. User.current = User.find(2)
  392. # new project
  393. assert Project.new.allowed_parents.include?(nil)
  394. assert Project.new.allowed_parents.include?(Project.find(1))
  395. # existing root project
  396. assert Project.find(1).allowed_parents.include?(nil)
  397. # existing child
  398. assert Project.find(3).allowed_parents.include?(Project.find(1))
  399. assert Project.find(3).allowed_parents.include?(nil)
  400. end
  401. def test_principals_by_role
  402. group = Group.find(10)
  403. Member.create!(:principal => group, :project_id => 1, :role_ids => [1])
  404. principals_by_role = Project.find(1).principals_by_role
  405. assert_kind_of Hash, principals_by_role
  406. role = Role.find(1)
  407. assert_kind_of Array, principals_by_role[role]
  408. assert principals_by_role[role].include?(User.find(2))
  409. assert principals_by_role[role].include?(group)
  410. end
  411. def test_principals_by_role_should_only_return_active_users
  412. principals_by_role = Project.find(1).principals_by_role
  413. locked_user = User.find(5)
  414. assert Project.find(1).memberships.map(&:principal).include?(locked_user)
  415. assert_not principals_by_role.values.flatten.include?(locked_user)
  416. end
  417. def test_rolled_up_trackers
  418. parent = Project.find(1)
  419. parent.trackers = Tracker.find([1, 2])
  420. child = parent.children.find(3)
  421. child.trackers = Tracker.find([2, 3])
  422. assert_equal [1, 2], parent.tracker_ids
  423. assert_equal [2, 3], child.trackers.collect(&:id)
  424. assert_kind_of Tracker, parent.rolled_up_trackers.first
  425. assert_equal Tracker.find(1), parent.rolled_up_trackers.first
  426. assert_equal [1, 2, 3], parent.rolled_up_trackers.collect(&:id)
  427. assert_equal [2, 3], child.rolled_up_trackers.collect(&:id)
  428. end
  429. def test_rolled_up_trackers_should_ignore_archived_subprojects
  430. parent = Project.find(1)
  431. parent.trackers = Tracker.find([1, 2])
  432. child = parent.children.find(3)
  433. child.trackers = Tracker.find([1, 3])
  434. parent.children.each(&:archive)
  435. assert_equal [1, 2], parent.rolled_up_trackers.collect(&:id)
  436. end
  437. def test_rolled_up_statuses
  438. project = Project.find(1)
  439. WorkflowTransition.delete_all
  440. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 3)
  441. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 4)
  442. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 2, :new_status_id => 3)
  443. WorkflowTransition.create(:role_id => 1, :tracker_id => 2, :old_status_id => 1, :new_status_id => 3)
  444. assert_kind_of IssueStatus, project.rolled_up_statuses.first
  445. assert_equal IssueStatus.find(1), project.rolled_up_statuses.first
  446. assert_equal [1, 2, 3, 4], project.rolled_up_statuses.collect(&:id)
  447. end
  448. test "#rolled_up_trackers should ignore projects with issue_tracking module disabled" do
  449. parent = Project.generate!
  450. parent.trackers = Tracker.find([1, 2])
  451. child = Project.generate_with_parent!(parent)
  452. child.trackers = Tracker.find([2, 3])
  453. assert_equal [1, 2, 3], parent.rolled_up_trackers.collect(&:id).sort
  454. assert child.disable_module!(:issue_tracking)
  455. parent.reload
  456. assert_equal [1, 2], parent.rolled_up_trackers.collect(&:id).sort
  457. end
  458. test "#rolled_up_versions should include the versions for the current project" do
  459. project = Project.generate!
  460. parent_version_1 = Version.generate!(:project => project)
  461. parent_version_2 = Version.generate!(:project => project)
  462. assert_equal [parent_version_1, parent_version_2].sort,
  463. project.rolled_up_versions.sort
  464. end
  465. test "#rolled_up_versions should include versions for a subproject" do
  466. project = Project.generate!
  467. parent_version_1 = Version.generate!(:project => project)
  468. parent_version_2 = Version.generate!(:project => project)
  469. subproject = Project.generate_with_parent!(project)
  470. subproject_version = Version.generate!(:project => subproject)
  471. assert_equal [parent_version_1, parent_version_2, subproject_version].sort,
  472. project.rolled_up_versions.sort
  473. end
  474. test "#rolled_up_versions should include versions for a sub-subproject" do
  475. project = Project.generate!
  476. parent_version_1 = Version.generate!(:project => project)
  477. parent_version_2 = Version.generate!(:project => project)
  478. subproject = Project.generate_with_parent!(project)
  479. sub_subproject = Project.generate_with_parent!(subproject)
  480. sub_subproject_version = Version.generate!(:project => sub_subproject)
  481. project.reload
  482. assert_equal [parent_version_1, parent_version_2, sub_subproject_version].sort,
  483. project.rolled_up_versions.sort
  484. end
  485. test "#rolled_up_versions should only check active projects" do
  486. project = Project.generate!
  487. parent_version_1 = Version.generate!(:project => project)
  488. parent_version_2 = Version.generate!(:project => project)
  489. subproject = Project.generate_with_parent!(project)
  490. subproject_version = Version.generate!(:project => subproject)
  491. assert subproject.archive
  492. project.reload
  493. assert !subproject.active?
  494. assert_equal [parent_version_1, parent_version_2].sort,
  495. project.rolled_up_versions.sort
  496. end
  497. def test_shared_versions_none_sharing
  498. p = Project.find(5)
  499. v = Version.create!(:name => 'none_sharing', :project => p, :sharing => 'none')
  500. assert p.shared_versions.include?(v)
  501. assert !p.children.first.shared_versions.include?(v)
  502. assert !p.root.shared_versions.include?(v)
  503. assert !p.siblings.first.shared_versions.include?(v)
  504. assert !p.root.siblings.first.shared_versions.include?(v)
  505. end
  506. def test_shared_versions_descendants_sharing
  507. p = Project.find(5)
  508. v = Version.create!(:name => 'descendants_sharing', :project => p, :sharing => 'descendants')
  509. assert p.shared_versions.include?(v)
  510. assert p.children.first.shared_versions.include?(v)
  511. assert !p.root.shared_versions.include?(v)
  512. assert !p.siblings.first.shared_versions.include?(v)
  513. assert !p.root.siblings.first.shared_versions.include?(v)
  514. end
  515. def test_shared_versions_hierarchy_sharing
  516. p = Project.find(5)
  517. v = Version.create!(:name => 'hierarchy_sharing', :project => p, :sharing => 'hierarchy')
  518. assert p.shared_versions.include?(v)
  519. assert p.children.first.shared_versions.include?(v)
  520. assert p.root.shared_versions.include?(v)
  521. assert !p.siblings.first.shared_versions.include?(v)
  522. assert !p.root.siblings.first.shared_versions.include?(v)
  523. end
  524. def test_shared_versions_tree_sharing
  525. p = Project.find(5)
  526. v = Version.create!(:name => 'tree_sharing', :project => p, :sharing => 'tree')
  527. assert p.shared_versions.include?(v)
  528. assert p.children.first.shared_versions.include?(v)
  529. assert p.root.shared_versions.include?(v)
  530. assert p.siblings.first.shared_versions.include?(v)
  531. assert !p.root.siblings.first.shared_versions.include?(v)
  532. end
  533. def test_shared_versions_system_sharing
  534. p = Project.find(5)
  535. v = Version.create!(:name => 'system_sharing', :project => p, :sharing => 'system')
  536. assert p.shared_versions.include?(v)
  537. assert p.children.first.shared_versions.include?(v)
  538. assert p.root.shared_versions.include?(v)
  539. assert p.siblings.first.shared_versions.include?(v)
  540. assert p.root.siblings.first.shared_versions.include?(v)
  541. end
  542. def test_shared_versions
  543. parent = Project.find(1)
  544. child = parent.children.find(3)
  545. private_child = parent.children.find(5)
  546. assert_equal [1, 2, 3], parent.version_ids.sort
  547. assert_equal [4], child.version_ids
  548. assert_equal [6], private_child.version_ids
  549. assert_equal [7], Version.where(:sharing => 'system').collect(&:id)
  550. assert_equal 6, parent.shared_versions.size
  551. parent.shared_versions.each do |version|
  552. assert_kind_of Version, version
  553. end
  554. assert_equal [1, 2, 3, 4, 6, 7], parent.shared_versions.collect(&:id).sort
  555. end
  556. def test_shared_versions_should_ignore_archived_subprojects
  557. parent = Project.find(1)
  558. child = parent.children.find(3)
  559. child.archive
  560. parent.reload
  561. assert_equal [1, 2, 3], parent.version_ids.sort
  562. assert_equal [4], child.version_ids
  563. assert !parent.shared_versions.collect(&:id).include?(4)
  564. end
  565. def test_shared_versions_visible_to_user
  566. user = User.find(3)
  567. parent = Project.find(1)
  568. child = parent.children.find(5)
  569. assert_equal [1, 2, 3], parent.version_ids.sort
  570. assert_equal [6], child.version_ids
  571. versions = parent.shared_versions.visible(user)
  572. assert_equal 4, versions.size
  573. versions.each do |version|
  574. assert_kind_of Version, version
  575. end
  576. assert !versions.collect(&:id).include?(6)
  577. end
  578. def test_shared_versions_for_new_project_should_include_system_shared_versions
  579. p = Project.find(5)
  580. v = Version.create!(:name => 'system_sharing', :project => p, :sharing => 'system')
  581. assert_include v, Project.new.shared_versions
  582. end
  583. def test_next_identifier
  584. ProjectCustomField.delete_all
  585. Project.create!(:name => 'last', :identifier => 'p2008040')
  586. assert_equal 'p2008041', Project.next_identifier
  587. end
  588. def test_next_identifier_first_project
  589. Project.delete_all
  590. assert_nil Project.next_identifier
  591. end
  592. def test_enabled_module_names
  593. with_settings :default_projects_modules => ['issue_tracking', 'repository'] do
  594. project = Project.new
  595. project.enabled_module_names = %w(issue_tracking news)
  596. assert_equal %w(issue_tracking news), project.enabled_module_names.sort
  597. end
  598. end
  599. def test_enabled_modules_names_with_nil_should_clear_modules
  600. p = Project.find(1)
  601. p.enabled_module_names = nil
  602. assert_equal [], p.enabled_modules
  603. end
  604. test "enabled_modules should define module by names and preserve ids" do
  605. @project = Project.find(1)
  606. # Remove one module
  607. modules = @project.enabled_modules.to_a.slice(0..-2)
  608. assert modules.any?
  609. assert_difference 'EnabledModule.count', -1 do
  610. @project.enabled_module_names = modules.collect(&:name)
  611. end
  612. @project.reload
  613. # Ids should be preserved
  614. assert_equal @project.enabled_module_ids.sort, modules.collect(&:id).sort
  615. end
  616. test "enabled_modules should enable a module" do
  617. @project = Project.find(1)
  618. @project.enabled_module_names = []
  619. @project.reload
  620. assert_equal [], @project.enabled_module_names
  621. # with string
  622. @project.enable_module!("issue_tracking")
  623. assert_equal ["issue_tracking"], @project.enabled_module_names
  624. # with symbol
  625. @project.enable_module!(:gantt)
  626. assert_equal ["issue_tracking", "gantt"], @project.enabled_module_names
  627. # don't add a module twice
  628. @project.enable_module!("issue_tracking")
  629. assert_equal ["issue_tracking", "gantt"], @project.enabled_module_names
  630. end
  631. test "enabled_modules should disable a module" do
  632. @project = Project.find(1)
  633. # with string
  634. assert @project.enabled_module_names.include?("issue_tracking")
  635. @project.disable_module!("issue_tracking")
  636. assert ! @project.reload.enabled_module_names.include?("issue_tracking")
  637. # with symbol
  638. assert @project.enabled_module_names.include?("gantt")
  639. @project.disable_module!(:gantt)
  640. assert ! @project.reload.enabled_module_names.include?("gantt")
  641. # with EnabledModule object
  642. first_module = @project.enabled_modules.first
  643. @project.disable_module!(first_module)
  644. assert ! @project.reload.enabled_module_names.include?(first_module.name)
  645. end
  646. def test_enabled_module_names_should_not_recreate_enabled_modules
  647. project = Project.find(1)
  648. # Remove one module
  649. modules = project.enabled_modules.to_a.slice(0..-2)
  650. assert modules.any?
  651. assert_difference 'EnabledModule.count', -1 do
  652. project.enabled_module_names = modules.collect(&:name)
  653. end
  654. project.reload
  655. # Ids should be preserved
  656. assert_equal project.enabled_module_ids.sort, modules.collect(&:id).sort
  657. end
  658. def test_copy_from_existing_project
  659. source_project = Project.find(1)
  660. copied_project = Project.copy_from(1)
  661. assert copied_project
  662. # Cleared attributes
  663. assert copied_project.id.blank?
  664. assert copied_project.name.blank?
  665. assert copied_project.identifier.blank?
  666. # Duplicated attributes
  667. assert_equal source_project.description, copied_project.description
  668. assert_equal source_project.trackers, copied_project.trackers
  669. # Default attributes
  670. assert_equal 1, copied_project.status
  671. end
  672. def test_copy_from_should_copy_enabled_modules
  673. source = Project.generate!
  674. source.enabled_module_names = %w(issue_tracking wiki)
  675. copy = Project.copy_from(source)
  676. copy.name = 'Copy'
  677. copy.identifier = 'copy'
  678. assert_difference 'EnabledModule.count', 2 do
  679. copy.save!
  680. end
  681. assert_equal 2, copy.reload.enabled_modules.count
  682. assert_equal 2, source.reload.enabled_modules.count
  683. end
  684. def test_activities_should_use_the_system_activities
  685. project = Project.find(1)
  686. assert_equal project.activities.to_a, TimeEntryActivity.where(:active => true).to_a
  687. assert_kind_of ActiveRecord::Relation, project.activities
  688. end
  689. def test_activities_should_use_the_project_specific_activities
  690. project = Project.find(1)
  691. overridden_activity = TimeEntryActivity.new({:name => "Project", :project => project})
  692. assert overridden_activity.save!
  693. assert project.activities.include?(overridden_activity), "Project specific Activity not found"
  694. assert_kind_of ActiveRecord::Relation, project.activities
  695. end
  696. def test_activities_should_not_include_the_inactive_project_specific_activities
  697. project = Project.find(1)
  698. overridden_activity = TimeEntryActivity.new({:name => "Project",
  699. :project => project,
  700. :parent => TimeEntryActivity.first,
  701. :active => false})
  702. assert overridden_activity.save!
  703. assert !project.activities.include?(overridden_activity), "Inactive Project specific Activity found"
  704. end
  705. def test_activities_should_not_include_project_specific_activities_from_other_projects
  706. project = Project.find(1)
  707. overridden_activity = TimeEntryActivity.new({:name => "Project", :project => Project.find(2)})
  708. assert overridden_activity.save!
  709. assert !project.activities.include?(overridden_activity),
  710. "Project specific Activity found on a different project"
  711. end
  712. def test_activities_should_handle_nils
  713. overridden_activity =
  714. TimeEntryActivity.new({:name => "Project", :project => Project.find(1),
  715. :parent => TimeEntryActivity.first})
  716. TimeEntryActivity.delete_all
  717. # No activities
  718. project = Project.find(1)
  719. assert project.activities.empty?
  720. # No system, one overridden
  721. assert overridden_activity.save!
  722. project.reload
  723. assert_equal [overridden_activity], project.activities
  724. end
  725. def test_activities_should_override_system_activities_with_project_activities
  726. project = Project.find(1)
  727. parent_activity = TimeEntryActivity.first
  728. overridden_activity =
  729. TimeEntryActivity.new({:name => "Project", :project => project,
  730. :parent => parent_activity})
  731. assert overridden_activity.save!
  732. assert project.activities.include?(overridden_activity),
  733. "Project specific Activity not found"
  734. assert !project.activities.include?(parent_activity),
  735. "System Activity found when it should have been overridden"
  736. end
  737. def test_activities_should_include_inactive_activities_if_specified
  738. project = Project.find(1)
  739. overridden_activity =
  740. TimeEntryActivity.new(
  741. {:name => "Project", :project => project,
  742. :parent => TimeEntryActivity.first, :active => false}
  743. )
  744. assert overridden_activity.save!
  745. assert project.activities(true).include?(overridden_activity), "Inactive Project specific Activity not found"
  746. end
  747. test 'activities should not include active System activities if the project has an override that is inactive' do
  748. project = Project.find(1)
  749. system_activity = TimeEntryActivity.find_by_name('Design')
  750. assert system_activity.active?
  751. overridden_activity =
  752. TimeEntryActivity.create!(:name => "Project", :project => project,
  753. :parent => system_activity, :active => false)
  754. assert overridden_activity.save!
  755. assert !project.activities.include?(overridden_activity),
  756. "Inactive Project specific Activity not found"
  757. assert !project.activities.include?(system_activity),
  758. "System activity found when the project has an inactive override"
  759. end
  760. def test_close_completed_versions
  761. Version.update_all("status = 'open'")
  762. project = Project.find(1)
  763. assert_not_nil project.versions.detect {|v| v.completed? && v.status == 'open'}
  764. assert_not_nil project.versions.detect {|v| !v.completed? && v.status == 'open'}
  765. project.close_completed_versions
  766. project.reload
  767. assert_nil project.versions.detect {|v| v.completed? && v.status != 'closed'}
  768. assert_not_nil project.versions.detect {|v| !v.completed? && v.status == 'open'}
  769. end
  770. test "#start_date should be nil if there are no issues on the project" do
  771. project = Project.generate!
  772. assert_nil project.start_date
  773. end
  774. test "#start_date should be nil when issues have no start date" do
  775. project = Project.generate!
  776. project.trackers << Tracker.generate!
  777. early = 7.days.ago.to_date
  778. Issue.generate!(:project => project, :start_date => nil)
  779. assert_nil project.start_date
  780. end
  781. test "#start_date should be the earliest start date of it's issues" do
  782. project = Project.generate!
  783. project.trackers << Tracker.generate!
  784. early = 7.days.ago.to_date
  785. Issue.generate!(:project => project, :start_date => Date.today)
  786. Issue.generate!(:project => project, :start_date => early)
  787. assert_equal early, project.start_date
  788. end
  789. test "#due_date should be nil if there are no issues on the project" do
  790. project = Project.generate!
  791. assert_nil project.due_date
  792. end
  793. test "#due_date should be nil if there are no issues with due dates" do
  794. project = Project.generate!
  795. project.trackers << Tracker.generate!
  796. Issue.generate!(:project => project, :due_date => nil)
  797. assert_nil project.due_date
  798. end
  799. test "#due_date should be the latest due date of it's issues" do
  800. project = Project.generate!
  801. project.trackers << Tracker.generate!
  802. future = 7.days.from_now.to_date
  803. Issue.generate!(:project => project, :due_date => future)
  804. Issue.generate!(:project => project, :due_date => Date.today)
  805. assert_equal future, project.due_date
  806. end
  807. test "#due_date should be the latest due date of it's versions" do
  808. project = Project.generate!
  809. future = 7.days.from_now.to_date
  810. project.versions << Version.generate!(:effective_date => future)
  811. project.versions << Version.generate!(:effective_date => Date.today)
  812. assert_equal future, project.due_date
  813. end
  814. test "#due_date should pick the latest date from it's issues and versions" do
  815. project = Project.generate!
  816. project.trackers << Tracker.generate!
  817. future = 7.days.from_now.to_date
  818. far_future = 14.days.from_now.to_date
  819. Issue.generate!(:project => project, :due_date => far_future)
  820. project.versions << Version.generate!(:effective_date => future)
  821. assert_equal far_future, project.due_date
  822. end
  823. test "#completed_percent with no versions should be 100" do
  824. project = Project.generate!
  825. assert_equal 100, project.completed_percent
  826. end
  827. test "#completed_percent with versions should return 0 if the versions have no issues" do
  828. project = Project.generate!
  829. Version.generate!(:project => project)
  830. Version.generate!(:project => project)
  831. assert_equal 0, project.completed_percent
  832. end
  833. test "#completed_percent with versions should return 100 if the version has only closed issues" do
  834. project = Project.generate!
  835. project.trackers << Tracker.generate!
  836. v1 = Version.generate!(:project => project)
  837. Issue.generate!(:project => project, :status => IssueStatus.find_by_name('Closed'),
  838. :fixed_version => v1)
  839. v2 = Version.generate!(:project => project)
  840. Issue.generate!(:project => project, :status => IssueStatus.find_by_name('Closed'),
  841. :fixed_version => v2)
  842. assert_equal 100, project.completed_percent
  843. end
  844. test "#completed_percent with versions should return the averaged completed percent of the versions (not weighted)" do
  845. project = Project.generate!
  846. project.trackers << Tracker.generate!
  847. v1 = Version.generate!(:project => project)
  848. Issue.generate!(:project => project, :status => IssueStatus.find_by_name('New'),
  849. :estimated_hours => 10, :done_ratio => 50, :fixed_version => v1)
  850. v2 = Version.generate!(:project => project)
  851. Issue.generate!(:project => project, :status => IssueStatus.find_by_name('New'),
  852. :estimated_hours => 10, :done_ratio => 50, :fixed_version => v2)
  853. assert_equal 50, project.completed_percent
  854. end
  855. test "#notified_users" do
  856. project = Project.generate!
  857. role = Role.generate!
  858. user_with_membership_notification = User.generate!(:mail_notification => 'selected')
  859. Member.create!(:project => project, :roles => [role],
  860. :principal => user_with_membership_notification, :mail_notification => true)
  861. all_events_user = User.generate!(:mail_notification => 'all')
  862. Member.create!(:project => project, :roles => [role], :principal => all_events_user)
  863. no_events_user = User.generate!(:mail_notification => 'none')
  864. Member.create!(:project => project, :roles => [role], :principal => no_events_user)
  865. only_my_events_user = User.generate!(:mail_notification => 'only_my_events')
  866. Member.create!(:project => project, :roles => [role], :principal => only_my_events_user)
  867. only_assigned_user = User.generate!(:mail_notification => 'only_assigned')
  868. Member.create!(:project => project, :roles => [role], :principal => only_assigned_user)
  869. only_owned_user = User.generate!(:mail_notification => 'only_owner')
  870. Member.create!(:project => project, :roles => [role], :principal => only_owned_user)
  871. assert project.notified_users.include?(user_with_membership_notification),
  872. "should include members with a mail notification"
  873. assert project.notified_users.include?(all_events_user),
  874. "should include users with the 'all' notification option"
  875. assert !project.notified_users.include?(no_events_user),
  876. "should not include users with the 'none' notification option"
  877. assert !project.notified_users.include?(only_my_events_user),
  878. "should not include users with the 'only_my_events' notification option"
  879. assert !project.notified_users.include?(only_assigned_user),
  880. "should not include users with the 'only_assigned' notification option"
  881. assert !project.notified_users.include?(only_owned_user),
  882. "should not include users with the 'only_owner' notification option"
  883. end
  884. def test_override_roles_without_builtin_group_memberships
  885. project = Project.generate!
  886. assert_equal [Role.anonymous], project.override_roles(Role.anonymous)
  887. assert_equal [Role.non_member], project.override_roles(Role.non_member)
  888. end
  889. def test_css_classes
  890. p = Project.new
  891. assert_kind_of String, p.css_classes
  892. assert_not_include 'archived', p.css_classes.split
  893. assert_not_include 'closed', p.css_classes.split
  894. assert_include 'public', p.css_classes.split
  895. end
  896. def test_css_classes_for_archived_project
  897. p = Project.new
  898. p.status = Project::STATUS_ARCHIVED
  899. assert_include 'archived', p.css_classes.split
  900. assert_include 'public', p.css_classes.split
  901. end
  902. def test_css_classes_for_closed_project
  903. p = Project.new
  904. p.status = Project::STATUS_CLOSED
  905. assert_include 'closed', p.css_classes.split
  906. assert_include 'public', p.css_classes.split
  907. end
  908. def test_css_classes_for_private_project
  909. p = Project.new
  910. p.is_public = false
  911. assert_not_include 'public', p.css_classes.split
  912. end
  913. def test_combination_of_visible_and_distinct_scopes_in_case_anonymous_group_has_memberships_should_not_error
  914. project = Project.find(1)
  915. member = Member.create!(:project => project, :principal => Group.anonymous,
  916. :roles => [Role.generate!])
  917. project.members << member
  918. assert_nothing_raised do
  919. Project.distinct.visible.to_a
  920. end
  921. end
  922. def test_safe_attributes_should_include_only_custom_fields_visible_to_user
  923. cf1 = ProjectCustomField.create!(:name => 'Visible field',
  924. :field_format => 'string',
  925. :visible => false, :role_ids => [1])
  926. cf2 = ProjectCustomField.create!(:name => 'Non visible field',
  927. :field_format => 'string',
  928. :visible => false, :role_ids => [3])
  929. user = User.find(2)
  930. project = Project.find(1)
  931. project.send(
  932. :safe_attributes=,
  933. {
  934. 'custom_field_values' => {
  935. cf1.id.to_s => 'value1', cf2.id.to_s => 'value2'
  936. }
  937. },
  938. user
  939. )
  940. assert_equal 'value1', project.custom_field_value(cf1)
  941. assert_nil project.custom_field_value(cf2)
  942. project.send(
  943. :safe_attributes=,
  944. {
  945. 'custom_fields' =>
  946. [
  947. {'id' => cf1.id.to_s, 'value' => 'valuea'},
  948. {'id' => cf2.id.to_s, 'value' => 'valueb'}
  949. ]
  950. },
  951. user
  952. )
  953. assert_equal 'valuea', project.custom_field_value(cf1)
  954. assert_nil project.custom_field_value(cf2)
  955. end
  956. def test_like_scope_should_escape_query
  957. project = Project.find 'ecookbook'
  958. r = Project.like('eco_k')
  959. assert_not_include project, r
  960. r = Project.like('eco%k')
  961. assert_not_include project, r
  962. project.update_column :name, 'Eco%kbook'
  963. r = Project.like('eco%k')
  964. assert_include project, r
  965. project.update_column :name, 'Eco_kbook'
  966. r = Project.like('eco_k')
  967. assert_include project, r
  968. end
  969. end