????
Your IP : 18.223.134.71
# frozen_string_literal: true
# Redmine - project management software
# Copyright (C) 2006-2023 Jean-Philippe Lang
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
require File.expand_path('../../test_helper', __FILE__)
class IssueImportTest < ActiveSupport::TestCase
fixtures :projects, :enabled_modules,
:users, :email_addresses,
:roles, :members, :member_roles,
:issues, :issue_statuses,
:trackers, :projects_trackers,
:versions,
:issue_categories,
:enumerations,
:workflows,
:custom_fields,
:custom_values,
:custom_fields_projects,
:custom_fields_trackers
include Redmine::I18n
def setup
User.current = nil
set_language_if_valid 'en'
end
def test_authorized
assert IssueImport.authorized?(User.find(1)) # admins
assert IssueImport.authorized?(User.find(2)) # has import_issues permission
assert !IssueImport.authorized?(User.find(3)) # does not have permission
end
def test_create_versions_should_create_missing_versions
import = generate_import_with_mapping
import.mapping.merge!('fixed_version' => '9', 'create_versions' => '1')
import.save!
version = new_record(Version) do
assert_difference 'Issue.count', 3 do
import.run
end
end
assert_equal '2.1', version.name
end
def test_create_categories_should_create_missing_categories
import = generate_import_with_mapping
import.mapping.merge!('category' => '10', 'create_categories' => '1')
import.save!
category = new_record(IssueCategory) do
assert_difference 'Issue.count', 3 do
import.run
end
end
assert_equal 'New category', category.name
end
def test_mapping_with_fixed_tracker
import = generate_import_with_mapping
import.mapping['tracker'] = 'value:2'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal [2], issues.map(&:tracker_id).uniq
end
def test_mapping_with_mapped_tracker
import = generate_import_with_mapping
import.mapping['tracker'] = '13'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal [1, 2, 1], issues.map(&:tracker_id)
end
def test_should_not_import_with_default_tracker_when_tracker_is_invalid
Tracker.find_by_name('Feature request').update!(:name => 'Feature')
import = generate_import_with_mapping
import.mapping['tracker'] = '13'
import.save!
import.run
assert_equal 1, import.unsaved_items.count
item = import.unsaved_items.first
assert_include "Tracker cannot be blank", item.message
end
def test_status_should_be_set
import = generate_import_with_mapping
import.mapping['status'] = '14'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal ['New', 'New', 'Assigned'], issues.map(&:status).map(&:name)
end
def test_parent_should_be_set
import = generate_import_with_mapping
import.mapping['parent_issue_id'] = '5'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_nil issues[0].parent
assert_equal issues[0].id, issues[1].parent_id
assert_equal 2, issues[2].parent_id
end
def test_import_utf8_with_bom
import = generate_import_with_mapping('import_issues_utf8_with_bom.csv')
import.settings['encoding'] = 'UTF-8'
import.save
issues = new_records(Issue, 3) {import.run}
assert_equal 3, issues.count
end
def test_backward_and_forward_reference_to_parent_should_work
import = generate_import('import_subtasks.csv')
import.settings = {
'separator' => ";", 'wrapper' => '"', 'encoding' => "UTF-8",
'mapping' => {'project_id' => '1', 'tracker' => '1', 'subject' => '2', 'parent_issue_id' => '3'}
}
import.save!
root, child1, grandchild, child2 = new_records(Issue, 4) {import.run}
assert_equal root, child1.parent
assert_equal child2, grandchild.parent
end
def test_references_with_unique_id
import = generate_import_with_mapping('import_subtasks_with_unique_id.csv')
import.settings['mapping'] = {'project_id' => '1', 'unique_id' => '0', 'tracker' => '1', 'subject' => '2', 'parent_issue_id' => '3', 'relation_follows' => '4'}
import.save!
red4, red3, red2, red1, blue1, blue2, blue3, blue4, green = new_records(Issue, 9) {import.run}
# future references
assert_equal red1, red2.parent
assert_equal red3, red4.parent
assert IssueRelation.where('issue_from_id' => red2.id, 'issue_to_id' => red3.id, 'delay' => 1, 'relation_type' => 'precedes').present?
# past references
assert_equal blue1, blue2.parent
assert_equal blue3, blue4.parent
assert IssueRelation.where('issue_from_id' => blue2.id, 'issue_to_id' => blue3.id, 'delay' => 1, 'relation_type' => 'precedes').present?
assert_equal issues(:issues_001), green.parent
assert IssueRelation.where('issue_from_id' => issues(:issues_002).id, 'issue_to_id' => green.id, 'delay' => 3, 'relation_type' => 'precedes').present?
end
def test_follow_relation
import = generate_import_with_mapping('import_subtasks.csv')
import.settings['mapping'] = {'project_id' => '1', 'tracker' => '1', 'subject' => '2', 'relation_relates' => '4'}
import.save!
one, one_one, one_two_one, one_two = new_records(Issue, 4) {import.run}
assert_equal 2, one.relations.count
assert one.relations.all? {|r| r.relation_type == 'relates'}
assert one.relations.any? {|r| r.other_issue(one) == one_one}
assert one.relations.any? {|r| r.other_issue(one) == one_two}
assert_equal 2, one_one.relations.count
assert one_one.relations.all? {|r| r.relation_type == 'relates'}
assert one_one.relations.any? {|r| r.other_issue(one_one) == one}
assert one_one.relations.any? {|r| r.other_issue(one_one) == one_two}
assert_equal 3, one_two.relations.count
assert one_two.relations.all? {|r| r.relation_type == 'relates'}
assert one_two.relations.any? {|r| r.other_issue(one_two) == one}
assert one_two.relations.any? {|r| r.other_issue(one_two) == one_one}
assert one_two.relations.any? {|r| r.other_issue(one_two) == one_two_one}
assert_equal 1, one_two_one.relations.count
assert one_two_one.relations.all? {|r| r.relation_type == 'relates'}
assert one_two_one.relations.any? {|r| r.other_issue(one_two_one) == one_two}
end
def test_delayed_relation
import = generate_import_with_mapping('import_subtasks.csv')
import.settings['mapping'] = {'project_id' => '1', 'tracker' => '1', 'subject' => '2', 'relation_precedes' => '5'}
import.save!
one, one_one, one_two_one, one_two = new_records(Issue, 4) {import.run}
assert_equal 2, one.relations_to.count
assert one.relations_to.all? {|r| r.relation_type == 'precedes'}
assert one.relations_to.any? {|r| r.issue_from == one_one && r.delay == 2}
assert one.relations_to.any? {|r| r.issue_from == one_two && r.delay == 1}
assert_equal 1, one_one.relations_from.count
assert one_one.relations_from.all? {|r| r.relation_type == 'precedes'}
assert one_one.relations_from.any? {|r| r.issue_to == one && r.delay == 2}
assert_equal 1, one_two.relations_to.count
assert one_two.relations_to.all? {|r| r.relation_type == 'precedes'}
assert one_two.relations_to.any? {|r| r.issue_from == one_two_one && r.delay == -1}
assert_equal 1, one_two.relations_from.count
assert one_two.relations_from.all? {|r| r.relation_type == 'precedes'}
assert one_two.relations_from.any? {|r| r.issue_to == one && r.delay == 1}
assert_equal 1, one_two_one.relations_from.count
assert one_two_one.relations_from.all? {|r| r.relation_type == 'precedes'}
assert one_two_one.relations_from.any? {|r| r.issue_to == one_two && r.delay == -1}
end
def test_parent_and_follows_relation
import = generate_import_with_mapping('import_subtasks_with_relations.csv')
import.settings['mapping'] = {
'project_id' => '1',
'tracker' => '1',
'subject' => '2',
'start_date' => '3',
'due_date' => '4',
'parent_issue_id' => '5',
'relation_follows' => '6'
}
import.save!
second, first, parent, third = assert_difference('IssueRelation.count', 2) {new_records(Issue, 4) {import.run}}
# Parent relations
assert_equal parent, first.parent
assert_equal parent, second.parent
assert_equal parent, third.parent
# Issue relations
assert IssueRelation.where(
:issue_from_id => first.id,
:issue_to_id => second.id,
:relation_type => 'precedes',
:delay => 1).present?
assert IssueRelation.where(
:issue_from_id => second.id,
:issue_to_id => third.id,
:relation_type => 'precedes',
:delay => 1).present?
# Checking dates, because they might act weird, when relations are added
assert_equal Date.new(2020, 1, 1), parent.start_date
assert_equal Date.new(2020, 2, 3), parent.due_date
assert_equal Date.new(2020, 1, 1), first.start_date
assert_equal Date.new(2020, 1, 10), first.due_date
assert_equal Date.new(2020, 1, 14), second.start_date
assert_equal Date.new(2020, 1, 21), second.due_date
assert_equal Date.new(2020, 1, 23), third.start_date
assert_equal Date.new(2020, 2, 3), third.due_date
end
def test_import_with_relations_and_invalid_issue_should_not_fail
import = generate_import_with_mapping('import_issues_with_relation_and_invalid_issues.csv')
import.settings['mapping'] = {
'project_id' => '1',
'tracker' => '1',
'subject' => '2',
'status' => '3',
'relation_relates' => '4',
}
import.save!
first, second, third, fourth = new_records(Issue, 4) {import.run}
assert_equal 1, import.unsaved_items.count
item = import.unsaved_items.first
assert_include "Subject cannot be blank", item.message
assert_equal 1, first.relations_from.count
assert_equal 1, second.relations_to.count
end
def test_assignee_should_be_set
import = generate_import_with_mapping
import.mapping['assigned_to'] = '11'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal [User.find(3), nil, nil], issues.map(&:assigned_to)
end
def test_user_custom_field_should_be_set
field = IssueCustomField.generate!(:field_format => 'user', :is_for_all => true, :trackers => Tracker.all)
import = generate_import_with_mapping
import.mapping["cf_#{field.id}"] = '11'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal '3', issues.first.custom_field_value(field)
end
def test_list_custom_field_should_be_set
field = CustomField.find(1)
field.tracker_ids = Tracker.all.ids
field.save!
import = generate_import_with_mapping
import.mapping["cf_1"] = '8'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal 'PostgreSQL', issues[0].custom_field_value(1)
assert_equal 'MySQL', issues[1].custom_field_value(1)
assert_equal '', issues.third.custom_field_value(1)
end
def test_multiple_list_custom_field_should_be_set
field = CustomField.find(1)
field.tracker_ids = Tracker.all.ids
field.multiple = true
field.save!
import = generate_import_with_mapping
import.mapping["cf_1"] = '15'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal ['Oracle', 'PostgreSQL'], issues[0].custom_field_value(1).sort
assert_equal ['MySQL'], issues[1].custom_field_value(1)
assert_equal [''], issues.third.custom_field_value(1)
end
def test_is_private_should_be_set_based_on_user_locale
import = generate_import_with_mapping
import.mapping['is_private'] = '6'
import.save!
issues = new_records(Issue, 3) {import.run}
assert_equal [false, true, false], issues.map(&:is_private)
end
def test_dates_should_be_parsed_using_date_format_setting
field = IssueCustomField.generate!(:field_format => 'date', :is_for_all => true, :trackers => Tracker.all)
import = generate_import_with_mapping('import_dates.csv')
import.settings['date_format'] = Import::DATE_FORMATS[1]
import.mapping.merge!('tracker' => 'value:1', 'subject' => '0', 'start_date' => '1', 'due_date' => '2', "cf_#{field.id}" => '3')
import.save!
issue = new_record(Issue) {import.run} # only 1 valid issue
assert_equal "Valid dates", issue.subject
assert_equal Date.parse('2015-07-10'), issue.start_date
assert_equal Date.parse('2015-08-12'), issue.due_date
assert_equal '2015-07-14', issue.custom_field_value(field)
# Tests using other date formats
import = generate_import_with_mapping('import_dates_ja.csv')
import.settings['date_format'] = Import::DATE_FORMATS[3]
import.mapping.merge!('tracker' => 'value:1', 'subject' => '0', 'start_date' => '1')
import.save!
issue = new_record(Issue) {import.run}
assert_equal Date.parse('2019-05-28'), issue.start_date
end
def test_date_format_should_default_to_user_language
user = User.generate!(:language => 'fr')
import = Import.new
import.user = user
assert_nil import.settings['date_format']
import.set_default_settings
assert_equal '%d/%m/%Y', import.settings['date_format']
end
def test_run_should_remove_the_file
import = generate_import_with_mapping
file_path = import.filepath
assert File.exist?(file_path)
import.run
assert !File.exist?(file_path)
end
def test_run_should_consider_project_shared_versions
system_version = Version.generate!(:project_id => 2, :sharing => 'system', :name => '2.1')
system_version.save!
import = generate_import_with_mapping
import.mapping['fixed_version'] = '9'
import.save!
issues = new_records(Issue, 3) {import.run}
assert [nil, 3, system_version.id], issues.map(&:fixed_version_id)
end
def test_set_default_settings_with_project_id
import = Import.new
import.set_default_settings(:project_id => 3)
assert_equal 3, import.mapping['project_id']
end
def test_set_default_settings_with_project_identifier
import = Import.new
import.set_default_settings(:project_id => 'ecookbook')
assert_equal 1, import.mapping['project_id']
end
def test_set_default_settings_without_project_id
import = Import.new
import.set_default_settings
assert_empty import.mapping
end
def test_set_default_settings_with_invalid_project_should_not_fail
import = Import.new
import.set_default_settings(:project_id => 'abc')
assert_empty import.mapping
end
def test_set_default_settings_should_guess_encoding
import = generate_import('import_iso8859-1.csv')
user = User.generate!(:language => 'ja')
import.user = user
assert_equal 'CP932', lu(user, :general_csv_encoding)
with_settings :repositories_encodings => 'UTF-8,ISO-8859-1' do
import.set_default_settings
guessed_encoding = import.settings['encoding']
assert_equal 'ISO-8859-1', guessed_encoding
end
with_settings :repositories_encodings => 'UTF-8,iso8859-1' do
import.set_default_settings
guessed_encoding = import.settings['encoding']
assert_equal 'ISO-8859-1', guessed_encoding
assert_includes Setting::ENCODINGS, guessed_encoding
end
end
def test_set_default_settings_should_use_general_csv_encoding_when_cannnot_guess_encoding
import = generate_import('import_iso8859-1.csv')
user = User.generate!(:language => 'ja')
import.user = user
with_settings :repositories_encodings => 'UTF-8' do
import.set_default_settings
guessed_encoding = import.settings['encoding']
assert_equal 'CP932', lu(user, :general_csv_encoding)
assert_equal 'CP932', guessed_encoding
end
end
end