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.

issue_import_test.rb 7.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. # Redmine - project management software
  2. # Copyright (C) 2006-2017 Jean-Philippe Lang
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; either version 2
  7. # of the License, or (at your option) any later version.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with this program; if not, write to the Free Software
  16. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  17. require File.expand_path('../../test_helper', __FILE__)
  18. class IssueImportTest < ActiveSupport::TestCase
  19. fixtures :projects, :enabled_modules,
  20. :users, :email_addresses,
  21. :roles, :members, :member_roles,
  22. :issues, :issue_statuses,
  23. :trackers, :projects_trackers,
  24. :versions,
  25. :issue_categories,
  26. :enumerations,
  27. :workflows,
  28. :custom_fields,
  29. :custom_values,
  30. :custom_fields_projects,
  31. :custom_fields_trackers
  32. include Redmine::I18n
  33. def setup
  34. User.current = nil
  35. set_language_if_valid 'en'
  36. end
  37. def test_create_versions_should_create_missing_versions
  38. import = generate_import_with_mapping
  39. import.mapping.merge!('fixed_version' => '9', 'create_versions' => '1')
  40. import.save!
  41. version = new_record(Version) do
  42. assert_difference 'Issue.count', 3 do
  43. import.run
  44. end
  45. end
  46. assert_equal '2.1', version.name
  47. end
  48. def test_create_categories_should_create_missing_categories
  49. import = generate_import_with_mapping
  50. import.mapping.merge!('category' => '10', 'create_categories' => '1')
  51. import.save!
  52. category = new_record(IssueCategory) do
  53. assert_difference 'Issue.count', 3 do
  54. import.run
  55. end
  56. end
  57. assert_equal 'New category', category.name
  58. end
  59. def test_mapping_with_fixed_tracker
  60. import = generate_import_with_mapping
  61. import.mapping.merge!('tracker' => 'value:2')
  62. import.save!
  63. issues = new_records(Issue, 3) { import.run }
  64. assert_equal [2], issues.map(&:tracker_id).uniq
  65. end
  66. def test_mapping_with_mapped_tracker
  67. import = generate_import_with_mapping
  68. import.mapping.merge!('tracker' => '13')
  69. import.save!
  70. issues = new_records(Issue, 3) { import.run }
  71. assert_equal [1, 2, 1], issues.map(&:tracker_id)
  72. end
  73. def test_should_not_import_with_default_tracker_when_tracker_is_invalid
  74. Tracker.find_by_name('Feature request').update!(:name => 'Feature')
  75. import = generate_import_with_mapping
  76. import.mapping.merge!('tracker' => '13')
  77. import.save!
  78. import.run
  79. assert_equal 1, import.unsaved_items.count
  80. item = import.unsaved_items.first
  81. assert_include "Tracker cannot be blank", item.message
  82. end
  83. def test_status_should_be_set
  84. import = generate_import_with_mapping
  85. import.mapping.merge!('status' => '14')
  86. import.save!
  87. issues = new_records(Issue, 3) { import.run }
  88. assert_equal ['New', 'New', 'Assigned'], issues.map(&:status).map(&:name)
  89. end
  90. def test_parent_should_be_set
  91. import = generate_import_with_mapping
  92. import.mapping.merge!('parent_issue_id' => '5')
  93. import.save!
  94. issues = new_records(Issue, 3) { import.run }
  95. assert_nil issues[0].parent
  96. assert_equal issues[0].id, issues[1].parent_id
  97. assert_equal 2, issues[2].parent_id
  98. end
  99. def test_import_utf8_with_bom
  100. import = generate_import_with_mapping('import_issues_utf8_with_bom.csv')
  101. import.settings.merge!('encoding' => 'UTF-8')
  102. import.save
  103. issues = new_records(Issue,3) { import.run }
  104. assert_equal 3, issues.count
  105. end
  106. def test_backward_and_forward_reference_to_parent_should_work
  107. import = generate_import('import_subtasks.csv')
  108. import.settings = {
  109. 'separator' => ";", 'wrapper' => '"', 'encoding' => "UTF-8",
  110. 'mapping' => {'project_id' => '1', 'tracker' => '1', 'subject' => '2', 'parent_issue_id' => '3'}
  111. }
  112. import.save!
  113. root, child1, grandchild, child2 = new_records(Issue, 4) { import.run }
  114. assert_equal root, child1.parent
  115. assert_equal child2, grandchild.parent
  116. end
  117. def test_assignee_should_be_set
  118. import = generate_import_with_mapping
  119. import.mapping.merge!('assigned_to' => '11')
  120. import.save!
  121. issues = new_records(Issue, 3) { import.run }
  122. assert_equal [User.find(3), nil, nil], issues.map(&:assigned_to)
  123. end
  124. def test_user_custom_field_should_be_set
  125. field = IssueCustomField.generate!(:field_format => 'user', :is_for_all => true, :trackers => Tracker.all)
  126. import = generate_import_with_mapping
  127. import.mapping.merge!("cf_#{field.id}" => '11')
  128. import.save!
  129. issues = new_records(Issue, 3) { import.run }
  130. assert_equal '3', issues.first.custom_field_value(field)
  131. end
  132. def test_list_custom_field_should_be_set
  133. field = CustomField.find(1)
  134. field.tracker_ids = Tracker.all.ids
  135. field.save!
  136. import = generate_import_with_mapping
  137. import.mapping.merge!("cf_1" => '8')
  138. import.save!
  139. issues = new_records(Issue, 3) { import.run }
  140. assert_equal 'PostgreSQL', issues[0].custom_field_value(1)
  141. assert_equal 'MySQL', issues[1].custom_field_value(1)
  142. assert_equal '', issues.third.custom_field_value(1)
  143. end
  144. def test_multiple_list_custom_field_should_be_set
  145. field = CustomField.find(1)
  146. field.tracker_ids = Tracker.all.ids
  147. field.multiple = true
  148. field.save!
  149. import = generate_import_with_mapping
  150. import.mapping.merge!("cf_1" => '15')
  151. import.save!
  152. issues = new_records(Issue, 3) { import.run }
  153. assert_equal ['Oracle', 'PostgreSQL'], issues[0].custom_field_value(1).sort
  154. assert_equal ['MySQL'], issues[1].custom_field_value(1)
  155. assert_equal [''], issues.third.custom_field_value(1)
  156. end
  157. def test_is_private_should_be_set_based_on_user_locale
  158. import = generate_import_with_mapping
  159. import.mapping.merge!('is_private' => '6')
  160. import.save!
  161. issues = new_records(Issue, 3) { import.run }
  162. assert_equal [false, true, false], issues.map(&:is_private)
  163. end
  164. def test_dates_should_be_parsed_using_date_format_setting
  165. field = IssueCustomField.generate!(:field_format => 'date', :is_for_all => true, :trackers => Tracker.all)
  166. import = generate_import_with_mapping('import_dates.csv')
  167. import.settings.merge!('date_format' => Import::DATE_FORMATS[1])
  168. import.mapping.merge!('tracker' => 'value:1', 'subject' => '0', 'start_date' => '1', 'due_date' => '2', "cf_#{field.id}" => '3')
  169. import.save!
  170. issue = new_record(Issue) { import.run } # only 1 valid issue
  171. assert_equal "Valid dates", issue.subject
  172. assert_equal Date.parse('2015-07-10'), issue.start_date
  173. assert_equal Date.parse('2015-08-12'), issue.due_date
  174. assert_equal '2015-07-14', issue.custom_field_value(field)
  175. end
  176. def test_date_format_should_default_to_user_language
  177. user = User.generate!(:language => 'fr')
  178. import = Import.new
  179. import.user = user
  180. assert_nil import.settings['date_format']
  181. import.set_default_settings
  182. assert_equal '%d/%m/%Y', import.settings['date_format']
  183. end
  184. def test_run_should_remove_the_file
  185. import = generate_import_with_mapping
  186. file_path = import.filepath
  187. assert File.exists?(file_path)
  188. import.run
  189. assert !File.exists?(file_path)
  190. end
  191. def test_run_should_consider_project_shared_versions
  192. system_version = Version.generate!(:project_id => 2, :sharing => 'system', :name => '2.1')
  193. system_version.save!
  194. import = generate_import_with_mapping
  195. import.mapping.merge!('fixed_version' => '9')
  196. import.save!
  197. issues = new_records(Issue, 3) { import.run }
  198. assert [nil, 3, system_version.id], issues.map(&:fixed_version_id)
  199. end
  200. end