ImportExportTicket
10.1.2
Rother OSS GmbH
https://rother-oss.com/
GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007
Fix article identification.
ImportExport plugin for tickets.
Ticket backend for the ImportExport module.
10.1.x
ImportExport
2022-09-05 06:10:23
opms
<?xml version="1.0" encoding="utf-8" ?>
<otobo_config version="2.0" init="Application">
    <Setting Name="ImportExport::ObjectBackendRegistration###Ticket" Required="0" Valid="1">
        <Description Translatable="1">Object backend module registration for the import/export module.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::ImportExport::ObjectBackend::Ticket</Item>
                <Item Key="Name">Ticket</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::SynchronizeWithForeignDB" Required="0" Valid="0">
        <Description Translatable="1">If activated additional data such as the history and links will be read from a foreign DB containing the exported tickets and added to the imported tickets on this system. This is only available for created, not for updated tickets.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="DatabaseDSN"></Item>
                <Item Key="DatabaseUser"></Item>
                <Item Key="DatabasePw"></Item>
                <Item Key="Type"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###001-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key">Queue</Item>
                <Item Key="Map">
                    <Hash>
                        <Item Key="OtherSystemMisc">Misc</Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###002-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key">Article_DynamicField_NoMulti</Item>
                <Item Key="Map">
                    <Hash>
                        <Item Key="OtherSystem5">7</Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###003-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###004-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###005-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###006-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###007-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###008-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###009-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###010-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###011-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###012-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###013-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###014-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###015-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###016-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###017-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###018-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###019-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###020-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###021-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###022-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###023-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###024-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::Ticket::ImportValueMap###025-Custom" Required="0" Valid="0">
        <Description Translatable="1">Value map. Define a key and a value map from import file to OTOBO.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::Ticket::ImportValueMaps</Navigation>
        <Value>
            <Hash>
                <Item Key="Key"></Item>
                <Item Key="Map">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
</otobo_config>

# --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2022 Rother OSS GmbH, https://otobo.de/
# --
# 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 3 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, see <https://www.gnu.org/licenses/>.
# --

package Kernel::System::ImportExport::ObjectBackend::Ticket;

use strict;
use warnings;

use Kernel::Language qw(Translatable);
use Kernel::System::VariableCheck qw(:all);

use Encode;
use MIME::Base64 qw(encode_base64 decode_base64);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::DB',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::ImportExport',
    'Kernel::System::LinkObject',
    'Kernel::System::Lock',
    'Kernel::System::Log',
    'Kernel::System::Priority',
    'Kernel::System::Queue',
    'Kernel::System::SLA',
    'Kernel::System::Service',
    'Kernel::System::State',
    'Kernel::System::Ticket',
    'Kernel::System::Ticket::Article',
    'Kernel::System::Type',
    'Kernel::System::User',
);

=head1 NAME

Kernel::System::ImportExport::ObjectBackend::Ticket - import/export backend for Tickets

=head1 DESCRIPTION

All functions to import and export tickets.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $BackendObject = $Kernel::OM->Get('Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem');

=cut

sub new {
    my ( $Type, %Param ) = @_;

    # allocate new hash for object
    my $Self = {
        ConfigObject           => $Kernel::OM->Get('Kernel::Config'),
        ImportExportObject     => $Kernel::OM->Get('Kernel::System::ImportExport'),
        TicketIDRelation       => {},
        TicketNumberIDRelation => {},
    };
    bless( $Self, $Type );

    return $Self;
}

=head2 ObjectAttributesGet()

get the object attributes of an object as a ref to an array of hash references

    my $Attributes = $ObjectBackend->ObjectAttributesGet(
        UserID => 1,
    );

=cut

sub ObjectAttributesGet {
    my ( $Self, %Param ) = @_;

    # check needed object
    if ( !$Param{UserID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need UserID!',
        );
        return;
    }

    my %QueueList = $Kernel::OM->Get('Kernel::System::Queue')->QueueList();
    my %StateList = $Kernel::OM->Get('Kernel::System::State')->StateList(
        UserID => 1,
    );
    my %PriorityList = $Kernel::OM->Get('Kernel::System::Priority')->PriorityList(
        Valid => 1,
    );
    my %LockList = $Kernel::OM->Get('Kernel::System::Lock')->LockList(
        UserID => 1,
    );
    my %UserList = $Kernel::OM->Get('Kernel::System::User')->UserList();

    my @Attributes = (
        {
            Key   => 'QueueID',
            Name  => 'Default Queue',
            Input => {
                Type        => 'Selection',
                Data        => \%QueueList,
                Required    => 1,
                Translation => 0,
                Class       => 'Modernize',
            },
        },
    );

    if ( $Self->{ConfigObject}->Get('Ticket::Type') ) {
        my %TypeList = $Kernel::OM->Get('Kernel::System::Type')->TypeList(
            Valid => 1,
        );

        push @Attributes, {
            Key   => 'TypeID',
            Name  => 'Default Type',
            Input => {
                Type        => 'Selection',
                Data        => \%TypeList || {},
                Required    => 1,
                Translation => 1,
                Class       => 'Modernize',
            },
        };
    }

    if ( $Self->{ConfigObject}->Get('Ticket::Service') ) {
        my %ServiceList = $Kernel::OM->Get('Kernel::System::Service')->ServiceList(
            UserID => 1,
        );

        push @Attributes, {
            Key   => 'ServiceID',
            Name  => 'Default Service',
            Input => {
                Type         => 'Selection',
                Data         => \%ServiceList || {},
                Required     => 0,
                Translation  => 0,
                PossibleNone => 1,
                Class        => 'Modernize',
            },
        };

        my %SLAList = $Kernel::OM->Get('Kernel::System::SLA')->SLAList(
            UserID => 1,
        );

        push @Attributes, {
            Key   => 'SLAID',
            Name  => 'Default SLA',
            Input => {
                Type         => 'Selection',
                Data         => \%SLAList || {},
                Required     => 0,
                Translation  => 0,
                PossibleNone => 1,
                Class        => 'Modernize',
            },
        };
    }

    my %SenderType = map { $_ => $_ } qw(agent customer system);

    push @Attributes, (
        {
            Key   => 'StateID',
            Name  => 'Default state',
            Input => {
                Type        => 'Selection',
                Data        => \%StateList,
                Required    => 1,
                Translation => 1,
                Class       => 'Modernize',
            },
        },
        {
            Key   => 'PriorityID',
            Name  => 'Default priority',
            Input => {
                Type        => 'Selection',
                Data        => \%PriorityList,
                Required    => 1,
                Translation => 1,
                Class       => 'Modernize',
            },
        },
        {
            Key   => 'OwnerID',
            Name  => 'Default owner',
            Input => {
                Type         => 'Selection',
                Data         => \%UserList,
                Required     => 1,
                Translation  => 0,
                Class        => 'Modernize',
            },
        },
        {
            Key   => 'ResponsibleID',
            Name  => 'Default responsible',
            Input => {
                Type         => 'Selection',
                Data         => \%UserList,
                Required     => 0,
                PossibleNone => 1,
                Translation  => 0,
                Class        => 'Modernize',
            },
        },
        {
            Key   => 'LockID',
            Name  => 'Default lock',
            Input => {
                Type        => 'Selection',
                Data        => \%LockList,
                Required    => 1,
                Translation => 1,
                Class       => 'Modernize',
            },
        },
        {
            Key   => 'CustomerID',
            Name  => 'Default CustomerID',
            Input => {
                Type      => 'Text',
                Required  => 0,
                Size      => 50,
                MaxLength => 250,
            },
        },
        {
            Key   => 'CustomerUserID',
            Name  => 'Default CustomerUserID',
            Input => {
                Type      => 'Text',
                Required  => 0,
                Size      => 50,
                MaxLength => 250,
            },
        },
        {
            Key   => 'ArchiveFlag',
            Name  => 'Default ArchiveFlag',
            Input => {
                Type         => 'Selection',
                Data         => { map { $_ => $_ } qw( y n ) },
                Required     => 1,
                Translation  => 0,
                Class        => 'Modernize',
                ValueDefault => 'n',
            },
        },
        {
            Key   => 'IncludeArticles',
            Name  => 'Import/Export articles',
            Input => {
                Type => 'Checkbox',
            },
        },
        {
            Key   => 'ArticleBackend',
            Name  => 'Default Backend',
            Input => {
                Type         => 'Selection',
                Data         => { map { $_ => $_ } qw( Email Phone Internal ) },
                Required     => 1,
                Translation  => 0,
                Class        => 'Modernize',
                ValueDefault => 'n',
            },
        },
        {
            Key   => 'Subject',
            Name  => 'Default subject',
            Input => {
                Type         => 'Text',
                Required     => 0,
                Size         => 50,
                MaxLength    => 250,
                ValueDefault => $Self->{ConfigObject}->Get('TicketImport::DefaultSubject'),
            },
        },
        {
            Key   => 'Body',
            Name  => 'Default body',
            Input => {
                Type         => 'Text',
                Required     => 0,
                Size         => 50,
                MaxLength    => 250,
                ValueDefault => $Self->{ConfigObject}->Get('TicketImport::DefaultBody'),
            },
        },
        {
            Key   => 'SenderType',
            Name  => 'Default sender type',
            Input => {
                Type     => 'Selection',
                Data     => \%SenderType,
                Required => 1,
                Class    => 'Modernize',
            },
        },
        {
            Key   => 'IsVisibleToCustomer',
            Name  => 'Default is visible to customer',
            Input => {
                Type => 'Checkbox',
            },
        },
        {
            Key   => 'ArticleSeparateLines',
            Name  => 'Store articles on separate lines indicated by a blank first entry',
            Input => {
                Type => 'Checkbox',
            },
        },
        {
            Key   => 'IncludeAttachments',
            Name  => 'Import/Export attachments (as the last entries per line)',
            Input => {
                Type => 'Checkbox',
            },
        },
        {
            Key   => 'EmptyFieldsLeaveTheOldValues',
            Name  => 'Empty fields indicate that the current values are kept',
            Input => {
                Type => 'Checkbox',
            },
        },
    );

    return \@Attributes;
}

=head2 MappingObjectAttributesGet()

get the mapping attributes of an object as array/hash reference

    my $Attributes = $ObjectBackend->MappingObjectAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

sub MappingObjectAttributesGet {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    for my $Argument (qw(TemplateID UserID)) {
        if ( !$Param{$Argument} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # get object data
    my $ObjectData = $Self->{ImportExportObject}->ObjectDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    return [] if !$ObjectData;
    return [] if ref $ObjectData ne 'HASH';

    my @ElementList = map { { Key => $_, Value => $_ } }
        qw( TicketID TicketNumber Title Type TypeID Queue QueueID Service ServiceID SLA
        SLAID State StateID Priority PriorityID CustomerID CustomerUserID Owner OwnerID Lock
        LockID Responsible ResponsibleID ArchiveFlag Created );

    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    # columns for dynamic fields
    my $DynFieldList = $DynamicFieldObject->DynamicFieldList(
        ObjectType => 'Ticket',
        ResultType => 'HASH',
    );
    push @ElementList, map { { Key => "DynamicField_$_", Value => "DynamicField_$_" } } sort values %{$DynFieldList};

    if ( $ObjectData->{IncludeArticles} ) {
        if ( $ObjectData->{ArticleSeparateLines} ) {
            push @ElementList, {
                Key   => 'Article_TicketID',
                Value => 'Article_TicketID',
            };
        }

        # columns for articles
        push @ElementList, map { { Key => "Article_$_", Value => "Article_$_" } }
            qw( ArticleID ArticleBackend From To Cc Bcc Subject Body SenderType IsVisibleForCustomer
            MessageID ReplyTo InReplyTo References Charset MimeType PlainEmail CreateTime );

        # columns for article dynamic fields
        my $DynFieldList = $DynamicFieldObject->DynamicFieldList(
            ObjectType => 'Article',
            ResultType => 'HASH',
        );
        push @ElementList, map { { Key => "Article_DynamicField_$_", Value => "Article_DynamicField_$_" } } values %{$DynFieldList};
    }

    my $Attributes = [
        {
            Key   => 'Key',
            Name  => Translatable('Key'),
            Input => {
                Type         => 'Selection',
                Data         => \@ElementList,
                Required     => 1,
                Translation  => 0,
                PossibleNone => 1,
                Class        => 'Modernize',
            },
        },
        {
            Key   => 'Identifier',
            Name  => Translatable('Identifier'),
            Input => {
                Type => 'Checkbox',
            },
        },
    ];

    return $Attributes;
}

=head2 SearchAttributesGet()

get the search object attributes of an object as array/hash reference

    my $AttributeList = $ObjectBackend->SearchAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

sub SearchAttributesGet {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    for my $Argument (qw(TemplateID UserID)) {
        if ( !$Param{$Argument} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # get object data
    my $ObjectData = $Kernel::OM->Get('Kernel::System::ImportExport')->ObjectDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    return [] if !$ObjectData;
    return [] if ref $ObjectData ne 'HASH';

    my %QueueList = $Kernel::OM->Get('Kernel::System::Queue')->QueueList();

    my %StateList = $Kernel::OM->Get('Kernel::System::State')->StateList(
        UserID => 1,
    );

    my %PriorityList = $Kernel::OM->Get('Kernel::System::Priority')->PriorityList(
        Valid => 1,
    );

    my @Attributes = (
        {
            Key   => 'QueueIDs',
            Name  => 'Queue',
            Input => {
                Type         => 'Selection',
                Data         => \%QueueList,
                Required     => 0,
                Translation  => 0,
                PossibleNone => 1,
                Size         => 5,
                Multiple     => 1,
                Class        => 'Modernize',
            },
        },
    );

    if ( $Self->{ConfigObject}->Get('Ticket::Type') ) {
        my %TypeList = $Kernel::OM->Get('Kernel::System::Type')->TypeList(
            Valid => 1,
        );

        push @Attributes, {
            Key   => 'TypeIDs',
            Name  => 'Type',
            Input => {
                Type         => 'Selection',
                Data         => \%TypeList || {},
                Required     => 0,
                Translation  => 1,
                PossibleNone => 1,
                Size         => 5,
                Multiple     => 1,
                Class        => 'Modernize',
            },
        };
    }

    if ( $Self->{ConfigObject}->Get('Ticket::Service') ) {
        my %ServiceList = $Kernel::OM->Get('Kernel::System::Service')->ServiceList(
            UserID => 1,
        );

        push @Attributes, {
            Key   => 'ServiceIDs',
            Name  => 'Service',
            Input => {
                Type         => 'Selection',
                Data         => \%ServiceList || {},
                Required     => 0,
                Translation  => 0,
                PossibleNone => 1,
                Size         => 5,
                Multiple     => 1,
                Class        => 'Modernize',
            },
        };

        my %SLAList = $Kernel::OM->Get('Kernel::System::SLA')->SLAList(
            UserID => 1,
        );

        push @Attributes, {
            Key   => 'SLAIDs',
            Name  => 'SLA',
            Input => {
                Type         => 'Selection',
                Data         => \%SLAList || {},
                Required     => 0,
                Translation  => 0,
                PossibleNone => 1,
                Size         => 5,
                Multiple     => 1,
                Class        => 'Modernize',
            },
        };
    }

    push @Attributes, (
        {
            Key   => 'StateIDs',
            Name  => 'State',
            Input => {
                Type         => 'Selection',
                Data         => \%StateList,
                Required     => 0,
                PossibleNone => 1,
                Translation  => 1,
                Size         => 5,
                Multiple     => 1,
                Class        => 'Modernize',
            },
        },
        {
            Key   => 'PriorityIDs',
            Name  => 'Priority',
            Input => {
                Type         => 'Selection',
                Data         => \%PriorityList,
                Required     => 0,
                PossibleNone => 1,
                Translation  => 1,
                Size         => 5,
                Multiple     => 1,
                Class        => 'Modernize',
            },
        },
        {
            Key   => 'CustomerID',
            Name  => 'CustomerID',
            Input => {
                Type      => 'Text',
                Size      => 50,
                MaxLength => 250,
            },
        },
        {
            Key   => 'TicketCreateTimeOlderMinutes',
            Name  => 'Ticket Create Time (older) [min]',
            Input => {
                Type      => 'Text',
                Size      => 50,
                MaxLength => 250,
            },
        },
        {
            Key   => 'TicketCreateTimeNewerMinutes',
            Name  => 'Ticket Create Time (newer) [min]',
            Input => {
                Type      => 'Text',
                Size      => 50,
                MaxLength => 250,
            },
        },
        {
            Key   => 'TicketChangeTimeOlderMinutes',
            Name  => 'Ticket Change Time (older) [min]',
            Input => {
                Type      => 'Text',
                Size      => 50,
                MaxLength => 50,
            },
        },
        {
            Key   => 'TicketChangeTimeNewerMinutes',
            Name  => 'Ticket Change Time (newer) [min]',
            Input => {
                Type      => 'Text',
                Size      => 50,
                MaxLength => 50,
            },
        },
    );

    my %DateRestrictions = (
        TicketCreateTimeOlderDate     => 'Ticket Create Time (before)',
        TicketCreateTimeNewerDate     => 'Ticket Create Time (after)',
        TicketLastChangeTimeOlderDate => 'Ticket Last Change Time (before)',
        TicketLastChangeTimeNewerDate => 'Ticket Last Change Time (after)',
    );

    for my $Key ( keys %DateRestrictions ) {
        push @Attributes, {
            Key   => $Key,
            Name  => $DateRestrictions{$Key},
            Input => {
                Type     => 'DateTime',
                Optional => 1,
            },
        };
    }

    return \@Attributes;
}

=head2 ExportDataGet()

get export data as C<2D-array-hash> reference

    my $ExportData = $ObjectBackend->ExportDataGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

sub ExportDataGet {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    for my $Argument (qw(TemplateID UserID)) {
        if ( !$Param{$Argument} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # get object data
    my $ObjectData = $Self->{ImportExportObject}->ObjectDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check object data
    if ( !$ObjectData || ref $ObjectData ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No object data found for the template id $Param{TemplateID}",
        );
        return;
    }

    # get the mapping list
    my $MappingList = $Self->{ImportExportObject}->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check the mapping list
    if ( !$MappingList || ref $MappingList ne 'ARRAY' || !@{$MappingList} ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No valid mapping list found for the template id $Param{TemplateID}",
        );
        return;
    }

    # create the mapping object list
    my @MappingObjectList;
    for my $MappingID ( @{$MappingList} ) {

        # get mapping object data
        my $MappingObjectData = $Self->{ImportExportObject}->MappingObjectDataGet(
            MappingID => $MappingID,
            UserID    => $Param{UserID},
        );

        # check mapping object data
        if ( !$MappingObjectData || ref $MappingObjectData ne 'HASH' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "No valid mapping list found for the template id $Param{TemplateID}",
            );
            return;
        }

        push @MappingObjectList, $MappingObjectData;
    }

    # get search data
    my $SearchData = $Kernel::OM->Get('Kernel::System::ImportExport')->SearchDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    my %IsSelection = map { $_ => 1 } qw( TypeIDs QueueIDs ServiceIDs SLAIDs StateIDs PriorityIDs CustomerID );

    my %SearchDataPrepared;
    KEY:
    for my $Key ( keys $SearchData->%* ) {
        next KEY if !defined $SearchData->{$Key};

        $SearchDataPrepared{$Key} = $IsSelection{$Key} ? [ split '#####', $SearchData->{$Key} ] : $SearchData->{$Key};
    }

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    # Search all Ticket_IDs
    my @TicketIDs = sort { $a <=> $b } $TicketObject->TicketSearch(
        %SearchDataPrepared,
        Result => 'ARRAY',
        UserID => 1,
    );

    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

    my @ExportData;
    my %TicketData;
    my %ArticleData;

    my @TicketMapping;
    my @ArticleMapping;
    my $TicketMappingLast;
    my $ArticleMappingLast = 0;    # in case of separate lines an empty entry will precede (thus not -1)
    my $StorePlainEmail    = 0;

    for my $i ( 0 .. $#MappingObjectList ) {

        # handle empty key
        if ( !$MappingObjectList[$i]{Key} ) {
            $TicketMapping[$i]{Key} = '';
            if ( !$ObjectData->{ArticleSeparateLines} ) {
                $TicketMappingLast = $i;
            }
        }

        # article keys
        elsif ( $MappingObjectList[$i]{Key} =~ /^Article_(.+)$/ ) {
            if ( $ObjectData->{ArticleSeparateLines} ) {
                push @ArticleMapping, $MappingObjectList[$i];
                $ArticleMapping[ ++$ArticleMappingLast ]{Key} = $1;
                $TicketMapping[$i]{Key} = '';
            }
            else {
                $ArticleMapping[$i]      = $MappingObjectList[$i];
                $ArticleMapping[$i]{Key} = $1;
                $ArticleMappingLast      = $i;
            }

            if ( $1 eq 'PlainEmail' ) {
                $StorePlainEmail = 1;
            }
        }

        # ticket keys
        else {
            $TicketMapping[$i] = $MappingObjectList[$i];
            $TicketMappingLast = $i;
        }

    }

    # export tickets ...
    for my $TicketID (@TicketIDs) {

        %TicketData = $TicketObject->TicketGet(
            TicketID      => $TicketID,
            UserID        => 1,
            DynamicFields => 1,
        );

        # add data to the export data array
        my @TicketItem;
        for my $i ( 0 .. $TicketMappingLast ) {

            # extract key
            my $Key = $TicketMapping[$i]{Key};

            $TicketItem[$i] = $Key && defined $TicketData{$Key}
                ? IsArrayRefWithData( $TicketData{$Key} ) ? join( '###', ( map { encode_base64( Encode::encode( 'UTF-8', $_ ) ) } $TicketData{$Key}->@* ) )
                : $TicketData{$Key} : '';
        }

        if ( !$ObjectData->{IncludeArticles} || $ObjectData->{ArticleSeparateLines} ) {
            push @ExportData, \@TicketItem;
        }

        if ( $ObjectData->{IncludeArticles} ) {

            # Get article data.
            my @Articles = $ArticleObject->ArticleList(
                TicketID => $TicketID,
            );

            my $ArticleCount = 0;
            for my $Article (@Articles) {

                my $ArticleBackendObject = $ArticleObject->BackendForArticle( %{$Article} );

                my %ArticleFull = $ArticleBackendObject->ArticleGet(
                    TicketID      => $TicketID,
                    ArticleID     => $Article->{ArticleID},
                    DynamicFields => 1,
                    UserID        => 1,
                );

                $ArticleFull{ArticleBackend} = $ArticleBackendObject->ChannelNameGet();

                if ( $StorePlainEmail && $ArticleBackendObject->can('ArticlePlain') ) {
                    $ArticleFull{PlainEmail} = $ArticleBackendObject->ArticlePlain(
                        ArticleID => $Article->{ArticleID},
                        UserID    => 1,
                    );
                }

                my @ArticleItem = $ObjectData->{ArticleSeparateLines} ? ('') : @TicketItem;
                ARTICLEMAP:
                for my $i ( 0 .. $ArticleMappingLast ) {

                    # empty fields are done in the ticket part
                    next ARTICLEMAP if !$ArticleMapping[$i];

                    # extract key
                    my $Key = $ArticleMapping[$i]{Key};

                    $ArticleItem[$i] = defined $ArticleFull{$Key}
                        ? IsArrayRefWithData( $ArticleFull{$Key} ) ? join( '###', ( map { encode_base64( Encode::encode( 'UTF-8', $_ ) ) } $ArticleFull{$Key}->@* ) )
                        : $ArticleFull{$Key} : '';
                }

                if ( $ObjectData->{IncludeAttachments} && $ArticleBackendObject->can('ArticleAttachmentIndex') ) {
                    my %Index = $ArticleBackendObject->ArticleAttachmentIndex(
                        ArticleID => $Article->{ArticleID},
                    );

                    for my $Key ( sort keys %Index ) {
                        my %Attachment = $ArticleBackendObject->ArticleAttachment(
                            TicketID  => $TicketID,
                            ArticleID => $Article->{ArticleID},
                            FileID    => $Key,
                        );

                        for my $Key (qw( Filename ContentType Disposition )) {
                            $Attachment{$Key} = Encode::encode( 'UTF-8', $Attachment{$Key} );
                        }

                        my $AttachmentString;
                        for my $Key (qw( Filename ContentID ContentType Disposition Content ContentAlternative )) {
                            $Attachment{$Key} //= '';
                            $AttachmentString  .= $AttachmentString ? '###' : '';
                            $AttachmentString  .= $Key . '###' . encode_base64( $Attachment{$Key} );
                        }

                        push @ArticleItem, $AttachmentString;
                    }
                }

                push @ExportData, \@ArticleItem;
            }
        }
    }

    return \@ExportData;
}

=head2 ImportDataSave()

imports a single entity of the import data. The C<TemplateID> points to the definition
of the current import. C<ImportDataRow> holds the data. C<Counter> is only used in
error messages, for indicating which item was not imported successfully.

The decision what constitute an empty value is a bit hairy.
Here are the rules.
Fields that are not even mentioned in the Import definition are empty. These are the 'not defined' fields.
Empty strings and undefined values constitute empty fields.
Fields with with only one or more whitespace characters are not empty.
Fields with the digit '0' are not empty.

    my ( $TicketID, $RetCode ) = $ObjectBackend->ImportDataSave(
        TemplateID    => 123,
        ImportDataRow => $ArrayRef,
        Counter       => 367,
        UserID        => 1,
    );

An empty C<TicketID> indicates failure. Otherwise it indicates the
location of the imported data.
C<RetCode> is either 'Created', 'Updated' or 'Skipped'.
'Created' means that a new ticket has been created.
'Updated' means that the ticket has been updated.
'Skipped' means that the data is identical and no changes were made.

=cut

sub ImportDataSave {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    for my $Argument (qw(TemplateID ImportDataRow Counter UserID)) {
        if ( !$Param{$Argument} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # check import data row
    if ( ref $Param{ImportDataRow} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  =>
                "Can't import entity $Param{Counter}: "
                . "ImportDataRow must be an array reference",
        );
        return;
    }

    # get object data
    my $ObjectData = $Self->{ImportExportObject}->ObjectDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check object data
    if ( !$ObjectData || ref $ObjectData ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  =>
                "Can't import entity $Param{Counter}: "
                . "No object data found for the template id '$Param{TemplateID}'",
        );
        return;
    }

    # just for convenience
    my $EmptyFieldsLeaveTheOldValues = $ObjectData->{EmptyFieldsLeaveTheOldValues};

    # get the mapping list
    my $MappingList = $Self->{ImportExportObject}->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check the mapping list
    if ( !$MappingList || ref $MappingList ne 'ARRAY' || !@{$MappingList} ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  =>
                "Can't import entity $Param{Counter}: "
                . "No valid mapping list found for the template id '$Param{TemplateID}'",
        );
        return;
    }

    # create the mapping object list
    my @MappingObjectList;
    for my $MappingID ( @{$MappingList} ) {

        # get mapping object data
        my $MappingObjectData = $Kernel::OM->Get('Kernel::System::ImportExport')->MappingObjectDataGet(
            MappingID => $MappingID,
            UserID    => $Param{UserID},
        );

        # check mapping object data
        if ( !$MappingObjectData || ref $MappingObjectData ne 'HASH' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  =>
                    "Can't import entity $Param{Counter}: "
                    . "No mapping object data found for the mapping id '$MappingID'",
            );
            return;
        }

        push @MappingObjectList, $MappingObjectData;
    }

    # prepare import data
    my %Ticket;
    my %Article;
    my %Identifier;

    my $MappingConfig = $Self->{ConfigObject}->Get('ImportExport::Ticket::ImportValueMap') // {};
    my %ValueMap      = map { $_->{Key} => $_->{Map} } values $MappingConfig->%*;

    # handle a separate article
    if ( $ObjectData->{IncludeArticles} && $ObjectData->{ArticleSeparateLines} && !$Param{ImportDataRow}[0] ) {
        my $i = 1;
        MAPPINGOBJECTDATA:
        for my $MappingObjectData (@MappingObjectList) {

            if ( $MappingObjectData->{Key} =~ /^Article_(.+)$/ ) {
                my $Value = $Param{ImportDataRow}[ $i++ ];
                $Article{$1} = defined $Value && $ValueMap{ $MappingObjectData->{Key} } && defined $ValueMap{ $MappingObjectData->{Key} }{$Value}
                    ? $ValueMap{ $MappingObjectData->{Key} }{$Value} : $Value;
            }
            else {
                next MAPPINGOBJECTDATA;
            }

            next MAPPINGOBJECTDATA if !$MappingObjectData->{Identifier};

            if ( $MappingObjectData->{Key} ne 'Article_ArticleID' ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  =>
                        "Can't import entity $Param{Counter}: "
                        . "Articles can only be identified via 'Article_ArticleID' not by '$MappingObjectData->{Key}'.",
                );
                return;
            }
            elsif ( !$Param{ImportDataRow}[ $i - 1 ] ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  =>
                        "Can't import entity $Param{Counter}: "
                        . "'Article_ArticleID' can not be empty or 0 when used as identifier.",
                );
                return;
            }

            $Identifier{Article}{ArticleID} = 1;
        }

        if ( $ObjectData->{IncludeAttachments} && $i <= $#{ $Param{ImportDataRow} } ) {

            # the last and unmapped entries are attachments
            $Article{Attachments} = [ @{ $Param{ImportDataRow} }[ $i .. $#{ $Param{ImportDataRow} } ] ];
        }
    }

    else {
        MAPPINGOBJECTDATA:
        for my $i ( 0 .. $#MappingObjectList ) {

            my $MappingObjectData = $MappingObjectList[$i];

            my $Value = $Param{ImportDataRow}[$i];
            $Value = defined $Value && $ValueMap{ $MappingObjectData->{Key} } && defined $ValueMap{ $MappingObjectData->{Key} }{$Value}
                ? $ValueMap{ $MappingObjectData->{Key} }{$Value} : $Value;

            if ( $MappingObjectData->{Key} =~ /^Article_(.+)$/ ) {
                next MAPPINGOBJECTDATA if $ObjectData->{ArticleSeparateLines} || !$ObjectData->{IncludeArticles};

                $Article{$1} = $Value;
            }
            else {
                $Ticket{ $MappingObjectData->{Key} } = $Value;
            }

            next MAPPINGOBJECTDATA if !$MappingObjectData->{Identifier};

            if ( !$Value ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  =>
                        "Can't import entity $Param{Counter}: "
                        . "'$MappingObjectData->{Key}' can not be empty or 0 when used as identifier.",
                );
                return;
            }

            if ( $MappingObjectData->{Key} eq 'Article_ArticleID' ) {
                $Identifier{Article}{ArticleID} = 1;
            }
            elsif ( $MappingObjectData->{Key} =~ /^Article_/ ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  =>
                        "Can't import entity $Param{Counter}: "
                        . "Currently only Article_ArticleID is a valid identifier for articles (not '$MappingObjectData->{Key}').",
                );
                return;
            }
            else {
                $Identifier{Ticket}{ $MappingObjectData->{Key} } = 1;
            }
        }

        if ( %Article && $ObjectData->{IncludeAttachments} && $#MappingObjectList < $#{ $Param{ImportDataRow} } ) {

            # the last and unmapped entries are attachments
            $Article{Attachments} = [ @{ $Param{ImportDataRow} }[ $#MappingObjectList + 1 .. $#{ $Param{ImportDataRow} } ] ];
        }
    }

    my $Status = 'Skipped';
    $Self->{Error} = '';
    if ( %Ticket && !exists $Self->{TicketIDRelation}{ $Ticket{TicketID} } ) {
        $Status = $Self->_ImportTicket(
            Ticket     => \%Ticket,
            Identifier => $Identifier{Ticket},
            ObjectData => $ObjectData,
        );

        if ( !$Status ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  =>
                    "Can't import entity $Param{Counter}: "
                    . "TicketImport: $Self->{Error}",
            );
            return;
        }
    }

    if ( %Article && $ObjectData->{IncludeArticles} ) {
        my $ArticleStatus = $Self->_ImportArticle(
            Article    => \%Article,
            Identifier => $Identifier{Article},
            ObjectData => $ObjectData,
        );

        if ( !$ArticleStatus ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  =>
                    "Can't import entity $Param{Counter}: "
                    . "ArticleImport: $Self->{Error}",
            );
            return;
        }

        if ( $Status eq 'Skipped' && $ArticleStatus eq 'Created' ) {
            $Status = 'Updated';
        }
    }

    return ( $Self->{LastTicketID}, $Status );
}

sub _ImportTicket {
    my ( $Self, %Param ) = @_;

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    my %Ticket = $Param{Ticket}->%*;
    my %DBTicket;
    my $Status = 'Skipped';

    if ( $Param{Identifier} ) {
        my $DBTicketID;

        # TicketID takes precedence over TicketNumber
        if ( $Param{Identifier}{TicketID} ) {

            # check previously imported tickets of this run
            if ( $Self->{TicketIDRelation}{ $Ticket{TicketID} } ) {

                # we silently skip this ticket, if it already has been imported
                # this situation will always occur when articles are not imported separately
                # for the sake of consistency, we treat other situations the same, although they are less clear
                my $Prio = !$Param{ObjectData}{ArticleSeparateLines}
                    && $Self->{LastTicketID} == $Self->{TicketIDRelation}{ $Ticket{TicketID} } ? 'debug' : 'info';
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => $Prio,
                    Message  => "Skipping ticket creation for entity $Param{Counter} (TicketID $Ticket{TicketID}) as it was handled before."
                );
                $Self->{LastTicketID} = $Self->{TicketIDRelation}{ $Ticket{TicketID} };
                return $Status;
            }

            # existing but undefined entry means previous error - we skip and do nothing
            elsif ( exists $Self->{TicketIDRelation}{ $Ticket{TicketID} } ) {
                return $Self->_ImportError(
                    %Param,
                    Message => "TicketID $Ticket{TicketID} was incorrectly imported before canceling new try.",
                );
            }

            # exclude tickets which were created in this run and by chance got the current TicketID
            elsif ( !{ reverse $Self->{TicketIDRelation}->%* }->{ $Ticket{TicketID} } ) {
                $DBTicketID = $Ticket{TicketID};
            }
        }

        elsif ( $Param{Identifier}{TicketNumber} ) {

            # check previously imported tickets of this run
            if ( $Self->{TicketNumberIDRelation}{ $Ticket{TicketNumber} } ) {

                # we silently skip this ticket, if it already has been imported
                # this situation will always occur when articles are not imported separately
                # for the sake of consistency, we treat other situations the same, although they are less clear
                my $Prio = !$Param{ObjectData}{ArticleSeparateLines}
                    && $Self->{LastTicketID} == $Self->{TicketNumberIDRelation}{ $Ticket{TicketNumber} } ? 'debug' : 'info';
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => $Prio,
                    Message  => "Skipping ticket creation for entity $Param{Counter} (TicketNumber $Ticket{TicketNumber}) as it was handled before."
                );
                $Self->{LastTicketID} = $Self->{TicketNumberIDRelation}{ $Ticket{TicketNumber} };
                return $Self->{LastTicketID};
            }

            # existing but undefined entry means previous error - we skip and do nothing
            elsif ( exists $Self->{TicketNumberIDRelation}{ $Ticket{TicketNumber} } ) {
                return $Self->_ImportError(
                    %Param,
                    Message => "TicketNumber $Ticket{TicketNumber} was incorrectly imported before canceling new try.",
                );
            }

            # exclude tickets which were created in this run and by chance got the current TicketNumber
            elsif ( !$Self->{CreatedNumbers}{ $Ticket{TicketNumber} } ) {
                $DBTicketID = $TicketObject->TicketIDLookup(
                    Tn => $Ticket{TicketNumber},
                );
            }
        }

        else {
            return $Self->_ImportError(
                %Param,
                Message =>
                    'For ticket correlation checks either TicketID or TicketNumber has to be included as identifier. (Using no identifier to just create new tickets is also possible.)',
            );
        }

        if ($DBTicketID) {
            %DBTicket = $TicketObject->TicketGet(
                TicketID      => $DBTicketID,
                DynamicFields => 0,
                UserID        => 1,
                Silent        => 1,
            );
        }
    }

    # verify whether we really got the right ticket
    if (%DBTicket) {
        ATTR:
        for my $Attr ( keys $Param{Identifier}->%* ) {
            if ( $DBTicket{$Attr} ne $Ticket{$Attr} ) {
                %DBTicket = ();
                last ATTR;
            }
        }
    }

    # just update the ticket if it is already present
    if (%DBTicket) {
        $Status = 'Skipped';
        my $SkipEmpty = $Param{ObjectData}{EmptyFieldsLeaveTheOldValues};

        # customer
        $Ticket{CustomerID}     ||= $SkipEmpty ? $DBTicket{CustomerID}     : $Param{ObjectData}{CustomerID};
        $Ticket{CustomerUserID} ||= $SkipEmpty ? $DBTicket{CustomerUserID} : $Param{ObjectData}{CustomerUserID};
        $DBTicket{CustomerID}     //= '';
        $DBTicket{CustomerUserID} //= '';

        if ( ( !$Ticket{CustomerID} && $DBTicket{CustomerID} ) || ( !$Ticket{CustomerUserID} && $DBTicket{CustomerUserID} ) ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "Unexpected state encountered for existing TicketID $Ticket{TicketID} (Entity $Param{Counter}): "
                    . "Ticket customer can not be emptied - not a real chronological update; ignoring this, and keeping Customer(User)ID.",
            );
        }

        {
            my $Update = $Ticket{CustomerID} && $Ticket{CustomerID} ne $DBTicket{CustomerID} ? 1 :
                $Ticket{CustomerUserID} && $Ticket{CustomerUserID} ne $DBTicket{CustomerUserID} ? 1 : 0;

            if ($Update) {
                $Status = 'Updated';
                my $Success = $TicketObject->TicketCustomerSet(
                    No       => $Ticket{CustomerID},
                    User     => $Ticket{CustomerUserID},
                    TicketID => $DBTicket{TicketID},
                    UserID   => 1,
                );

                return $Self->_ImportError(
                    %Param,
                    Message => "Could not update Customer(User) for TicketID $DBTicket{TicketID}",
                ) if !$Success;
            }
        }

        # title
        $Ticket{Title} ||= $Ticket{Title} eq '0' ? '0' :
            $SkipEmpty                  ? $DBTicket{Title} :
            $Param{ObjectData}{Subject} ? $Param{ObjectData}{Subject} : '';

        if ( $Ticket{Title} ne $DBTicket{Title} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketTitleUpdate(
                Title    => $Ticket{Title},
                TicketID => $DBTicket{TicketID},
                UserID   => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Title for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # queue
        if ( !$Ticket{QueueID} && $Ticket{Queue} ) {
            $Ticket{QueueID} = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
                Queue => $Ticket{Queue},
            );
        }
        $Ticket{QueueID} ||= $SkipEmpty ? $DBTicket{QueueID} : $Param{ObjectData}{QueueID};

        if ( $Ticket{QueueID} ne $DBTicket{QueueID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketQueueSet(
                QueueID  => $Ticket{QueueID},
                TicketID => $DBTicket{TicketID},
                UserID   => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Queue for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # type
        if ( !$Ticket{TypeID} && $Ticket{Type} ) {
            $Ticket{TypeID} = $Kernel::OM->Get('Kernel::System::Type')->TypeLookup(
                Type => $Ticket{Type},
            );
        }
        $Ticket{TypeID} ||= $SkipEmpty ? $DBTicket{TypeID} : $Param{ObjectData}{TypeID};

        if ( $Ticket{TypeID} ne $DBTicket{TypeID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketTypeSet(
                TypeID   => $Ticket{TypeID},
                TicketID => $DBTicket{TicketID},
                UserID   => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Type for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        if ( $Self->{ConfigObject}->Get('Ticket::Service') ) {

            # service
            if ( !$Ticket{ServiceID} && $Ticket{Service} ) {
                $Ticket{ServiceID} = $Kernel::OM->Get('Kernel::System::Service')->ServiceLookup(
                    Name => $Ticket{Service},
                );
            }
            $Ticket{ServiceID} ||= $SkipEmpty ? $DBTicket{ServiceID} : $Param{ObjectData}{ServiceID};
            $DBTicket{ServiceID} //= '';

            if ( !$Ticket{ServiceID} && $DBTicket{ServiceID} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'notice',
                    Message  => "Unexpected state encountered for existing TicketID $Ticket{TicketID} (Entity $Param{Counter}): "
                        . "Services can not be emptied - not a real chronological update; ignoring this, and keeping the Service.",
                );
            }

            if ( $Ticket{ServiceID} && $Ticket{ServiceID} ne $DBTicket{ServiceID} ) {
                $Status = 'Updated';
                my $Success = $TicketObject->TicketServiceSet(
                    ServiceID => $Ticket{ServiceID},
                    TicketID  => $DBTicket{TicketID},
                    UserID    => 1,
                );

                return $Self->_ImportError(
                    %Param,
                    Message => "Could not update Service for TicketID $DBTicket{TicketID}",
                ) if !$Success;
            }

            # sla
            if ( !$Ticket{SLAID} && $Ticket{SLA} ) {
                $Ticket{SLAID} = $Kernel::OM->Get('Kernel::System::SLA')->SLALookup(
                    SLA => $Ticket{SLA},
                );
            }
            $Ticket{SLAID} ||= $SkipEmpty ? $DBTicket{SLAID} : $Param{ObjectData}{SLAID};
            $DBTicket{SLAID} //= '';

            if ( !$Ticket{SLAID} && $DBTicket{SLAID} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'notice',
                    Message  => "Unexpected state encountered for existing TicketID $Ticket{TicketID} (Entity $Param{Counter}): "
                        . "SLAs can not be emptied - not a real chronological update; ignoring this, and keeping the SLA.",
                );
            }

            if ( $Ticket{SLAID} && $Ticket{SLAID} ne $DBTicket{SLAID} ) {
                $Status = 'Updated';
                my $Success = $TicketObject->TicketSLASet(
                    SLAID    => $Ticket{SLAID},
                    TicketID => $DBTicket{TicketID},
                    UserID   => 1,
                );

                return $Self->_ImportError(
                    %Param,
                    Message => "Could not update SLA for TicketID $DBTicket{TicketID}",
                ) if !$Success;
            }
        }

        # owner
        if ( !$Ticket{OwnerID} && $Ticket{Owner} ) {
            $Ticket{OwnerID} = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
                UserLogin => $Ticket{Owner},
            );
        }
        $Ticket{OwnerID} ||= $SkipEmpty ? $DBTicket{OwnerID} : $Param{ObjectData}{OwnerID};

        if ( !$Ticket{OwnerID} && $DBTicket{OwnerID} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "Unexpected state encountered for existing TicketID $Ticket{TicketID} (Entity $Param{Counter}): "
                    . "Owners can not be emptied - not a real chronological update; ignoring this, and keeping the Owner.",
            );
        }

        if ( $Ticket{OwnerID} && $Ticket{OwnerID} ne $DBTicket{OwnerID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketOwnerSet(
                NewUserID => $Ticket{OwnerID},
                TicketID  => $DBTicket{TicketID},
                UserID    => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Owner for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # lock
        if ( !$Ticket{LockID} && $Ticket{Lock} ) {
            $Ticket{LockID} = $Kernel::OM->Get('Kernel::System::Lock')->LockLookup(
                Lock => $Ticket{Lock},
            );
        }
        $Ticket{LockID} ||= $SkipEmpty ? $DBTicket{LockID} : $Param{ObjectData}{LockID};

        # check whether an owner exists here - if not trying to lock will fail
        if ( $Ticket{OwnerID} && $Ticket{LockID} ne $DBTicket{LockID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketLockSet(
                LockID   => $Ticket{LockID},
                TicketID => $DBTicket{TicketID},
                UserID   => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Lock for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # responsible
        if ( !$Ticket{ResponsibleID} && $Ticket{Responsible} ) {
            $Ticket{ResponsibleID} = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
                UserLogin => $Ticket{Responsible},
            );
        }
        $Ticket{ResponsibleID} ||= $SkipEmpty ? $DBTicket{ResponsibleID} : $Param{ObjectData}{ResponsibleID};

        if ( !$Ticket{ResponsibleID} && $DBTicket{ResponsibleID} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "Unexpected state encountered for existing TicketID $Ticket{TicketID} (Entity $Param{Counter}): "
                    . "Responsibles can not be emptied - not a real chronological update; ignoring this, and keeping the Responsible.",
            );
        }

        if ( $Ticket{ResponsibleID} && $Ticket{ResponsibleID} ne $DBTicket{ResponsibleID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketResponsibleSet(
                NewUserID => $Ticket{ResponsibleID},
                TicketID  => $DBTicket{TicketID},
                UserID    => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Responsible for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # priority
        if ( !$Ticket{PriorityID} && $Ticket{Priority} ) {
            $Ticket{PriorityID} = $Kernel::OM->Get('Kernel::System::Priority')->PriorityLookup(
                Priority => $Ticket{Priority},
            );
        }
        $Ticket{PriorityID} ||= $SkipEmpty ? $DBTicket{PriorityID} : $Param{ObjectData}{PriorityID};

        if ( $Ticket{PriorityID} ne $DBTicket{PriorityID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketPrioritySet(
                PriorityID => $Ticket{PriorityID},
                TicketID   => $DBTicket{TicketID},
                UserID     => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update Priority for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # state
        if ( !$Ticket{StateID} && $Ticket{State} ) {
            $Ticket{StateID} = $Kernel::OM->Get('Kernel::System::State')->StateLookup(
                State => $Ticket{State},
            );
        }
        $Ticket{StateID} ||= $SkipEmpty ? $DBTicket{StateID} : $Param{ObjectData}{StateID};

        if ( $Ticket{StateID} ne $DBTicket{StateID} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketStateSet(
                StateID  => $Ticket{StateID},
                TicketID => $DBTicket{TicketID},
                UserID   => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update State for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }

        # archive flag
        $Ticket{ArchiveFlag} ||= $SkipEmpty ? $DBTicket{ArchiveFlag} : $Param{ObjectData}{ArchiveFlag};

        if ( $Ticket{ArchiveFlag} ne $DBTicket{ArchiveFlag} ) {
            $Status = 'Updated';
            my $Success = $TicketObject->TicketArchiveFlagSet(
                ArchiveFlag => $Ticket{ArchiveFlag},
                TicketID    => $DBTicket{TicketID},
                UserID      => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not update ArchiveFlag for TicketID $DBTicket{TicketID}",
            ) if !$Success;
        }
    }

    # create a new ticket
    else {
        $Status = 'Created';

        # customer
        $DBTicket{CustomerID}   = $Ticket{CustomerID}     || $Param{ObjectData}{CustomerID};
        $DBTicket{CustomerUser} = $Ticket{CustomerUserID} || $Param{ObjectData}{CustomerUserID};

        # title
        $DBTicket{Title} = $Ticket{Title} || $Ticket{Title} eq '0' ? $Ticket{Title} : $Param{ObjectData}{Subject};

        # queue
        if ( !$Ticket{QueueID} && $Ticket{Queue} ) {
            $Ticket{QueueID} = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
                Queue => $Ticket{Queue},
            );
        }
        $DBTicket{QueueID} = $Ticket{QueueID} || $Param{ObjectData}{QueueID};

        # type
        if ( !$Ticket{TypeID} && $Ticket{Type} ) {
            $Ticket{TypeID} = $Kernel::OM->Get('Kernel::System::Type')->TypeLookup(
                Type => $Ticket{Type},
            );
        }
        $DBTicket{TypeID} = $Ticket{TypeID} || $Param{ObjectData}{TypeID};

        if ( $Self->{ConfigObject}->Get('Ticket::Service') ) {

            # service
            if ( !$Ticket{ServiceID} && $Ticket{Service} ) {
                $Ticket{ServiceID} = $Kernel::OM->Get('Kernel::System::Service')->ServiceLookup(
                    Name => $Ticket{Service},
                );
            }
            $DBTicket{ServiceID} = $Ticket{ServiceID} || $Param{ObjectData}{ServiceID};

            # sla
            if ( !$Ticket{SLAID} && $Ticket{SLA} ) {
                $Ticket{SLAID} = $Kernel::OM->Get('Kernel::System::SLA')->SLALookup(
                    SLA => $Ticket{SLA},
                );
            }
            $DBTicket{SLAID} = $Ticket{SLAID} || $Param{ObjectData}{SLAID};
        }

        # owner
        if ( !$Ticket{OwnerID} && $Ticket{Owner} ) {
            $Ticket{OwnerID} = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
                UserLogin => $Ticket{Owner},
            );
        }
        $DBTicket{OwnerID} = $Ticket{OwnerID} || $Param{ObjectData}{OwnerID};

        # lock
        if ( !$Ticket{LockID} && $Ticket{Lock} ) {
            $Ticket{LockID} = $Kernel::OM->Get('Kernel::System::Lock')->LockLookup(
                Lock => $Ticket{Lock},
            );
        }
        $DBTicket{LockID} = $Ticket{LockID} || $Param{ObjectData}{LockID};

        # responsible
        if ( !$Ticket{ResponsibleID} && $Ticket{Responsible} ) {
            $Ticket{ResponsibleID} = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
                UserLogin => $Ticket{Responsible},
            );
        }
        $DBTicket{ResponsibleID} = $Ticket{ResponsibleID} || $Param{ObjectData}{ResponsibleID};

        # priority
        if ( !$Ticket{PriorityID} && $Ticket{Priority} ) {
            $Ticket{PriorityID} = $Kernel::OM->Get('Kernel::System::Priority')->PriorityLookup(
                Priority => $Ticket{Priority},
            );
        }
        $DBTicket{PriorityID} = $Ticket{PriorityID} || $Param{ObjectData}{PriorityID};

        # state
        if ( !$Ticket{StateID} && $Ticket{State} ) {
            $Ticket{StateID} = $Kernel::OM->Get('Kernel::System::State')->StateLookup(
                State => $Ticket{State},
            );
        }
        $DBTicket{StateID} = $Ticket{StateID} || $Param{ObjectData}{StateID};

        # archive flag
        $DBTicket{ArchiveFlag} = $Ticket{ArchiveFlag} || $Param{ObjectData}{ArchiveFlag};

        $DBTicket{TicketID} = $TicketObject->TicketCreate(
            %DBTicket,
            TN     => $Ticket{TicketNumber} // '',
            UserID => 1,
        );

        return $Self->_ImportError(
            %Param,
            Message => 'Could not create new ticket',
        ) if !$DBTicket{TicketID};

        $DBTicket{TicketNumber} = $TicketObject->TicketNumberLookup(
            TicketID => $DBTicket{TicketID},
        );

        if ( $Ticket{Created} ) {
            $Self->{DBObject} //= $Kernel::OM->Get('Kernel::System::DB');

            return if !$Self->{DBObject}->Do(
                SQL => "UPDATE ticket SET create_time = ? WHERE id = ?",
                Bind => [ \$Ticket{Created}, \$DBTicket{TicketID} ],
            );
        }

        my $SyncDBConfig = $Self->{ConfigObject}->Get('ImportExport::Ticket::SynchronizeWithForeignDB');
        if ($SyncDBConfig) {
            my $Success = $Self->_SynchronizeExtendedDBEntries(
                ForeignDB       => $SyncDBConfig,
                ForeignTicketID => $Ticket{TicketID},
                TicketID        => $DBTicket{TicketID},
            );

            return $Self->_ImportError(
                %Param,
                Message => "Could not synchronize extended DB entries for ticket $DBTicket{TicketID}",
            ) if !$Success;
        }
    }

    my $DynamicFieldObject        = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my %StandardMultiSelect       = map { $_ => 1 } qw( Multiselect Database WebService );

    # dynamic fields
    DYNAMICFIELD:
    for my $Attr ( keys %Ticket ) {
        next DYNAMICFIELD if $Attr !~ /^DynamicField_(.+)$/;

        my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
            Name => $1,
        );

        # get the current value
        my $DBValue = $DynamicFieldBackendObject->ValueGet(
            DynamicFieldConfig => $DynamicFieldConfig,
            ObjectID           => $DBTicket{TicketID},
            UserID             => 1,
        );

        # multi select fields need special treatment
        if ( $StandardMultiSelect{ $DynamicFieldConfig->{FieldType} } || ref $DBValue eq 'ARRAY' ) {
            $Ticket{$Attr} = $Ticket{$Attr} ? [ map { decode( 'UTF-8', decode_base64($_) ) } split( '###', $Ticket{$Attr} ) ] : [];
        }

        next DYNAMICFIELD if !$DynamicFieldBackendObject->ValueIsDifferent(
            DynamicFieldConfig => $DynamicFieldConfig,
            Value1             => $Ticket{$Attr},
            Value2             => $DBValue,
        );

        if ( $Status eq 'Skipped' ) {
            $Status = 'Updated';
        }

        my $Success = $DynamicFieldBackendObject->FieldValueValidate(
            DynamicFieldConfig => $DynamicFieldConfig,
            Value              => $Ticket{$Attr},
            UserID             => 1,
        );

        if ($Success) {

            # set the value
            $Success = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectID           => $DBTicket{TicketID},
                Value              => $Ticket{$Attr},
                UserID             => 1,
            );
        }

        return $Self->_ImportError(
            %Param,
            Message => "Could not update $Attr to '$Ticket{$Attr}' for TicketID $DBTicket{TicketID}",
        ) if !$Success;
    }

    if ( $Param{Identifier}{TicketID} || $Param{ObjectData}{IncludeArticles} ) {
        $Self->{TicketIDRelation}{ $Ticket{TicketID} } = $DBTicket{TicketID};
    }
    if ( $Param{Identifier}{TicketNumber} ) {
        $Self->{TicketNumberIDRelation}{ $Ticket{TicketNumber} } = $DBTicket{TicketID};
    }
    $Self->{LastTicketID} = $DBTicket{TicketID};

    return $Status;
}

sub _ImportArticle {
    my ( $Self, %Param ) = @_;

    my %Article = $Param{Article}->%*;

    my $TicketID =
        $Article{TicketID} && $Self->{TicketIDRelation}{ $Article{TicketID} } ? $Self->{TicketIDRelation}{ $Article{TicketID} } :
        $Self->{LastTicketID} ? $Self->{LastTicketID} : '';

    return $Self->_ImportError(
        %Param,
        Message => "Could not find new TicketID for Article",
    ) if !$TicketID;

    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

    if ( $Param{Identifier} && $Param{Identifier}{ArticleID} && $Article{ArticleID} ) {
        my @DBArticles = $ArticleObject->ArticleList(
            TicketID  => $TicketID,
            ArticleID => $Article{ArticleID},
        );

        return 'Skipped' if @DBArticles;
    }

    my %ValidImportChannel = map { $_ => 1 } qw( Email Internal Phone );
    my $ChannelName        = $Article{ArticleBackend} || $Param{ObjectData}{ArticleBackend};

    return $Self->_ImportError(
        %Param,
        Message => "'$ChannelName' is (currently) not a supported ArticleBackend",
    ) if !$ValidImportChannel{$ChannelName};

    my $ArticleBackendObject = $ArticleObject->BackendForChannel( ChannelName => $ChannelName );

    if ( $ChannelName eq 'Email' && !$Article{MessageID} ) {
        my $Time   = $Kernel::OM->Create('Kernel::System::DateTime')->ToEpoch();
        my $Random = rand 999999;
        my $FQDN   = $Self->{ConfigObject}->Get('FQDN');
        $Article{MessageID} = "<$Time.$Random\@$FQDN>";
    }

    my $HistoryComment = '%%Imported Article';
    if ( $Article{TicketID} || $Article{ArticleID} ) {
        $HistoryComment .= ' - Legacy';
        $HistoryComment .= $Article{TicketID}  ? " TID: $Article{TicketID};"  : '';
        $HistoryComment .= $Article{ArticleID} ? " AID: $Article{ArticleID};" : '';
    }

    my $ArticleID = $ArticleBackendObject->ArticleCreate(
        NoAgentNotify        => 1,
        TicketID             => $TicketID,
        SenderType           => $Article{SenderType} || $Param{ObjectData}{SenderType},
        IsVisibleForCustomer => defined $Article{IsVisibleForCustomer} && $Article{IsVisibleForCustomer} ne '' ? $Article{IsVisibleForCustomer} : $Param{ObjectData}{IsVisibleForCustomer} // 0,
        From                 => $Article{From},
        To                   => $Article{To},
        Cc                   => $Article{Cc},
        Bcc                  => $Article{Bcc},
        ReplyTo              => $Article{ReplyTo},
        InReplyTo            => $Article{InReplyTo},
        References           => $Article{References},
        Subject              => $Article{Subject}  || $Param{ObjectData}{Subject},
        Body                 => $Article{Body}     || $Param{ObjectData}{Body},
        Charset              => $Article{Charset}  || 'utf-8',
        MimeType             => $Article{MimeType} || 'text/plain',
        HistoryType          => 'Misc',
        HistoryComment       => $HistoryComment,
        MessageID            => $Article{MessageID},
        UserID               => 1,
    );

    return $Self->_ImportError(
        %Param,
        Message => "Could not create the article",
    ) if !$ArticleID;

    if ( $Article{CreateTime} ) {
        $Self->{DBObject} //= $Kernel::OM->Get('Kernel::System::DB');

        return if !$Self->{DBObject}->Do(
            SQL => "UPDATE article SET create_time = ? WHERE id = ?",
            Bind => [ \$Article{CreateTime}, \$ArticleID ],
        );
    }

    my $SyncDBConfig = $Self->{ConfigObject}->Get('ImportExport::Ticket::SynchronizeWithForeignDB');
    if ($SyncDBConfig) {
        my $Success = $Self->_SynchronizeExtendedDBEntries(
            ForeignDB        => $SyncDBConfig,
            ForeignArticleID => $Article{ArticleID},
            ArticleID        => $ArticleID,
            TicketID         => $TicketID,
        );

        return $Self->_ImportError(
            %Param,
            Message => "Could not synchronize extended DB entries for ticket $ArticleID",
        ) if !$Success;
    }

    # attachments
    if ( $Param{ObjectData}{IncludeAttachments} && $Article{Attachments} ) {
        for my $AttachmentString ( $Article{Attachments}->@* ) {
            my %Attachment = split( '###', $AttachmentString, -1 );

            for my $Key ( keys %Attachment ) {
                $Attachment{$Key} = $Attachment{$Key} eq '' ? '' : decode_base64( $Attachment{$Key} );
            }

            KEY:
            for my $Key (qw( Filename ContentType Disposition )) {
                next KEY if !$Attachment{$Key};

                $Attachment{$Key} = Encode::decode( 'UTF-8', $Attachment{$Key} );
            }

            my $Success = $ArticleBackendObject->ArticleWriteAttachment(
                %Attachment,
                ArticleID => $ArticleID,
                UserID    => 1,
            );

            return $Self->_ImportError(
                %Param,
                Message => "Error with importing an attachment for article $ArticleID",
            ) if !$Success;
        }
    }

    # plain email
    if ( $Article{PlainEmail} ) {
        my $Success = $ArticleBackendObject->ArticleWritePlain(
            ArticleID => $ArticleID,
            Email     => $Article{PlainEmail},
            UserID    => 1,
        );

        return $Self->_ImportError(
            %Param,
            Message => "Error with importing the plain email for article $ArticleID",
        ) if !$Success;
    }

    my $DynamicFieldObject        = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my %StandardMultiSelect       = map { $_ => 1 } qw( Multiselect Database WebService );

    # dynamic fields
    DYNAMICFIELD:
    for my $Attr ( keys %Article ) {
        next DYNAMICFIELD if $Attr !~ /^DynamicField_(.+)$/;

        my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
            Name => $1,
        );

        # get the current value
        my $DBValue = $DynamicFieldBackendObject->ValueGet(
            DynamicFieldConfig => $DynamicFieldConfig,
            ObjectID           => $ArticleID,
            UserID             => 1,
        );

        # multi select fields need special treatment
        if ( $StandardMultiSelect{ $DynamicFieldConfig->{FieldType} } || ref $DBValue eq 'ARRAY' ) {
            $Article{$Attr} = $Article{$Attr} ? [ map { decode( 'UTF-8', decode_base64($_) ) } split( '###', $Article{$Attr} ) ] : [];
        }

        next DYNAMICFIELD if !$DynamicFieldBackendObject->ValueIsDifferent(
            DynamicFieldConfig => $DynamicFieldConfig,
            Value1             => $Article{$Attr},
            Value2             => $DBValue,
        );

        my $Success = $DynamicFieldBackendObject->FieldValueValidate(
            DynamicFieldConfig => $DynamicFieldConfig,
            Value              => $Article{$Attr},
            UserID             => 1,
        );

        if ($Success) {

            # set the value
            $Success = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectID           => $ArticleID,
                Value              => $Article{$Attr},
                UserID             => 1,
            );
        }

        return $Self->_ImportError(
            %Param,
            Message => "Could not update $Attr for ArticleID $ArticleID",
        ) if !$Success;
    }

    return 'Created';
}

sub _ImportError {
    my ( $Self, %Param ) = @_;

    $Self->{Error} = $Param{Message} || '';

    if ( exists $Param{Ticket} ) {
        $Self->{LastTicketID} = undef;

        if ( $Param{Identifier} && $Param{Identifier}{TicketID} && $Param{Ticket}{TicketID} ) {
            $Self->{TicketIDRelation}{ $Param{Ticket}{TicketID} } = undef;
        }
        elsif ( $Param{Identifier} && $Param{Identifier}{TicketNumber} && $Param{Ticket}{TicketNumber} ) {
            $Self->{TicketNumberIDRelation}{ $Param{Ticket}{TicketNumber} } = undef;
        }
    }

    elsif ( exists $Param{Article} ) {
        my $ImportedTID = $Param{Article}{TicketID} ? $Self->{TicketIDRelation}{ $Param{Article}{TicketID} } : $Self->{LastTicketID};
        my $ExtraInfo   = $Param{Article}{TicketID} ? " Originial TicketID: $Param{Article}{TicketID};"      : '';
        $ExtraInfo     .= $ImportedTID               ? " Imported TicketID: $ImportedTID;"                 : '';
        $ExtraInfo     .= $Param{Article}{ArticleID} ? " Originial ArticleID: $Param{Article}{ArticleID};" : '';
        $Self->{Error} .= $ExtraInfo;
    }

    return;
}

sub _SynchronizeExtendedDBEntries {
    my ( $Self, %Param ) = @_;

    if ( !$Param{ForeignDB}{DatabaseDSN} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need a DatabaseDSN in the ForeignDB settings',
        );

        return;
    }

    if ( !$Param{TicketID} || !( ( $Param{ArticleID} && $Param{ForeignArticleID} || $Param{ForeignTicketID} ) ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need TicketID and either ArticleID and ForeignArticleID or ForeignTicketID!',
        );

        return;
    }

    if ( !$Self->{FDBObject} || !$Self->{DBObject} ) {
        $Self->{DBObject}  = $Kernel::OM->Get('Kernel::System::DB');
        $Self->{FDBObject} = Kernel::System::DB->new(
            $Param{ForeignDB}->%*,
        );
    }

    if ( $Param{ArticleID} ) {

        # article_flag
        return if !$Self->{FDBObject}->Prepare(
            SQL  => 'SELECT create_by FROM article_flag WHERE article_id = ? AND article_key = ? AND article_value = ?',
            Bind => [ \$Param{ForeignArticleID}, \'Seen', \1 ],
        );

        while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {
            return if !$Self->{DBObject}->Do(
                SQL => 'INSERT INTO article_flag (article_id, article_key, article_value, create_time, create_by)' .
                    'VALUES (?, ?, ?, current_timestamp, ?)',
                Bind => [ \$Param{ArticleID}, \'Seen', \1, \$Row[0] ],
            );
        }

        # article_history
        return if !$Self->{FDBObject}->Prepare(
            SQL => 'SELECT name, history_type_id, type_id, queue_id, owner_id, priority_id, state_id, create_time, create_by, change_time, change_by ' .
                'FROM ticket_history WHERE article_id = ?',
            Bind => [ \$Param{ForeignArticleID} ],
        );

        while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {
            return if !$Self->{DBObject}->Do(
                SQL =>
                    'INSERT INTO ticket_history (ticket_id, article_id, name, history_type_id, type_id, queue_id, owner_id, priority_id, state_id, create_time, create_by, change_time, change_by)'
                    .
                    'VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)',
                Bind => [ \$Param{TicketID}, \$Param{ArticleID}, ( map { \$_ } @Row ) ],
            );
        }

        return 1;
    }

    # ticket_flag
    return if !$Self->{FDBObject}->Prepare(
        SQL  => 'SELECT create_by FROM ticket_flag WHERE ticket_id = ? AND ticket_key = ? AND ticket_value = ?',
        Bind => [ \$Param{ForeignTicketID}, \'Seen', \1 ],
    );

    while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {
        return if !$Self->{DBObject}->Do(
            SQL => 'INSERT INTO ticket_flag (ticket_id, ticket_key, ticket_value, create_time, create_by)' .
                'VALUES (?, ?, ?, current_timestamp, ?)',
            Bind => [ \$Param{TicketID}, \'Seen', \1, \$Row[0] ],
        );
    }

    # ticket_history
    return if !$Self->{DBObject}->Do(
        SQL  => 'DELETE FROM ticket_history WHERE ticket_id = ? AND (article_id IS NULL OR article_id = 0)',
        Bind => [ \$Param{TicketID} ],
    );

    $Kernel::OM->Get('Kernel::System::Ticket')->HistoryAdd(
        Name         => "\%\%Imported Ticket - Legacy TicketID: $Param{ForeignTicketID}",
        HistoryType  => 'Misc',
        TicketID     => $Param{TicketID},
        CreateUserID => 1,
    );

    return if !$Self->{FDBObject}->Prepare(
        SQL => 'SELECT name, history_type_id, type_id, queue_id, owner_id, priority_id, state_id, create_time, create_by, change_time, change_by ' .
            'FROM ticket_history WHERE ticket_id = ? AND (article_id IS NULL OR article_id = 0)',
        Bind => [ \$Param{ForeignTicketID} ],
    );

    while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {
        return if !$Self->{DBObject}->Do(
            SQL =>
                'INSERT INTO ticket_history (ticket_id, name, history_type_id, type_id, queue_id, owner_id, priority_id, state_id, create_time, create_by, change_time, change_by)'
                .
                'VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)',
            Bind => [ \$Param{TicketID}, ( map { \$_ } @Row ) ],
        );
    }

    # link_relation
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');
    my $ForeignTicketObjectID;

    my %ForeignLinkObjects;
    my %LObjectIDs;

    return if !$Self->{FDBObject}->Prepare(
        SQL => "SELECT id,name FROM link_object ",
        Bind => [],
    );
    while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {
        $ForeignLinkObjects{ $Row[0] } = $Row[1];
        $LObjectIDs{ $Row[1] } = $LinkObject->ObjectLookup(
            Name => $Row[1],
        );

        if ( $Row[1] eq 'Ticket' ) {
            $ForeignTicketObjectID = $Row[0];
        }
    }

    my %ForeignLinkTypes;
    my %LTypeIDs;

    return if !$Self->{FDBObject}->Prepare(
        SQL  => "SELECT id,name FROM link_type ",
        Bind => [],
    );
    while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {
        $ForeignLinkTypes{ $Row[0] } = $Row[1];
        $LTypeIDs{ $Row[1] } = $LinkObject->TypeLookup(
            Name   => $Row[1],
            UserID => 1,
        );
    }

    my %Direction = (
        source => 'target',
        target => 'source',
    );

    for my $Key ( keys %Direction ) {

        return if !$Self->{FDBObject}->Prepare(
            SQL => "SELECT $Key"."_object_id, $Key"."_key, type_id, state_id FROM link_relation " .
                "WHERE $Direction{$Key}_object_id = ? AND $Direction{$Key}_key = ?",
            Bind => [ \$ForeignTicketObjectID, \$Param{ForeignTicketID} ],
        );

        LINK:
        while ( my @Row = $Self->{FDBObject}->FetchrowArray() ) {

            $Row[0] = $LObjectIDs{ $ForeignLinkObjects{ $Row[0] } };
            $Row[2] = $LTypeIDs{ $ForeignLinkTypes{ $Row[2] } };

            # deal with other imported tickets
            if ( $Row[0] == $LObjectIDs{Ticket} ) {
                next LINK if $Row[1] > $Param{ForeignTicketID};

                $Row[1] = exists $Self->{TicketIDRelation}{ $Row[1] } ? $Self->{TicketIDRelation}{ $Row[1] } : $Row[1];
            }

            return if !$Self->{DBObject}->Do(
                SQL =>
                    "INSERT INTO link_relation ($Direction{$Key}_object_id, $Direction{$Key}_key, $Key\_object_id, $Key\_key, type_id, state_id, create_time, create_by) " .
                    "VALUES (?, ?, ?, ?, ?, ?, current_timestamp, ?)",
                Bind => [ \$LObjectIDs{Ticket}, \$Param{TicketID}, ( map { \$_ } @Row ), \1 ],
            );
        }
    }

    # ticket_watcher
    # TODO

    # form_draft
    # TODO

    # calender_appointment_ticket
    # TODO

    return 1;
}

1;

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