UserImportExport
5.0.0
c.a.p.e. IT GmbH
http://www.cape-it.de/
GNU GENERAL PUBLIC LICENSE Version 2, June 1991
CR: T2015082690000553 (added changes for OTRS 5.0.x) (tlange).
CR: (added changes for use with OTRS 5.0.x beta 3)
Provides ImportExportBackend for Users.
5.0.x
<br/>
<strong>WELCOME</strong>
<br/>
<br/>
You are about to INSTALL the UserImportExport extensions package for OTRS.<br/>
<br/>
<br/>
<b>Required OTRS packages:</b><br><br>
ImportExport 5.0.1 or higher
<br/>
<br/>
<br/>
c.a.p.e. IT ...easy.<br/>
<br/>
<br/>
<strong>WELCOME</strong>
<br/>
<br/>
You are about to UNINSTALL the UserImportExport extensions package for OTRS.<br/>
<br/>
<br/>
<br/>
<br/>
c.a.p.e. IT ...easy.<br/>
<br/>
<br/>
<strong>WELCOME</strong>
<br/>
<br/>
You are about to UPGRADE the UserImportExport extensions package for OTRS.<br/>
<br/>
<br/>
<b>Required OTRS packages:</b><br><br>
ImportExport 5.0.1 or higher
<br/>
<br/>
<br/>
c.a.p.e. IT ...easy.<br/>
<br/>
ImportExport
$Kernel::OM->Get('var::packagesetup::UserImportExport')->CodeInstall();
$Kernel::OM->Get('var::packagesetup::UserImportExport')->CodeUpgrade();
$Kernel::OM->Get('var::packagesetup::UserImportExport')->CodeUninstall();
$Kernel::OM->Get('var::packagesetup::UserImportExport')->CodeReinstall();
2015-11-16 08:04:13
cvs.intra.cape-it.de
IyAkSWQ6IFVzZXJJbXBvcnRFeHBvcnQucG9kLHYgMS41IDIwMTMvMTAvMTYgMTQ6MTU6MDggdHRvIEV4cCAkCgo9aGVhZDEgR2VuZXJhbAoKVGhpcyBwYWNrYWdlIHByb3ZpZGVzIGFuIENTVi1iYXNlZCBpbXBvcnQvZXhwb3J0IGJhY2tlbmQgZm9yIHVzZXIgKHdpdGggY3VzdG9tIHF1ZXVlcywgcm9sZXMgYW5kIHNvbWUgcHJlZmVyZW5jZXMpLgoKVGhpcyBwYWNrYWdlIGF1dG9tYXRpY2FsbHkgY3JlYXRlcyBpbXBvcnQtL2V4cG9ydCBtYXBwaW5nIGR1cmluZyByZS0vaW5zdGFsbGF0aW9uIGZvciBlYWNoIHJlZ2lzdGVyZWQgdXNlci4KCj1vdmVyIDQgCgo9aXRlbSAqICJEZWZhdWx0IEVtYWlsIgoKICAgICAgICBkZWZhdWx0IGVtYWlsIGFkZHJlc3MgaWYgbm90IHNwZWNpZmllZCBpbiBpbXBvcnQgZGF0YQoKPWl0ZW0gKiAiRGVmYXVsdCBQYXNzd29yZC1TdWZmaXgiCgogICAgICAgIGEgcGFzc3dvcmQgd2lsbCBiZSBpZGVudGljYWwgd2l0aCB0aGUgbG9naW4gZm9sbG93ZWQgYnkgdGhpcyBzdWZmaXggaWYgbm90IHNwZWNpZmllZCBpbiBpbXBvcnQgZGF0YQoKPWl0ZW0gKiAiTWF4LiBudW1iZXIgb2YgQ3VzdG9tIFF1ZXVlcwoKICAgICAgICBzcGVjaWZpZXMgaG93IG1hbnkgY3VzdG9tIHF1ZXVlcyBtaWdodCBiZSBpbXBvcnRlZCBvciBleHBvcnRlZCAobnVtYmVyIG9mIGNvbHVtbnMpCgo9aXRlbSAqICJNYXguIG51bWJlciBvZiByb2xlcyIKCiAgICAgICAgc3BlY2lmaWVzIGhvdyBtYW55IHJvbGVzIG1pZ2h0IGJlIGltcG9ydGVkIG9yIGV4cG9ydGVkIChudW1iZXIgb2YgY29sdW1ucykKCj1pdGVtICogIkRlZmF1bHQgVmFsaWRpdHkiCgogICAgICAgIGlmIG5vIHZhbGlkIGluZm9ybWF0aW9uIGlzIHByb3ZpZGVkIGluIGltcG9ydCBkYXRhLCB0aGlzIHZhbGlkIGluZm9ybWF0aW9uIGlzIHVzZWQKCj1iYWNrCgo9aGVhZDEgQ09QWVJJR0hUCgpDb3B5cmlnaHQgKEMpIDIwMDYtMjAxMyBjLmEucC5lLiBJVCBHbWJILCBodHRwOi8vd3d3LmNhcGUtaXQuZGUvCgpUaGUgZm9sbG93aW5nIHBlcnNvbnMgaGF2ZSBjb250cmlidXRlZCB0byB0aGlzIGV4dGVuc2lvbjoKCj1vdmVyIDQKCj1pdGVtICogUmlja3koZG90KUthaXNlcihhdCljYXBlKGRhc2gpaXQoZG90KWRlCgo9aXRlbSAqIEZyYW5rKGRvdClPYmVyZW5kZXIoYXQpY2FwZShkYXNoKWl0KGRvdClkZQoKPWl0ZW0gKiBUb3JzdGVuKGRvdClUaGF1KGF0KWNhcGUoZGFzaClpdChkb3QpZGUKCj1iYWNrCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZSB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91IGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCgo9Y3V0Cg==
IyBDaGFuZ2UgbG9nIG9mIFVzZXJJbXBvcnRFeHBvcnQKKiBDb3B5cmlnaHQgKEMpIDIwMDYtMjAxNSBjLmEucC5lLiBJVCBHbWJILCBodHRwOi8vd3d3LmNhcGUtaXQuZGUvCiogJElkOiBDSEFOR0VTX1VzZXJJbXBvcnRFeHBvcnQubWQsdiAxLjkgMjAxNS8xMS8xNiAwNzowMDoxMCB0bGFuZ2UgRXhwICQKCiMgUk9BRE1BUCBUQVNLUwoKIzUuMC4wICgyMDE1LzExLzE2KQoqICgyMDE1LzExLzExKSAtIENSOiBUMjAxNTA4MjY5MDAwMDU1MyAoYWRkZWQgY2hhbmdlcyBmb3IgT1RSUyA1LjAueCkgKHRsYW5nZSkKIAojIHI0LjAuMCAoMjAxNS8wOC8yNikKKiAoMjAxNS8wNC8xNikgLSBCdWdmaXg6IFQyMDE1MDQxNTkwMDAwNjA3IChmaXggRGVzY3JpcHRpb24gaW4gc29wbSkgKHNyZWlzcykKCiMgcjEuMy4wICgyMDE0LzEyLzAyKQoqICgyMDE0LzEyLzAyKSAtIENSOiBUMjAxNDEwMjk5MDAwMDUyNyAoYWRkZWQgZnJhbWV3b3JrIDQuMCkgKGFsaXR2aW5vdmEpCgojICAxLjIuMCAoMjAxMy8xMC8/PykKKiAoMjAxMy8xMC8xNikgLSBDUjogVDIwMTMxMDE0OTAwMDA4NDIgKGFkZGVkIGZyYW1ld29yayAzLjMpICh0dG8pCiogKDIwMTMvMTAvMTYpIC0gQ1I6IFQyMDEzMTAxNDkwMDAwODQyIChhZGRlZCBhbmQgbW92ZWQgQ0hBTkdFUy1maWxlKSAodHRvKQoKIyAgMS4xLjAgKDIwMTMvMDIvMTgpCiogKDIwMTMvMDIvMTgpIC0gQ1I6IFQyMDEzMDIxMjkwMDAwMTU2IChhZGRlZCBTeXNjb25maWcgb3B0aW9uIGZvciBhdW90bS4gQ1NWLW1hcHBpbmcgY3JlYXRpb24gb24gcmVpbnN0YWxsL3VwZ3JhZGUpIChhbGl0dmlub3ZhKQoqICgyMDEzLzAyLzE4KSAtIENSOiBUMjAxMzAyMTI5MDAwMDE1NiAobW9kaWZpY2F0aW9ucyBmb3IgSVRTTTMuMS43IGFuZCBmcmFtZXdvcmsgT1RSUyAzLjIueCkgKGFsaXR2aW5vdmEpCiogKDIwMTIvMTIvMTcpIC0gQnVnZml4OiBjb3JyZWN0IHNvcG0tRmlsZSAoZm9iZXIpCgojICAxLjAuMCAoMjAxMi8xMi8xMikKKiAoMjAxMi8xMi8xMikgLSBDUjogZmlyc3QgcmVsZWFzZSBmb3IgcHJvZHVjdGl2ZSB1c2UgKHR0bykKKiAoMjAxMi8xMi8xMikgLSBDUjogY29kZSBjbGVhbnVwICh0dG8pICAKCiMgIDAuMS4xICgyMDExLzA5LzI5KQoqICgyMDExLzA5LzI5KSAtIENSOiBzbWFsbCBjaGFuZ2VzIGFib3V0IE91dE9mT2ZmaWNlIChya2Fpc2VyKQoKIyAgMC4xLjAgKDIwMTEvMDkvMjgpCiogKDIwMTEvMDkvMjgpIC0gQ1I6IGZpcnN0IHJlbGVhc2UgZm9yIHRlc3RpbmcgcHVycG9zZSAocmthaXNlcikKCg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQXBwbGljYXRpb24iPgogICAgPENWUz4kSWQ6IFVzZXJJbXBvcnRFeHBvcnQueG1sLHYgMS4zIDIwMTMvMDIvMTggMDk6NDY6NTcgYWxpdHZpbm92YSBFeHAgJDwvQ1ZTPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSW1wb3J0RXhwb3J0OjpPYmplY3RCYWNrZW5kUmVnaXN0cmF0aW9uIyMjVXNlciIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPk9iamVjdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5PYmpla3QtQmFja2VuZCBNb2R1bCBSZWdpc3RyYXRpb24gZGVzIEltcG9ydC9FeHBvcnQgTW9kdWxzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkltcG9ydEV4cG9ydDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPk9iamVjdEJhY2tlbmQ6Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQ6Ok9iamVjdEJhY2tlbmQ6OlVzZXI8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik5hbWUiPlVzZXI8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVXNlckltcG9ydDo6RGVmYXVsdEVtYWlsQWRkcmVzcyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmluZXMgd2hpY2ggZW1haWwgYWRkcmVzcyB0byB1c2UgaWYgbm90IGRlZmluZWQgLSBzdHJvbmdseSBkZXBlbmRzIG9uIGJhY2tlbmQgY29uZmlndXJhdGlvbiEhITwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQgd2VsY2hlIE1haWxhZHJlc3NlIGdlbnV0enQgd2lyZCwgd2VubiBuaWNodCBnZWdlYmVuIC0gc3RhcmsgYWJoYWVuZ2lnIHZvbiBCYWNrZW5ka29uZmlndXJhdGlvbiEhITwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkltcG9ydEV4cG9ydDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPlVzZXJJbXBvcnRFeHBvcnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPmR1bW15QGxvY2FsaG9zdC5jb208L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJVc2VySW1wb3J0OjpEZWZhdWx0UGFzc3dvcmQiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZpbmVzIHdoaWNoIGRlZmF1bHQgcGFzc3dvcmQtc3VmZml4IHRvIHVzZSBmb3IgaW1wb3J0IGlmIG5vdCBkZWZpbmVkIC0gc3Ryb25nbHkgZGVwZW5kcyBvbiBiYWNrZW5kIGNvbmZpZ3VyYXRpb24hISE8L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0IHdlbGNoZXMgU3RhbmRhcmQtUGFzc3dvcnQtU3VmZml4IGbDvHIgSW1wb3J0IGdlbnV0enQgd2lyZCwgd2VubiBuaWNodCBnZWdlYmVuIC0gc3RhcmsgYWJoYWVuZ2lnIHZvbiBCYWNrZW5ka29uZmlndXJhdGlvbiEhITwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkltcG9ydEV4cG9ydDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPlVzZXJJbXBvcnRFeHBvcnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPnBhc3MxMjM8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICAKICAgIDxDb25maWdJdGVtIE5hbWU9IlVzZXJJbXBvcnQ6OkZvcmNlQ1NWTWFwcGluZ1JlY3JlYXRpb24iIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+RGVmaW5lcyBpZiBDU1YtbWFwcGluZ3MgYXJlIHJlLWNyZWF0ZWQgb24gbmV4dCBwYWNrYWdlIHJlaW5zdGFsbGF0aW9uIG9yIHVwZ3JhZGUgaWYgYWxyZWFkeSBleGlzdGVudC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JbXBvcnRFeHBvcnQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Vc2VySW1wb3J0RXhwb3J0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIwIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+Tm88L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPlllczwvSXRlbT4KICAgICAgICAgICAgPC9PcHRpb24+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPiAgICAgICAgCiAgICAKPC9vdHJzX2NvbmZpZz4K
# --
# Kernel/System/ImportExport/ObjectBackend/User.pm
#- import/export backend for User
# Copyright (C) 2006-2015 c.a.p.e. IT GmbH, http://www.cape-it.de
#
# written/edited by:
# * Ricky(dot)Kaiser(at)cape(dash)it(dot)de
# * Torsten(dot)Thau(at)cape(dash)it(dot)de
# * Anna(dot)Litvinova(at)cape(dash)it(dot)de
# --
# $Id: User.pm,v 1.8 2015/09/07 15:29:04 tlange Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

package Kernel::System::ImportExport::ObjectBackend::User;

use strict;
use warnings;
use Kernel::System::User;
use Kernel::System::Valid;
use Kernel::System::Time;
use Kernel::System::Queue;
use Kernel::System::Group;
use Time::Local;

use vars qw($VERSION);
$VERSION = qw($Revision: 1.8 $) [1];

our @ObjectDependencies = (
    'Kernel::System::ImportExport',
    'Kernel::System::User',
    'Kernel::System::Queue',
    'Kernel::System::Group',
    'Kernel::System::Log',
    'Kernel::Config'
);

=head1 NAME

Kernel::System::ImportExport::ObjectBackend::User - import/export backend for User

=head1 SYNOPSIS

All functions to import and export User entries

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::DB;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::ImportExport::ObjectBackend::User;

    my $ConfigObject = Kernel::Config->new();
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $BackendObject = Kernel::System::ImportExport::ObjectBackend::User->new(
        ConfigObject       => $ConfigObject,
        LogObject          => $LogObject,
        DBObject           => $DBObject,
        MainObject         => $MainObject,
        ImportExportObject => $ImportExportObject,
    );

=cut

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

    # allocate new hash for object
    my $Self = {};
    bless( $Self, $Type );

    return $Self;
}

=item ObjectAttributesGet()

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

    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 %Validlist = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();

    my $Attributes = [
        {
            Key   => 'DefaultUserEmail',
            Name  => 'Default Email',
            Input => {
                Type         => 'Text',
                Required     => 0,
                Size         => 50,
                MaxLength    => 250,
                ValueDefault => $Kernel::OM->Get('Kernel::Config')->Get(
                    'UserImport::DefaultEmailAddress',
                    )
            },
        },
        {
            Key   => 'DefaultPassword',
            Name  => 'Password Suffix (pw=login+suffix - only import)',
            Input => {
                Type         => 'Text',
                Required     => 0,
                Size         => 50,
                MaxLength    => 250,
                ValueDefault => $Kernel::OM->Get('Kernel::Config')->Get(
                    'UserImport::DefaultPassword',
                    )
            },
        },
        {
            Key   => 'NumberOfCustomQueues',
            Name  => 'Max. number of Custom Queues',
            Input => {
                Type         => 'Text',
                Required     => 1,
                Size         => 3,
                MaxLength    => 3,
                ValueDefault => '10',
            },
        },
        {
            Key   => 'NumberOfRoles',
            Name  => 'Max. number of roles',
            Input => {
                Type         => 'Text',
                Required     => 1,
                Size         => 3,
                MaxLength    => 3,
                ValueDefault => '10',
            },
        },
        {
            Key   => 'DefaultValid',
            Name  => 'Default Validity',
            Input => {
                Type         => 'Selection',
                Data         => \%Validlist,
                Required     => 1,
                Translation  => 1,
                PossibleNone => 0,
                ValueDefault => 1,
            },
        },
    ];

    return $Attributes;
}

=item 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 = $Kernel::OM->Get('Kernel::System::ImportExport')->ObjectDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );
    my @ElementList = qw{};
    for my $Parameter (
        qw(UserTitle UserLogin UserFirstname UserLastname UserEmail UserPw ValidID
        UserTheme UserLanguage UserComment UserSkin OutOfOffice OutOfOfficeStartYear OutOfOfficeStartMonth
        OutOfOfficeStartDay OutOfOfficeEndYear OutOfOfficeEndMonth OutOfOfficeEndDay UserSendMoveNotification
        UserSendFollowUpNotification UserSendNewTicketNotification UserSendLockTimeoutNotification)
        )
    {
        my $CurrAttribute = {
            Key   => $Parameter,
            Value => $Parameter,
        };

        # if ValidID is available - offer Valid instead..
        if ( $Parameter eq 'ValidID' ) {
            $CurrAttribute = { Key => 'Valid', Value => 'Validity', };
        }

        # if UserPw is available - add note to mapping..
        if ( $Parameter eq 'UserPw' ) {
            $CurrAttribute = {
                Key => 'UserPw',
                Value =>
                    'UserPw (not filled in export, relevant only for import of new entries)',
            };
        }

        # required mapping-elements
        if ( $Parameter eq 'UserLogin' ) {
            $CurrAttribute = {
                Key   => $Parameter,
                Value => "$Parameter (required for import)",
            };
        }
        if ( $Parameter eq 'UserFirstname' || $Parameter eq 'UserLastname' ) {
            $CurrAttribute = {
                Key   => $Parameter,
                Value => "$Parameter (required for import of new entries)",
            };
        }
        push( @ElementList, $CurrAttribute );
    }

=more Preferences:
UserCreateNextMask UserTicketOverviewSmallPageShown UserTicketOverviewPreviewPageShown
UserConfigItemOverviewSmallPageShown UserChangeOverviewSmallPageShown UserRefreshTime UserTicketOverviewMediumPageShown
=cut

    # columns for CustomQueues
    my $NumberOfCustomQueues = $ObjectData->{NumberOfCustomQueues} || 10;
    my $CurrIndex = 0;
    while ( $CurrIndex < $NumberOfCustomQueues ) {

        push(
            @ElementList,
            {
                Key   => 'CustomQueue' . sprintf( "%03d", $CurrIndex ),
                Value => 'CustomQueue' . sprintf( "%03d", $CurrIndex ),
            }
        );

        $CurrIndex++;
    }

    # columns for roles
    my $NumberOfRoles = $ObjectData->{NumberOfRoles} || 10;
    $CurrIndex = 0;
    while ( $CurrIndex < $NumberOfRoles ) {

        push(
            @ElementList,
            {
                Key   => 'Role' . sprintf( "%03d", $CurrIndex ),
                Value => 'Role' . sprintf( "%03d", $CurrIndex ),
            }
        );

        $CurrIndex++;
    }

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

    return $Attributes;
}

=item 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;
}

=item ExportDataGet()

get export data as 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 = $Kernel::OM->Get('Kernel::System::ImportExport')->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 = $Kernel::OM->Get('Kernel::System::ImportExport')->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 =
            $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  => "No valid mapping list found for the template id $Param{TemplateID}",
            );
            return;
        }

        push( @MappingObjectList, $MappingObjectData );
    }

    # search the users...
    my %UserList = $Kernel::OM->Get('Kernel::System::User')->UserSearch(
        Search => '*',
        Valid  => 0,
    );
    my @ExportData;

    # export user ...
    for my $CurrUser ( keys(%UserList) ) {
        my %UserData = $Kernel::OM->Get('Kernel::System::User')->GetUserData(
            UserID        => $CurrUser,
            NoOutOfOffice => 1,
        );

        # prepare validity...
        if ( $UserData{ValidID} ) {
            $UserData{Valid} = $Kernel::OM->Get('Kernel::System::Valid')->ValidLookup(
                ValidID => $UserData{ValidID},
            );
        }

        # prepare password...
        if ( $UserData{UserPw} ) {
            $UserData{UserPw} = '-';
        }

        # prepare preferences
        for my $Argument (
            qw(OutOfOffice UserSendMoveNotification UserSendFollowUpNotification UserSendNewTicketNotification UserSendLockTimeoutNotification)
            )
        {
            if   ( $UserData{$Argument} ) { $UserData{$Argument} = 'yes' }
            else                          { $UserData{$Argument} = 'no' }
        }

        # get CustomQueues
        my @QueueIDs = $Kernel::OM->Get('Kernel::System::Queue')->GetAllCustomQueues(
            UserID => $CurrUser,
        );
        if (@QueueIDs) {
            my $CurrIndex = 0;
            my $NumberOfCustomQueues = $ObjectData->{NumberOfCustomQueues} || 10;
            for my $QueueID (@QueueIDs) {
                if ( $CurrIndex < $NumberOfCustomQueues ) {
                    my $Queue = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
                        QueueID => $QueueID,
                    );
                    $UserData{ 'CustomQueue' . sprintf( "%03d", $CurrIndex ) } = $Queue;
                }
                $CurrIndex++;
            }
        }

        # get roles
        my @RoleIDs = $Kernel::OM->Get('Kernel::System::Group')->GroupUserRoleMemberList(
            UserID => $CurrUser,
            Result => 'ID',
        );
        if (@RoleIDs) {
            my $CurrIndex = 0;
            my $NumberOfRoles = $ObjectData->{NumberOfRoles} || 10;
            for my $RoleID (@RoleIDs) {
                if ( $CurrIndex < $NumberOfRoles ) {
                    my $Role = $Kernel::OM->Get('Kernel::System::Group')
                        ->RoleLookup( RoleID => $RoleID );
                    $UserData{ 'Role' . sprintf( "%03d", $CurrIndex ) } = $Role;
                }
                $CurrIndex++;
            }
        }

        my @CurrRow;
        for my $MappingObject (@MappingObjectList) {
            my $Key = $MappingObject->{Key};
            if ( !$Key ) {
                push @CurrRow, '';
            }
            else {
                push( @CurrRow, $UserData{$Key} || '' );
            }
        }
        push @ExportData, \@CurrRow;

    }
    return \@ExportData;
}

=item ImportDataSave()

import one row of the import data

    my $ConfigItemID = $ObjectBackend->ImportDataSave(
        TemplateID    => 123,
        ImportDataRow => $ArrayRef,
        UserID        => 1,
    );

=cut

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

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

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

    # get object data
    my $ObjectData = $Kernel::OM->Get('Kernel::System::ImportExport')->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 ( undef, 'Failed' );
    }

    # get the mapping list
    my $MappingList = $Kernel::OM->Get('Kernel::System::ImportExport')->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 ( undef, 'Failed' );
    }

    # create the mapping object list
    my @MappingObjectList;
    my %Identifier;
    my $Counter     = 0;
    my %NewUserData = qw{};
    my $UserKey     = "";

    #--------------------------------------------------------------------------
    #BUILD MAPPING TABLE...
    my $IsHeadline = 1;
    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  => "No valid mapping list found for template id $Param{TemplateID}",
            );
            return ( undef, 'Failed' );
        }

        push( @MappingObjectList, $MappingObjectData );
        if (
            $MappingObjectData->{Identifier}
            && $Identifier{ $MappingObjectData->{Key} }
            )
        {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Can't import this entity. "
                    . "'$MappingObjectData->{Key}' has been used multiple "
                    . "times as identifier (line $Param{Counter}).!",
            );
        }
        elsif ( $MappingObjectData->{Identifier} ) {
            $Identifier{ $MappingObjectData->{Key} } =
                $Param{ImportDataRow}->[$Counter];
            $UserKey = $MappingObjectData->{Key};
        }
        $NewUserData{ $MappingObjectData->{Key} } =
            $Param{ImportDataRow}->[$Counter];
        $Counter++;
    }

    #--------------------------------------------------------------------------
    #DO THE IMPORT...
    #(0) Preprocess data...
    # lookup Valid-ID...

    if ( !$NewUserData{ValidID} && $NewUserData{Valid} ) {
        $NewUserData{ValidID} = $Kernel::OM->Get('Kernel::System::Valid')->ValidLookup(
            Valid => $NewUserData{Valid}
        );
    }
    if ( !$NewUserData{ValidID} ) {
        $NewUserData{ValidID} = $ObjectData->{DefaultValid} || 1;
    }

    #(1) search user
    my %UserData = ();
    if ( !$UserKey || $UserKey ne 'UserLogin' ) {
        $UserKey = "UserLogin";
    }

    if ( $NewUserData{$UserKey} ) {
        %UserData = $Kernel::OM->Get('Kernel::System::User')->GetUserData(
            User => $NewUserData{$UserKey}
        );
    }

    # no update of root@localhost
    if ( $UserData{UserID} && $UserData{UserID} == 1 ) {
        next;
    }

    my $NewUser = 1;
    if (%UserData) {
        $NewUser = 0;
    }

    for my $Key ( keys(%NewUserData) ) {
        $UserData{$Key} = $NewUserData{$Key};
    }

    if ( !$UserData{ValidID} ) {
        $UserData{ValidID} = 1;
    }

    #(1) Preprocess data...

    #default UserEmail...
    if ( !$UserData{UserEmail} ) {
        $UserData{UserEmail} = $ObjectData->{DefaultUserEmail}
            || $Kernel::OM->Get('Kernel::Config')->Get(
            'UserImport::DefaultEmailAddress'
            );
    }

    #(2) if user DOES NOT exist => create
    my $Result     = 0;
    my $ReturnCode = "";    # Created | Changed | Failed

    if ($NewUser) {

        # set defaults
        delete $UserData{ID};

        # default UserPw
        if ( !$UserData{UserPw} || $UserData{UserPw} eq '-' ) {
            $UserData{UserPw} = $UserData{UserLogin} . (
                $ObjectData->{DefaultPassword} || $Kernel::OM->Get('Kernel::Config')->Get(
                    'UserImport::DefaultPassword'
                    )
                )
        }
        $Result = $Kernel::OM->Get('Kernel::System::User')->UserAdd(
            %UserData,
            ChangeUserID => $Param{UserID},
        );

        if ( !$Result ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "ImportDataSave: adding User ("
                    . "Login "
                    . $UserData{UserLogin}
                    . ") failed (line $Param{Counter}).",
            );
        }
        else {
            $ReturnCode = "Created";
        }
    }

    #(3) if user DOES exist => update...
    else {
        $UserData{ID} = $NewUserData{$UserKey};

        delete $UserData{UserPw};
        $Result = $Kernel::OM->Get('Kernel::System::User')->UserUpdate(
            %UserData,
            ChangeUserID => $Param{UserID},
        );

        if ( !$Result ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "ImportDataSave: updating User ("
                    . "Login "
                    . $UserData{UserLogin}
                    . ") failed (line $Param{Counter}).",
            );
        }
        else {
            $ReturnCode = "Changed";
        }
    }

    # (4) queues, roles, preferences
    if ($Result) {

        # get UserID
        my $UserID;
        if ($NewUser) {
            $UserID = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
                UserLogin => $UserData{UserLogin},
            );
        }
        else {
            $UserID = $UserData{UserID};
        }

        # set CustomQueues
        # delete existing entries
        $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL  => 'DELETE FROM personal_queues WHERE user_id = ?',
            Bind => [ \$UserID ],
        );

        my $CurrIndex = 0;
        my $NumberOfCustomQueues = $ObjectData->{NumberOfCustomQueues} || 10;
        while ( $CurrIndex < $NumberOfCustomQueues ) {
            if ( $UserData{ 'CustomQueue' . sprintf( "%03d", $CurrIndex ) } ) {

                # get QueueID
                my $QueueID = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
                    Queue => $UserData{ 'CustomQueue' . sprintf( "%03d", $CurrIndex ) }
                );

                # create new entry
                if ($QueueID) {
                    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
                        SQL => 'INSERT INTO personal_queues (user_id, queue_id) VALUES ('
                            . $UserID . ','
                            . $QueueID
                            . ')',
                    );
                    $ReturnCode = "Partially changed - see log for details" if ( !$Success );
                }
                else {
                    $ReturnCode = "Partially changed - see log for details";
                }
            }
            $CurrIndex++;
        }

        # set Preferences
        # check OutOfOffice-Date
        if ( $UserData{OutOfOffice} ) {
            my $Check;
            for (qw(Start End)) {
                my $CheckDate;
                if ( $UserData{ 'OutOfOffice' . $_ . 'Year' } =~ m/\d{4}/ ) {
                    $CheckDate = eval {
                        timelocal(
                            0, 0, 0, $UserData{ 'OutOfOffice' . $_ . 'Day' },
                            ( $UserData{ 'OutOfOffice' . $_ . 'Month' } - 1 ),
                            $UserData{ 'OutOfOffice' . $_ . 'Year' }
                        );
                    };
                }
                if ( !$CheckDate ) {
                    $UserData{ 'OutOfOffice' . $_ . 'Year' }  = '';
                    $UserData{ 'OutOfOffice' . $_ . 'Month' } = '';
                    $UserData{ 'OutOfOffice' . $_ . 'Day' }   = '';
                    if ( $UserData{OutOfOffice} ne 'no' ) {
                        $Kernel::OM->Get('Kernel::System::Log')->Log(
                            Priority => 'error',
                            Message =>
                                'Import: Invalid OutOfOffice' . $_ . '-Date for User '
                                . $UserData{UserLogin},
                        );

                    }
                    $Check = 1;
                }
            }
            if ($Check) { $UserData{OutOfOffice} = 'no' }
        }
        else {
            for (
                qw(OutOfOfficeStartYear OutOfOfficeStartMonth OutOfOfficeStartDay
                OutOfOfficeEndYear OutOfOfficeEndMonth OutOfOfficeEndDay)
                )
            {
                $UserData{$_} = "";
            }
        }
        for my $Preference (
            qw(UserTheme UserLanguage UserComment UserSkin OutOfOfficeStartYear OutOfOfficeStartMonth OutOfOfficeStartDay
            OutOfOfficeEndYear OutOfOfficeEndMonth OutOfOfficeEndDay OutOfOffice UserSendMoveNotification
            UserSendFollowUpNotification UserSendNewTicketNotification UserSendLockTimeoutNotification)
            )
        {

            if ( $UserData{$Preference} ) {

                #check OutOfOffice and UserSend...Notifications
                if ( $Preference =~ m/^UserSend.+/ || $Preference =~ m/^OutOfOffice$/ ) {
                    if ( $UserData{$Preference} eq 'no' ) {
                        $UserData{$Preference} = 0;
                    }
                    else { $UserData{$Preference} = 1; }
                }
                $Kernel::OM->Get('Kernel::System::User')->SetPreferences(
                    Key    => $Preference,
                    Value  => $UserData{$Preference},
                    UserID => $UserID,
                );
            }
        }

=more Preferences:
UserCreateNextMask UserTicketOverviewSmallPageShown UserTicketOverviewPreviewPageShown
UserConfigItemOverviewSmallPageShown UserChangeOverviewSmallPageShown UserRefreshTime UserTicketOverviewMediumPageShown
=cut

        # set roles
        # delete existing entries
        $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL  => 'DELETE FROM role_user WHERE user_id = ?',
            Bind => [ \$UserID ],
        );

        $CurrIndex = 0;
        my $NumberOfRoles = $ObjectData->{NumberOfRoles} || 10;
        while ( $CurrIndex < $NumberOfRoles ) {
            if ( $UserData{ 'Role' . sprintf( "%03d", $CurrIndex ) } ) {

                # get RoleID
                my $RoleID = $Kernel::OM->Get('Kernel::System::Group')->RoleLookup(
                    Role => $UserData{ 'Role' . sprintf( "%03d", $CurrIndex ) },
                );

                #create new entry
                if ($RoleID) {
                    my $Success = $Kernel::OM->Get('Kernel::System::Group')->GroupUserRoleMemberAdd(
                        UID    => $UserID,
                        RID    => $RoleID,
                        Active => 1,
                        UserID => $Param{UserID},
                    );
                    $ReturnCode = "Partially changed - see log for details" if ( !$Success );

                }
                else {
                    $ReturnCode = "Partially changed - see log for details";
                }
            }
            $CurrIndex++;
        }
    }

    #
    #--------------------------------------------------------------------------

    return ( $Result, $ReturnCode );
}

1;
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9Vc2VySW1wb3J0RXhwb3J0LnBtIC0gcHJvdmlkZXMgZ2VybWFuIGxhbmd1YWdlCiMgdHJhbnNsYXRpb24gZm9yIFVzZXJJbXBvcnRFeHBvcnQgbW9kdWxlCiMgQ29weXJpZ2h0IChDKSAyMDA2LTIwMTUgYy5hLnAuZS4gSVQgR21iSCwgaHR0cDovL3d3dy5jYXBlLWl0LmRlCiMKIyB3cml0dGVuL2VkaXRlZCBieToKIyAqIFJpY2t5KGRvdClLYWlzZXIoYXQpY2FwZShkYXNoKWl0KGRvdClkZQojICogVG9yc3Rlbihkb3QpVGhhdShhdCljYXBlKGRhc2gpaXQoZG90KWRlCiMgKiBBbm5hKGRvdClMaXR2aW5vdmEoYXQpY2FwZShkYXNoKWl0KGRvdClkZQojIC0tCiMgJElkOiBkZV9Vc2VySW1wb3J0RXhwb3J0LnBtLHYgMS43IDIwMTUvMDkvMDcgMTU6Mjk6MDQgdGxhbmdlIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZGVfVXNlckltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gJyRSZXZpc2lvbjogMS43ICQnOwokVkVSU0lPTiA9fiBzL15cJC4qOlxXKC4qKVxXLis/JC8kMS87CgojIC0tCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAjIHBvc3NpYmxlIGNoYXJzZXRzCiAgICAkU2VsZi0+e0NoYXJzZXR9ID0gWyd1dGYtOCcsIF07CgogICAgIyAkJFNUQVJUJCQKCiAgICAjIHRyYW5zbGF0aW9ucyBtaXNzaW5nIGluIEltcG9ydEV4cG9ydC4uLgogICAgJExhbmctPnsnQ29sdW1uIFNlcGVyYXRvcid9ICAgICAgICAgICA9ICdTcGFsdGVudHJlbm5lcic7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ1plaWNoZW5zYXR6JzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnRXhwb3J0IG1pdHRlbHMgU3VjaGUgZWluc2NocsOkbmtlbic7CiAgICAkTGFuZy0+eydWYWxpZElEIChub3QgdXNlZCBpbiBpbXBvcnQgYW55bW9yZSwgdXNlIFZhbGlkaXR5IGluc3RlYWQpJ30KICAgICAgICA9ICdWYWxpZElEICh3aXJkIG5pY2h0IGltIEltcG9ydCB2ZXJ3ZW5kZXQsIGJpdHRlIHN0YXR0ZGVzc2VuIFZhbGlkaXR5IG51dHplbic7CiAgICAkTGFuZy0+eydEZWZhdWx0IEVtYWlsJ30gICAgPSAnU3RhbmRhcmQgRW1haWwnOwogICAgJExhbmctPnsnRGVmYXVsdCBWYWxpZGl0eSd9ID0gJ1N0YW5kYXJkIEfDvGx0aWdrZWl0JzsKICAgICRMYW5nLT57J1Bhc3N3b3JkIFN1ZmZpeCAocHc9bG9naW4rc3VmZml4IC0gb25seSBpbXBvcnQpJ30gPSAnUGFzc3dvcnRzdWZmaXggKHB3PUxvZ2luK1N1ZmZpeCknOwogICAgJExhbmctPnsnTWF4LiBudW1iZXIgb2YgQ3VzdG9tIFF1ZXVlcyd9ICAgICAgICAgICAgPSAnTWF4LiBBbnphaGwgTWVpbmUgUXVldWVzJzsKICAgICRMYW5nLT57J01heC4gbnVtYmVyIG9mIHJvbGVzJ30gICAgICAgICAgICAgICAgICAgID0gJ01heC4gQW56YWhsIGRlciBSb2xsZW4nOwogICAgJExhbmctPnsnUGFydGlhbGx5IGNoYW5nZWQgLSBzZWUgbG9nIGZvciBkZXRhaWxzJ30gPSAnVGVpbHdlaXNlIGdlw6RuZGVydCAtIHNpZWhlIExvZyc7CiAgICAkTGFuZy0+eycnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwoKICAgIHJldHVybiAwOwoKICAgICMgJCRTVE9QJCQKfQoKIyAtLQoxOwo=
# --
# UserImportExport.pm - code run during package de-/installation
# Copyright (C) 2006-2015 c.a.p.e. IT GmbH, http://www.cape-it.de
#
# written/edited by:
# * Ricky(dot)Kaiser(at)cape(dash)it(dot)de
# * Torsten(dot)Thau(at)cape(dash)it(dot)de
# * Anna(dot)Litvinova(at)cape(dash)it(dot)de
#
# --
# $Id: UserImportExport.pm,v 1.8 2015/09/07 15:29:04 tlange Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

package var::packagesetup::UserImportExport;

use strict;
use warnings;

use vars qw(@ISA $VERSION);
$VERSION = qw($Revision: 1.8 $) [1];

our @ObjectDependencies = (
    'Kernel::System::ImportExport',
    'Kernel::System::User',
    'Kernel::System::Log',
    'Kernel::Config'
);

=head1 NAME

UserImportExport.pm - code to excecute during package installation

=head1 SYNOPSIS

All functions

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::Time;
    use Kernel::System::DB;
    use Kernel::System::XML;

    my $ConfigObject = Kernel::Config->new();
    my $LogObject    = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $TimeObject = Kernel::System::Time->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $XMLObject = Kernel::System::XML->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
    );
    my $CodeObject = var::packagesetup::UserImportExport.pm->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
        TimeObject   => $TimeObject,
        DBObject     => $DBObject,
        XMLObject    => $XMLObject,
    );

=cut

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

    my $Self = {};
    bless( $Self, $Type );

    return $Self;
}

=item CodeInstall()

run the code install part

    my $Result = $CodeObject->CodeInstall();

=cut

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

    $Self->_CreateMappings();

    return 1;
}

=item CodeReinstall()

run the code reinstall part

    my $Result = $CodeObject->CodeReinstall();

=cut

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

    $Self->_CreateMappings();

    return 1;
}

=item CodeUpgrade()

run the code upgrade part

    my $Result = $CodeObject->CodeUpgrade();

=cut

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

    $Self->_CreateMappings();
    
    return 1;
}

=item CodeUninstall()

run the code uninstall part

    my $Result = $CodeObject->CodeUninstall();

=cut

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

    $Self->_RemoveRelatedMappings();

    return 1;
}

sub _RemoveRelatedMappings() {

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

    my $TemplateList = $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateList(
        Object => 'User',
        Format => 'CSV',
        UserID => 1,
    );

    if ( ref($TemplateList) eq 'ARRAY' && @{$TemplateList} ) {
        $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateDelete(
            TemplateID => $TemplateList,
            UserID     => 1,
        );
    }

    return 1;
}

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

    my $TemplateObject = "User";
    my $TemplateName   = "User - ImportExport (auto-created map)";
    my %TemplateList   = ();

    # get config option
    my $ForceCSVMappingConfiguration = $Kernel::OM->Get('Kernel::Config')->Get(
        'UserImport::ForceCSVMappingRecreation'
    ) || '0';

    #---------------------------------------------------------------------------
    # add a template user
    # get list of all templates
    my $TemplateListRef = $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateList(
        Object => $TemplateObject,
        Format => 'CSV',
        UserID => 1,
    );

    # get data for each template and build hash with key = template name; value = template ID
    if ( $TemplateListRef && ref($TemplateListRef) eq 'ARRAY' ) {
        for my $CurrTemplateID ( @{$TemplateListRef} ) {
            my $TemplateDataRef = $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateGet(
                TemplateID => $CurrTemplateID,
                UserID     => 1,
            );
            if (
                $TemplateDataRef
                && ref($TemplateDataRef) eq 'HASH'
                && $TemplateDataRef->{Object}
                && $TemplateDataRef->{Name}
                )
            {
                $TemplateList{ $TemplateDataRef->{Object} . '::' . $TemplateDataRef->{Name} }
                    = $CurrTemplateID;
            }
        }
    }

    #---------------------------------------------------------------------------
    # add a template user
    my $TemplateID;

    # check if template already exists...
    if ( $TemplateList{ $TemplateObject . '::' . $TemplateName } ) {
        if ($ForceCSVMappingConfiguration) {

            # delete old template
            $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateDelete(
                TemplateID => $TemplateList{ $TemplateObject . '::' . $TemplateName },
                UserID     => 1,
            );
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "CSV mapping deleted for re-creation <"
                    . $TemplateName
                    . ">.",
            );

            # create new template
            $TemplateID = $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateAdd(
                Object  => $TemplateObject,
                Format  => 'CSV',
                Name    => $TemplateName,
                Comment => "Automatically created during UserImportExport installation",
                ValidID => 1,
                UserID  => 1,
            );
        }
        else {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "CSV mapping already exists and not re-created <"
                    . $TemplateName
                    . ">.",
            );
            return 1;
        }
    }
    else {

        # create new template
        $TemplateID = $Kernel::OM->Get('Kernel::System::ImportExport')->TemplateAdd(
            Object  => $TemplateObject,
            Format  => 'CSV',
            Name    => $TemplateName,
            Comment => "Automatically created during UserImportExport installation",
            ValidID => 1,
            UserID  => 1,
        );
    }

    #-----------------------------------------------------------------------
    # create attribute mapping...
    my @ElementList = qw{};
    for my $Parameter (
        qw(UserTitle UserLogin UserFirstname UserLastname UserEmail UserPw ValidID UserTheme UserLanguage
        UserComment UserSkin OutOfOffice OutOfOfficeStartYear OutOfOfficeStartMonth
        OutOfOfficeStartDay OutOfOfficeEndYear OutOfOfficeEndMonth OutOfOfficeEndDay UserSendMoveNotification
        UserSendFollowUpNotification UserSendNewTicketNotification UserSendLockTimeoutNotification)
        )
    {
        my $CurrAttribute = {
            Key   => $Parameter,
            Value => $Parameter,
        };

        # if ValidID is available - offer Valid instead..
        if ( $Parameter eq 'ValidID' ) {
            $CurrAttribute = { Key => 'Valid', Value => 'Validity', };
        }

        push( @ElementList, $CurrAttribute );

    }

    # for CustomQueues
    for my $Parameter ( 0 .. 9 ) {
        $Parameter = 'CustomQueue' . sprintf( "%03d", $Parameter );
        my $CurrAttribute = {
            Key   => $Parameter,
            Value => $Parameter,
        };
        push( @ElementList, $CurrAttribute );
    }

    # for roles
    for my $Parameter ( 0 .. 9 ) {
        $Parameter = 'Role' . sprintf( "%03d", $Parameter );
        my $CurrAttribute = {
            Key   => $Parameter,
            Value => $Parameter,
        };
        push( @ElementList, $CurrAttribute );
    }

    my $ExportDataSets = [
        {
            SourceExportData => {
                FormatData => {
                    ColumnSeparator      => 'Semicolon',
                    Charset              => 'UTF-8',
                    IncludeColumnHeaders => 1,
                },

                MappingObjectData => \@ElementList,
                ExportDataGet     => {
                    TemplateID => $TemplateID,
                    UserID     => 1,
                },
            },
        }
    ];

    # get object attributes
    my $ObjectAttributeList = $Kernel::OM->Get('Kernel::System::ImportExport')->ObjectAttributesGet(
        TemplateID => $ExportDataSets->[0]->{SourceExportData}->{ExportDataGet}->{TemplateID},
        UserID     => 1,
    );

    # ----------------------------------------------------------------------
    # run general ExportDataGet...
    EXPORTDATASET:
    for my $CurrentExportDataSet ( @{$ExportDataSets} ) {

        # check SourceExportData attribute
        if (
            !$CurrentExportDataSet->{SourceExportData}
            || ref $CurrentExportDataSet->{SourceExportData} ne 'HASH'
            )
        {

            next EXPORTDATASET;
        }

        # set the format data
        if (
            $CurrentExportDataSet->{SourceExportData}->{FormatData}
            && ref $CurrentExportDataSet->{SourceExportData}->{FormatData} eq 'HASH'
            && $CurrentExportDataSet->{SourceExportData}->{ExportDataGet}->{TemplateID}
            )
        {

            # save format data
            $Kernel::OM->Get('Kernel::System::ImportExport')->FormatDataSave(
                TemplateID =>
                    $CurrentExportDataSet->{SourceExportData}->{ExportDataGet}->{TemplateID},
                FormatData => $CurrentExportDataSet->{SourceExportData}->{FormatData},
                UserID     => 1,
            );
        }

        # set the mapping object data
        if (
            $CurrentExportDataSet->{SourceExportData}->{MappingObjectData}
            && ref $CurrentExportDataSet->{SourceExportData}->{MappingObjectData} eq 'ARRAY'
            && $CurrentExportDataSet->{SourceExportData}->{ExportDataGet}->{TemplateID}
            )
        {

            # delete all existing mapping data
            $Kernel::OM->Get('Kernel::System::ImportExport')->MappingDelete(
                TemplateID =>
                    $CurrentExportDataSet->{SourceExportData}->{ExportDataGet}->{TemplateID},
                UserID => 1,
            );

            # add the mapping object rows
            MAPPINGOBJECTDATA:
            for my $MappingObjectData (
                @{ $CurrentExportDataSet->{SourceExportData}->{MappingObjectData} }
                )
            {

                # add a new mapping row
                my $MappingID = $Kernel::OM->Get('Kernel::System::ImportExport')->MappingAdd(
                    TemplateID =>
                        $CurrentExportDataSet->{SourceExportData}->{ExportDataGet}
                        ->{TemplateID},
                    UserID => 1,
                );

                # add the mapping object data
                $Kernel::OM->Get('Kernel::System::ImportExport')->MappingObjectDataSave(
                    MappingID         => $MappingID,
                    MappingObjectData => $MappingObjectData,
                    UserID            => 1,
                );
            }
        }

    }

    return 1;

}

1;