<?xml version="1.0" encoding="utf-8" ?>
<otrs_package version="1.1">
    <Name>CustomerMultitenancy</Name>
    <Version>11.0.4</Version>
    <Vendor>Rother OSS GmbH</Vendor>
    <URL>https://otobo.io/</URL>
    <License>GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007</License>
    <ChangeLog Date="2026-04-27 14:07:55" Version="11.0.4">Update to OTOBO 11.0.16.</ChangeLog>
    <ChangeLog Date="2025-02-26 13:48:09" Version="11.0.3">Updated to OTOBO 11.0.8.</ChangeLog>
    <ChangeLog Date="2024-10-07 08:38:09" Version="11.0.2">Updated to OTOBO 11.0.6.</ChangeLog>
    <ChangeLog Date="2024-04-02 13:45:01" Version="11.0.1">Initial Release on OTOBO 11.</ChangeLog>
    <Description Lang="en">Group-based multitenancy for customer and customer user.</Description>
    <Description Lang="de">Gruppenbasierte Mandantenfähigkeit für Kunden und Kundenbenutzer.</Description>
    <Framework>11.0.x</Framework>
    <BuildCommitID>b0fbe78c06caf964467614d79d8f238a58c740e0</BuildCommitID>
    <BuildDate>2026-04-27 14:07:58</BuildDate>
    <BuildHost>opms.rother-oss.com</BuildHost>
    <Filelist>
        <File Location="Custom/Kernel/Modules/AdminCustomerCompany.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AdminCustomerCompany.pm
# --
# 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::Modules::AdminCustomerCompany;

use v5.24;
use strict;
use warnings;
use namespace::autoclean;

# core modules
use List::Util qw(any);

# CPAN modules

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

our $ObjectManagerDisabled = 1;

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

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

    my $DynamicFieldConfigs = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerCompany',
    );

    # set pref for columns key
    $Self->{PrefKeyIncludeInvalid} = 'IncludeInvalid' . '-' . $Self->{Action};

    my %Preferences = $Kernel::OM->Get('Kernel::System::User')->GetPreferences(
        UserID => $Self->{UserID},
    );

    $Self->{IncludeInvalid} = $Preferences{ $Self->{PrefKeyIncludeInvalid} };

    $Self->{DynamicFieldLookup} = { map { $_->{Name} => $_ } @{$DynamicFieldConfigs} };

    return $Self;
}

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

    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $Nav               = $ParamObject->GetParam( Param => 'Nav' ) || 0;
    my $NavigationBarType = $Nav eq 'Agent' ? 'Customers' : 'Admin';
    my $Search            = $ParamObject->GetParam( Param => 'Search' );
    $Search
        ||= $ConfigObject->Get('AdminCustomerCompany::RunInitialWildcardSearch') ? '*' : '';
    my $LayoutObject          = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $CustomerCompanyObject = $Kernel::OM->Get('Kernel::System::CustomerCompany');

    my %GetParam;
    $GetParam{Source}         = $ParamObject->GetParam( Param => 'Source' ) || 'CustomerCompany';
    $GetParam{IncludeInvalid} = $ParamObject->GetParam( Param => 'IncludeInvalid' );

    if ( defined $GetParam{IncludeInvalid} ) {
        $Kernel::OM->Get('Kernel::System::User')->SetPreferences(
            UserID => $Self->{UserID},
            Key    => $Self->{PrefKeyIncludeInvalid},
            Value  => $GetParam{IncludeInvalid},
        );

        $Self->{IncludeInvalid} = $GetParam{IncludeInvalid};
    }

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to set multitenancy.
    if ( $ConfigObject->Get('Multitenancy') ) {
        my $GroupObject = $Kernel::OM->Get('Kernel::System::Group');

        my $MultitenancyGroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
            Group => $ConfigObject->Get('Multitenancy::PermissionGroup'),
        );

        my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
            UserID => $Self->{UserID},
            Type   => 'rw',
        );

        if ( $Groups{$MultitenancyGroupID} ) {
            $Self->{MultitenancyPermission} = 1;
        }
    }
# EO CustomerMultitenancy

    # ------------------------------------------------------------ #
    # change
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'Change' ) {
        my $CustomerID   = $ParamObject->GetParam( Param => 'CustomerID' )   || $ParamObject->GetParam( Param => 'ID' ) || '';
        my $Notification = $ParamObject->GetParam( Param => 'Notification' ) || '';
        my %Data         = $CustomerCompanyObject->CustomerCompanyGet(
            CustomerID => $CustomerID,
        );
        $Data{CustomerCompanyID} = $CustomerID;
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar(
            Type => $NavigationBarType,
        );
        $Output .= $LayoutObject->Notify( Info => Translatable('Customer company updated!') )
            if ( $Notification && $Notification eq 'Update' );
        $Self->_Edit(
            Action => 'Change',
            Nav    => $Nav,
            %Data,
        );
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerCompany',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------ #
    # change action
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'ChangeAction' ) {

        # challenge token check for write action
        $LayoutObject->ChallengeTokenCheck();

        my %Errors;
        $GetParam{CustomerCompanyID} = $ParamObject->GetParam( Param => 'CustomerCompanyID' );

        my @CustomerCompanyMap = $ConfigObject->Get( $GetParam{Source} )->{Map}->@*;

        # The readonly fields should not be settable from the WebApp.
        # So update with the old values, regardless what was passed from the client.
        # The old data is only needed when there are any readonly fields.
        my %OldData;
        if ( any { $_->[7] } @CustomerCompanyMap ) {
            %OldData = $CustomerCompanyObject->CustomerCompanyGet(
                CustomerID => $GetParam{CustomerCompanyID},
            );
        }

        # Get dynamic field backend object.
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        ENTRY:
        for my $Entry (@CustomerCompanyMap) {

            # check dynamic fields
            if ( $Entry->[5] eq 'dynamic_field' ) {

                my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "DynamicField $Entry->[2] not found!",
                    );
                    next ENTRY;
                }

                my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    ParamObject        => $ParamObject,
                    Mandatory          => $Entry->[4],
                );

                if ( $ValidationResult->{ServerError} ) {
                    $Errors{ $Entry->[0] } = $ValidationResult;
                }
                else {

                    # generate storable value of dynamic field edit field
                    $GetParam{ $Entry->[0] } = $DynamicFieldBackendObject->EditFieldValueGet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ParamObject        => $ParamObject,
                        LayoutObject       => $LayoutObject,
                    );
                }
            }

            # reuse the old data for readonly field
            elsif ( $Entry->[7] ) {
                $GetParam{ $Entry->[0] } = $OldData{ $Entry->[0] };
            }

            # check remaining non-dynamic-field mandatory fields
            else {
                $GetParam{ $Entry->[0] } = $ParamObject->GetParam( Param => $Entry->[0] ) // '';
                if ( !$GetParam{ $Entry->[0] } && $Entry->[4] ) {
                    $Errors{ $Entry->[0] . 'Invalid' } = 'ServerError';
                }
            }
        }

        if ( !defined $GetParam{CustomerID} ) {
            $GetParam{CustomerID} = $ParamObject->GetParam( Param => 'CustomerID' ) || '';
        }

        # check for duplicate entries
        if ( $GetParam{CustomerCompanyID} ne $GetParam{CustomerID} ) {

            # get CustomerCompany list
            my %List = $CustomerCompanyObject->CustomerCompanyList(
                Search => $Param{Search},
                Valid  => 0,
            );

            # check duplicate field
            if ( %List && $List{ $GetParam{CustomerID} } ) {
                $Errors{Duplicate} = 'ServerError';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            # update group
            my $Update = $CustomerCompanyObject->CustomerCompanyUpdate( %GetParam, UserID => $Self->{UserID} );

            if ($Update) {

                my $SetDFError;

                # set dynamic field values
                ENTRY:
                for my $Entry (@CustomerCompanyMap) {
                    next ENTRY if $Entry->[5] ne 'dynamic_field';

                    my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                    if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                        $SetDFError .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Dynamic field %s not found!',
                                $Entry->[2],
                            ),
                        );

                        next ENTRY;
                    }

                    my $ValueSet = $DynamicFieldBackendObject->ValueSet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ObjectName         => $GetParam{CustomerID},
                        Value              => $GetParam{ $Entry->[0] },
                        UserID             => $Self->{UserID},
                    );

                    if ( !$ValueSet ) {
                        $SetDFError .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Unable to set value for dynamic field %s!',
                                $Entry->[2],
                            ),
                        );

                        next ENTRY;
                    }
                }

                my $ContinueAfterSave = $ParamObject->GetParam( Param => 'ContinueAfterSave' ) || 0;

                # if set DF error exists, create notification
                if ($SetDFError) {

                    # if the user would like to continue editing the customer company, just redirect to the edit screen
                    if ( $ContinueAfterSave eq '1' ) {
                        $Self->_Edit(
                            Action => 'Change',
                            Nav    => $Nav,
                            Errors => \%Errors,
                            %GetParam,
                        );
                    }
                    else {
                        $Self->_Overview(
                            Nav    => $Nav,
                            Search => $Search,
                            %GetParam,
                        );
                    }
                    my $Output = $LayoutObject->Header();
                    $Output .= $LayoutObject->NavigationBar(
                        Type => $NavigationBarType,
                    );
                    $Output .= $LayoutObject->Notify( Info => Translatable('Customer company updated!') );
                    $Output .= $SetDFError;
                    $Output .= $LayoutObject->Output(
                        TemplateFile => 'AdminCustomerCompany',
                        Data         => \%Param,
                    );
                    $Output .= $LayoutObject->Footer();
                    return $Output;
                }

                # if the user would like to continue editing the customer company, just redirect to the edit screen
                if ( $ContinueAfterSave eq '1' ) {
                    my $CustomerID = $ParamObject->GetParam( Param => 'CustomerID' ) || '';
                    return $LayoutObject->Redirect(
                        OP =>
                            "Action=$Self->{Action};Subaction=Change;CustomerID=" . $LayoutObject->LinkEncode($CustomerID) . ";Nav=$Nav;Notification=Update"
                    );
                }
                else {

                    # otherwise return to overview
                    return $LayoutObject->Redirect( OP => "Action=$Self->{Action};Notification=Update" );
                }
            }
        }

        # something went wrong
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar(
            Type => $NavigationBarType,
        );

        $Output .= $LayoutObject->Notify( Priority => 'Error' );

        # set notification for duplicate entry
        if ( $Errors{Duplicate} ) {
            $Output .= $LayoutObject->Notify(
                Priority => 'Error',
                Info     => $LayoutObject->{LanguageObject}->Translate(
                    'Customer Company %s already exists!',
                    $GetParam{CustomerID},
                ),
            );
        }

        $Self->_Edit(
            Action => 'Change',
            Nav    => $Nav,
            Errors => \%Errors,
            %GetParam,
        );
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerCompany',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------ #
    # add
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Add' ) {
        $GetParam{Name} = $ParamObject->GetParam( Param => 'Name' );
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar(
            Type => $NavigationBarType,
        );
        $Self->_Edit(
            Action => 'Add',
            Nav    => $Nav,
            %GetParam,
        );
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerCompany',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------ #
    # add action
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'AddAction' ) {

        # challenge token check for write action
        $LayoutObject->ChallengeTokenCheck();

        my %Errors;

        my $CustomerCompanyKey = $ConfigObject->Get( $GetParam{Source} )->{CustomerCompanyKey};
        my $CustomerCompanyID;

        # Get dynamic field backend object.
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        ENTRY:
        for my $Entry ( @{ $ConfigObject->Get( $GetParam{Source} )->{Map} } ) {

            # check dynamic fields
            if ( $Entry->[5] eq 'dynamic_field' ) {

                my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "DynamicField $Entry->[2] not found!",
                    );
                    next ENTRY;
                }

                my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    ParamObject        => $ParamObject,
                    Mandatory          => $Entry->[4],
                );

                if ( $ValidationResult->{ServerError} ) {
                    $Errors{ $Entry->[0] } = $ValidationResult;
                }
                else {

                    # generate storable value of dynamic field edit field
                    $GetParam{ $Entry->[0] } = $DynamicFieldBackendObject->EditFieldValueGet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ParamObject        => $ParamObject,
                        LayoutObject       => $LayoutObject,
                    );
                }
            }

            # check remaining non-dynamic-field mandatory fields
            else {
                $GetParam{ $Entry->[0] } = $ParamObject->GetParam( Param => $Entry->[0] ) // '';
                if ( !$GetParam{ $Entry->[0] } && $Entry->[4] ) {
                    $Errors{ $Entry->[0] . 'Invalid' } = 'ServerError';
                }
            }

            # save customer company key for checking duplicate
            if ( $Entry->[2] eq $CustomerCompanyKey ) {
                $CustomerCompanyID = $GetParam{ $Entry->[0] };
            }
        }

        # get CustomerCompany list
        my %List = $CustomerCompanyObject->CustomerCompanyList(
            Search => $Param{Search},
            Valid  => 0,
        );

        # check duplicate field
        if ( %List && $List{$CustomerCompanyID} ) {
            $Errors{Duplicate} = 'ServerError';
        }

        # if no errors occurred
        if ( !%Errors ) {

            # add company
            if (
                $CustomerCompanyObject->CustomerCompanyAdd(
                    %GetParam,
                    UserID => $Self->{UserID},
                )
                )
            {

                $Self->_Overview(
                    Nav    => $Nav,
                    Search => $Search,
                    %GetParam,
                );
                my $Output = join '',
                    $LayoutObject->Header,
                    $LayoutObject->NavigationBar(
                        Type => $NavigationBarType,
                    ),
                    $LayoutObject->Notify(
                        Info => Translatable('Customer company added!'),
                    );

                # set dynamic field values
                ENTRY:
                for my $Entry ( @{ $ConfigObject->Get( $GetParam{Source} )->{Map} } ) {
                    next ENTRY if $Entry->[5] ne 'dynamic_field';

                    my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                    if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                        $Output .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Dynamic field %s not found!',
                                $Entry->[2],
                            ),
                        );

                        next ENTRY;
                    }

                    my $ValueSet = $DynamicFieldBackendObject->ValueSet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ObjectName         => $GetParam{CustomerID},
                        Value              => $GetParam{ $Entry->[0] },
                        UserID             => $Self->{UserID},
                    );

                    if ( !$ValueSet ) {
                        $Output .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Unable to set value for dynamic field %s!',
                                $Entry->[2],
                            ),
                        );

                        next ENTRY;
                    }
                }

                $Output .= $LayoutObject->Output(
                    TemplateFile => 'AdminCustomerCompany',
                    Data         => \%Param,
                );
                $Output .= $LayoutObject->Footer();

                return $Output;
            }
        }

        # something went wrong
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar(
            Type => $NavigationBarType,
        );

        $Output .= $LayoutObject->Notify( Priority => 'Error' );

        # set notification for duplicate entry
        if ( $Errors{Duplicate} ) {
            $Output .= $LayoutObject->Notify(
                Priority => 'Error',
                Info     => $LayoutObject->{LanguageObject}->Translate(
                    'Customer Company %s already exists!',
                    $CustomerCompanyID,
                ),
            );
        }

        $Self->_Edit(
            Action => 'Add',
            Nav    => $Nav,
            Errors => \%Errors,
            %GetParam,
        );
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerCompany',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------
    # overview
    # ------------------------------------------------------------
    else {
        $Self->_Overview(
            Nav    => $Nav,
            Search => $Search,
            %GetParam,
        );
        my $Output       = $LayoutObject->Header();
        my $Notification = $ParamObject->GetParam( Param => 'Notification' ) || '';
        $Output .= $LayoutObject->NavigationBar(
            Type => $NavigationBarType,
        );
        $Output .= $LayoutObject->Notify( Info => Translatable('Customer company updated!') )
            if ( $Notification && $Notification eq 'Update' );

        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerCompany',
            Data         => \%Param,
        );

        $Output .= $LayoutObject->Footer();
        return $Output;
    }
}

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

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    $LayoutObject->Block(
        Name => 'Overview',
        Data => \%Param,
    );

    $LayoutObject->Block( Name => 'ActionList' );
    $LayoutObject->Block(
        Name => 'ActionOverview',
        Data => \%Param,
    );

    $LayoutObject->Block(
        Name => 'OverviewUpdate',
        Data => \%Param,
    );

    # send parameter ReadOnly to JS object
    $LayoutObject->AddJSData(
        Key   => 'ReadOnly',
        Value => $ConfigObject->{ $Param{Source} }->{ReadOnly},
    );

    # Get valid object.
    my $ValidObject = $Kernel::OM->Get('Kernel::System::Valid');

    $Param{'ValidOption'} = $LayoutObject->BuildSelection(
        Data       => { $ValidObject->ValidList(), },
        Name       => 'ValidID',
        Class      => 'Modernize',
        SelectedID => $Param{ValidID},
    );

    # Get needed objects.
    my $ParamObject               = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    ENTRY:
    for my $Entry ( @{ $ConfigObject->Get( $Param{Source} )->{Map} } ) {
        if ( $Entry->[0] ) {

            # Handle dynamic fields
            if ( $Entry->[5] eq 'dynamic_field' ) {

                my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                next ENTRY if !IsHashRefWithData($DynamicFieldConfig);

                # Get HTML for dynamic field
                my $DynamicFieldHTML = $DynamicFieldBackendObject->EditFieldRender(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    Value              => $Param{ $Entry->[0] } ? $Param{ $Entry->[0] } : undef,
                    Mandatory          => $Entry->[4],
                    LayoutObject       => $LayoutObject,
                    ParamObject        => $ParamObject,

                    # Server error, if any
                    %{ $Param{Errors}->{ $Entry->[0] } },
                );

                # skip fields for which HTML could not be retrieved
                next ENTRY if !IsHashRefWithData($DynamicFieldHTML);

                $LayoutObject->Block(
                    Name => 'PreferencesGeneric',
                    Data => {},
                );

                $LayoutObject->Block(
                    Name => 'DynamicField',
                    Data => {
                        Name  => $DynamicFieldConfig->{Name},
                        Label => $DynamicFieldHTML->{Label},
                        Field => $DynamicFieldHTML->{Field},
                    },
                );

                next ENTRY;
            }

            my $Block = 'Input';

            # build selections or input fields
            if ( $ConfigObject->Get( $Param{Source} )->{Selections}->{ $Entry->[0] } ) {
                my $OptionRequired = '';
                if ( $Entry->[4] ) {
                    $OptionRequired = 'Validate_Required';
                }

                # build ValidID string
                $Block = 'Option';
                $Param{Option} = $LayoutObject->BuildSelection(
                    Data =>
                        $ConfigObject->Get( $Param{Source} )->{Selections}
                        ->{ $Entry->[0] },
                    Name  => $Entry->[0],
                    Class => "$OptionRequired Modernize " .
                        ( $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '' ),
                    Translation => 1,
                    Sort        => 'AlphanumericKey',
                    SelectedID  => $Param{ $Entry->[0] },
                    Max         => 35,
                );

            }
            elsif ( $Entry->[0] =~ m/^CustomerCompanyCountry/i ) {

                # build Country selection with English names
                $Block = 'Option';
                my $OptionRequired = $Entry->[4] ? 'Validate_Required' : '';
                my $CountryList;
                if ( $ConfigObject->Get('ReferenceData::TranslatedCountryNames') ) {

                    # Flag+Name => code
                    $CountryList = $Kernel::OM->Get('Kernel::System::ReferenceData')->CLDRCountryList(
                        Language => $LayoutObject->{UserLanguage},
                    );

                    # Make sure that the previous value exists in the selection list even if it isn't a country code.
                    my $PreviousCountry = $Param{ $Entry->[0] };
                    if ($PreviousCountry) {
                        $CountryList->{$PreviousCountry} //= $PreviousCountry;
                    }
                }
                else {

                    # English name => English name
                    $CountryList = $Kernel::OM->Get('Kernel::System::ReferenceData')->CountryList;
                }

                $Param{Option} = $LayoutObject->BuildSelection(
                    Data         => $CountryList,
                    PossibleNone => 1,
                    Sort         => 'AlphanumericValue',
                    Name         => $Entry->[0],
                    Class        => "$OptionRequired Modernize " .
                        ( $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '' ),
                    SelectedID => ( $Param{ $Entry->[0] } // 1 ),
                );
            }
            elsif ( $Entry->[0] =~ m/^ValidID/i ) {

                # build ValidID string
                $Block = 'Option';
                my $OptionRequired = $Entry->[4] ? 'Validate_Required' : '';
                $Param{Option} = $LayoutObject->BuildSelection(
                    Data  => { $ValidObject->ValidList(), },
                    Name  => $Entry->[0],
                    Class => "$OptionRequired Modernize " .
                        ( $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '' ),
                    SelectedID => defined( $Param{ $Entry->[0] } ) ? $Param{ $Entry->[0] } : 1,
                );
            }
# Rother OSS / CustomerMultitenancy
            # Build the group field.
            elsif ( $Entry->[0] =~ /^UserGroupID$/i ) {
                # Check if the user has the permission to see/change the multitenancy field.
                if ( !$Self->{MultitenancyPermission} ) {
                    next ENTRY;
                } else {
                    # Build the field.
                    $Block = 'Option';
                    $Param{Option} = $LayoutObject->BuildSelection(
                        Data => {
                            $Kernel::OM->Get('Kernel::System::Group')->GroupList(
                                Valid => 1,
                            )
                        },
                        Name         => $Entry->[0],
                        PossibleNone => 1,
                        Class        => 'Modernize ' . ( $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '' ),
                        SelectedID   => $Param{ $Entry->[0] } || '',
                    );
                }
            }
# EO CustomerMultitenancy
            else {
                $Param{Value} = $Param{ $Entry->[0] } || '';
            }

            # show required flag
            if ( $Entry->[4] ) {
                $Param{MandatoryClass} = 'class="Mandatory"';
                $Param{StarLabel}      = '<span class="Marker">*</span>';
                $Param{RequiredClass}  = 'Validate_Required';
            }
            else {
                $Param{MandatoryClass} = '';
                $Param{StarLabel}      = '';
                $Param{RequiredClass}  = '';
            }

            # show readonly flag
            if ( $Entry->[7] ) {
                $Param{ReadOnlyType} = 'readonly';
            }
            else {
                $Param{ReadOnlyType} = '';
            }

            # add form option
            if ( $Param{Type} && $Param{Type} eq 'hidden' ) {
                $Param{Preferences} .= $Param{Value};
            }
            else {
                $LayoutObject->Block(
                    Name => 'PreferencesGeneric',
                    Data => {
                        Item => $Entry->[1],
                        %Param
                    },
                );
                $LayoutObject->Block(
                    Name => "PreferencesGeneric$Block",
                    Data => {
                        %Param,
                        Item         => $Entry->[1],
                        Name         => $Entry->[0],
                        Value        => $Param{ $Entry->[0] },
                        InvalidField => $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '',
                    },
                );
                if ( $Entry->[4] ) {
                    $LayoutObject->Block(
                        Name => "PreferencesGeneric${Block}Required",
                        Data => {
                            Name => $Entry->[0],
                        },
                    );
                }
            }
        }
    }

    return 1;
}

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

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    $LayoutObject->Block(
        Name => 'Overview',
        Data => \%Param,
    );

    $LayoutObject->Block(
        Name => 'IncludeInvalid',
        Data => {
            IncludeInvalid        => $Self->{IncludeInvalid},
            IncludeInvalidChecked => $Self->{IncludeInvalid} ? 'checked' : '',
        },
    );
    $LayoutObject->Block( Name => 'ActionList' );
    $LayoutObject->Block(
        Name => 'ActionSearch',
        Data => \%Param,
    );

    my $CustomerCompanyObject = $Kernel::OM->Get('Kernel::System::CustomerCompany');

    # get writable data sources
    my %CustomerCompanySource = $CustomerCompanyObject->CustomerCompanySourceList(
        ReadOnly => 0,
    );

    # only show Add option if we have at least one writable backend
    if ( scalar keys %CustomerCompanySource ) {
        $Param{SourceOption} = $LayoutObject->BuildSelection(
            Data       => { %CustomerCompanySource, },
            Name       => 'Source',
            SelectedID => $Param{Source} || '',
            Class      => 'Modernize',
        );

        $LayoutObject->Block(
            Name => 'ActionAdd',
            Data => \%Param,
        );
    }

    # if there are any registries to search, the table is filled and shown
    if ( $Param{Search} ) {

        # get config object
        my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

        # same Limit as $Self->{CustomerCompany}->{CustomerCompanySearchListLimit}
        # smallest Limit from all sources
        my $Limit;
        SOURCE:
        for my $Count ( '', 1 .. 10 ) {
            next SOURCE if !$ConfigObject->Get("CustomerCompany$Count");
            my $CustomerUserMap = $ConfigObject->Get("CustomerCompany$Count");
            next SOURCE if !$CustomerUserMap->{CustomerCompanySearchListLimit};
            if ( !defined $Limit || $CustomerUserMap->{CustomerCompanySearchListLimit} < $Limit ) {
                $Limit = $CustomerUserMap->{CustomerCompanySearchListLimit};
            }
        }

        # as fallback take the hardcoded limit of Kernel/System/CustomerCompany/DB.pm
        $Limit //= 50000;

        my %ListAllItems = $CustomerCompanyObject->CustomerCompanyList(
            Search => $Param{Search},
            Limit  => $Limit + 1,
            Valid  => 0,
        );

        if ( keys %ListAllItems <= $Limit ) {
            my $ListAllItems = keys %ListAllItems;
            $LayoutObject->Block(
                Name => 'OverviewHeader',
                Data => {
                    ListAll => $ListAllItems,
                    Limit   => $Limit,
                },
            );
        }

        my %List = $CustomerCompanyObject->CustomerCompanyList(
            Search => $Param{Search},
            Valid  => $Self->{IncludeInvalid} ? 0 : 1,
        );

        if ( keys %ListAllItems > $Limit ) {
            my $ListAllItems   = keys %ListAllItems;
            my $SearchListSize = keys %List;

            $LayoutObject->Block(
                Name => 'OverviewHeader',
                Data => {
                    SearchListSize => $SearchListSize,
                    ListAll        => $ListAllItems,
                    Limit          => $Limit,
                },
            );
        }

        $LayoutObject->Block(
            Name => 'OverviewResult',
            Data => \%Param,
        );

        # get valid list
        my %ValidList = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();

        if ( !$ConfigObject->Get( $Param{Source} )->{Params}->{ForeignDB} ) {
            $LayoutObject->Block( Name => 'LocalDB' );
        }

        # if there are results to show
        if (%List) {
            for my $ListKey ( sort { $List{$a} cmp $List{$b} } keys %List ) {

                my %Data = $CustomerCompanyObject->CustomerCompanyGet( CustomerID => $ListKey );
                $LayoutObject->Block(
                    Name => 'OverviewResultRow',
                    Data => {
                        %Data,
                        Search => $Param{Search},
                        Nav    => $Param{Nav},
                    },
                );

                if ( !$ConfigObject->Get( $Param{Source} )->{Params}->{ForeignDB} ) {
                    $LayoutObject->Block(
                        Name => 'LocalDBRow',
                        Data => {
                            Valid => $ValidList{ $Data{ValidID} },
                            %Data,
                        },
                    );
                }

            }
        }

        # otherwise it displays a no data found message
        else {
            $LayoutObject->Block(
                Name => 'NoDataFoundMsg',
                Data => {},
            );
        }
    }

    # if there is nothing to search it shows a message
    else
    {
        $LayoutObject->Block(
            Name => 'NoSearchTerms',
            Data => {},
        );
    }
    return 1;
}

1;
</File>
        <File Location="Custom/Kernel/Modules/AdminCustomerUser.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AdminCustomerUser.pm
# --
# 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::Modules::AdminCustomerUser;

use v5.24;
use strict;
use warnings;
use namespace::autoclean;

# core modules

# CPAN modules

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

our $ObjectManagerDisabled = 1;

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

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

    # set pref for columns key
    $Self->{PrefKeyIncludeInvalid} = 'IncludeInvalid' . '-' . $Self->{Action};

    my %Preferences = $Kernel::OM->Get('Kernel::System::User')->GetPreferences(
        UserID => $Self->{UserID},
    );

    $Self->{IncludeInvalid} = $Preferences{ $Self->{PrefKeyIncludeInvalid} };

    my $DynamicFieldConfigs = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
    );

    $Self->{DynamicFieldLookup} = { map { $_->{Name} => $_ } @{$DynamicFieldConfigs} };

    return $Self;
}

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

    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $Nav            = $ParamObject->GetParam( Param => 'Nav' )    || '';
    my $Source         = $ParamObject->GetParam( Param => 'Source' ) || 'CustomerUser';
    my $Search         = $ParamObject->GetParam( Param => 'Search' );
    my $IncludeInvalid = $ParamObject->GetParam( Param => 'IncludeInvalid' );

    if ( defined $IncludeInvalid ) {
        $Kernel::OM->Get('Kernel::System::User')->SetPreferences(
            UserID => $Self->{UserID},
            Key    => $Self->{PrefKeyIncludeInvalid},
            Value  => $IncludeInvalid,
        );

        $Self->{IncludeInvalid} = $IncludeInvalid;
    }
    $Search
        ||= $ConfigObject->Get('AdminCustomerUser::RunInitialWildcardSearch') ? '*' : '';

    # create local object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    my $NavBar       = '';
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    if ( $Nav eq 'None' ) {
        $NavBar = $LayoutObject->Header( Type => 'Small' );
    }
    else {
        $NavBar = $LayoutObject->Header();
        $NavBar .= $LayoutObject->NavigationBar(
            Type => $Nav eq 'Agent' ? 'Customers' : 'Admin',
        );
    }

    # Get list of valid IDs.
    my @ValidIDList = $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet();

    # check the permission for the SwitchToCustomer feature
    if ( $ConfigObject->Get('SwitchToCustomer') ) {

        my $GroupObject = $Kernel::OM->Get('Kernel::System::Group');

        # get the group id which is allowed to use the switch to customer feature
        my $SwitchToCustomerGroupID = $GroupObject->GroupLookup(
            Group => $ConfigObject->Get('SwitchToCustomer::PermissionGroup'),
        );

        # get user groups, where the user has the rw privilege
        my %Groups = $GroupObject->PermissionUserGet(
            UserID => $Self->{UserID},
            Type   => 'rw',
        );

        # if the user is a member in this group he can access the feature
        if ( $Groups{$SwitchToCustomerGroupID} ) {
            $Self->{SwitchToCustomerPermission} = 1;
        }
    }

    my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');
    my $MainObject         = $Kernel::OM->Get('Kernel::System::Main');

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to set multitenancy.
    if ( $ConfigObject->Get('Multitenancy') ) {
        my $GroupObject = $Kernel::OM->Get('Kernel::System::Group');

        my $MultitenancyGroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
            Group => $ConfigObject->Get('Multitenancy::PermissionGroup'),
        );

        my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
            UserID => $Self->{UserID},
            Type   => 'rw',
        );

        if ( $Groups{$MultitenancyGroupID} ) {
            $Self->{MultitenancyPermission} = 1;
        }
    }
# EO CustomerMultitenancy

    # ------------------------------------------------------------ #
    #  switch to customer
    # ------------------------------------------------------------ #
    if (
        $Self->{Subaction} eq 'Switch'
        && $ConfigObject->Get('SwitchToCustomer')
        && $Self->{SwitchToCustomerPermission}
        )
    {

        # challenge token check for write action
        $LayoutObject->ChallengeTokenCheck();

        # get user data
        my $UserID   = $ParamObject->GetParam( Param => 'ID' ) || '';
        my %UserData = $CustomerUserObject->CustomerUserDataGet(
            User  => $UserID,
            Valid => 1,
        );

        # create new session id
        my $NewSessionID = $Kernel::OM->Get('Kernel::System::AuthSession')->CreateSessionID(
            %UserData,
            UserLastRequest => $Kernel::OM->Create('Kernel::System::DateTime')->ToEpoch(),
            UserType        => 'Customer',
            SessionSource   => 'CustomerInterface',
        );

        # get customer interface session name
        my $SessionName = $ConfigObject->Get('CustomerPanelSessionName') || 'CSID';

        # create a new LayoutObject with SessionIDCookie
        my $Expires = '+' . $ConfigObject->Get('SessionMaxTime') . 's';
        if ( !$ConfigObject->Get('SessionUseCookieAfterBrowserClose') ) {
            $Expires = '';
        }

        my $LayoutObject = Kernel::Output::HTML::Layout->new(
            %{$Self},
            SessionID   => $NewSessionID,
            SessionName => $ConfigObject->Get('SessionName'),
        );
        $LayoutObject->SetCookie(
            Key     => 'SessionIDCookie',
            Name    => $SessionName,
            Value   => $NewSessionID,
            Expires => $Expires,
        );

        # log event
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'notice',
            Message  =>
                "Switched from Agent to Customer ($Self->{UserLogin} -=> $UserData{UserLogin})",
        );

        # build URL to customer interface
        my $URL = $ConfigObject->Get('HttpType')
            . '://'
            . $ConfigObject->Get('FQDN')
            . '/'
            . $ConfigObject->Get('ScriptAlias')
            . 'customer.pl';

        # if no sessions are used we attach the session as URL parameter
        if ( !$ConfigObject->Get('SessionUseCookie') ) {
            $URL .= "?$SessionName=$NewSessionID";
        }

        # redirect to customer interface with new session id
        return $LayoutObject->Redirect( ExtURL => $URL );
    }

    # search user list
    if ( $Self->{Subaction} eq 'Search' ) {
        $Self->_Overview(
            Nav    => $Nav,
            Search => $Search,
        );
        my $Output = $NavBar;
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerUser',
            Data         => \%Param,
        );

        if ( $Nav eq 'None' ) {
            $Output .= $LayoutObject->Footer( Type => 'Small' );
        }
        else {
            $Output .= $LayoutObject->Footer();
        }

        return $Output;
    }

    # ------------------------------------------------------------ #
    # download file preferences
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Download' ) {
        my $Group = $ParamObject->GetParam( Param => 'Group' ) || '';
        my $User  = $ParamObject->GetParam( Param => 'ID' )    || '';

        # get user data
        my %UserData    = $CustomerUserObject->CustomerUserDataGet( User => $User );
        my %Preferences = %{ $ConfigObject->Get('CustomerPreferencesGroups') };
        my $Module      = $Preferences{$Group}->{Module};
        if ( !$MainObject->Require($Module) ) {
            return $LayoutObject->FatalError();
        }
        my $Object = $Module->new(
            %{$Self},
            ConfigItem => $Preferences{$Group},
            UserObject => $CustomerUserObject,
            Debug      => $Self->{Debug},
        );
        my %File = $Object->Download( UserData => \%UserData );

        return $LayoutObject->Attachment(%File);
    }

    # ------------------------------------------------------------ #
    # change
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Change' ) {
        my $User         = $ParamObject->GetParam( Param => 'ID' )           || '';
        my $Notification = $ParamObject->GetParam( Param => 'Notification' ) || '';

        # get user data
        my %UserData = $CustomerUserObject->CustomerUserDataGet( User => $User );

        my $Output = $NavBar;
        $Output .= $LayoutObject->Notify( Info => Translatable('Customer updated!') )
            if ( $Notification && $Notification eq 'Update' );
        $Output .= $Self->_Edit(
            Nav    => $Nav,
            Action => 'Change',
            Source => $Source,
            Search => $Search,
            ID     => $User,
            %UserData,
        );

        if ( $Nav eq 'None' ) {
            $Output .= $LayoutObject->Footer( Type => 'Small' );
        }
        else {
            $Output .= $LayoutObject->Footer();
        }

        return $Output;
    }

    # ------------------------------------------------------------ #
    # change action
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'ChangeAction' ) {

        # challenge token check for write action
        $LayoutObject->ChallengeTokenCheck();

        # update only the preferences and dynamic fields, if the source is readonly or a ldap backend
        my $UpdateOnlyPreferences;

        if ( $ConfigObject->Get($Source)->{ReadOnly} || $ConfigObject->Get($Source)->{Module} =~ /LDAP/i ) {
            $UpdateOnlyPreferences = 1;
        }

        my $Note = '';
        my ( %GetParam, %Errors );

        # Get dynamic field backend object.
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        ENTRY:
        for my $Entry ( @{ $ConfigObject->Get($Source)->{Map} } ) {

            # check dynamic fields
            if ( $Entry->[5] eq 'dynamic_field' ) {

                my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "DynamicField $Entry->[2] not found!",
                    );
                    next ENTRY;
                }

                my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    ParamObject        => $ParamObject,
                    Mandatory          => $Entry->[4],
                );

                if ( $ValidationResult->{ServerError} ) {
                    $Errors{ $Entry->[0] } = $ValidationResult;
                }
                else {

                    # generate storable value of dynamic field edit field
                    $GetParam{ $Entry->[0] } = $DynamicFieldBackendObject->EditFieldValueGet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ParamObject        => $ParamObject,
                        LayoutObject       => $LayoutObject,
                    );
                }
            }

            # check remaining non-dynamic-field mandatory fields
            else {
                $GetParam{ $Entry->[0] } = $ParamObject->GetParam( Param => $Entry->[0] ) || '';

                next ENTRY if $UpdateOnlyPreferences;

                if ( !$GetParam{ $Entry->[0] } && $Entry->[4] ) {
                    $Errors{ $Entry->[0] . 'Invalid' } = 'ServerError';
                }
            }
        }
        $GetParam{ID} = $ParamObject->GetParam( Param => 'ID' ) || '';

        # check email address
        if (
            !$UpdateOnlyPreferences
            && $GetParam{UserEmail}
            && !$CheckItemObject->CheckEmail( Address => $GetParam{UserEmail} )
            && grep { $_ eq $GetParam{ValidID} } @ValidIDList
            )
        {
            $Errors{UserEmailInvalid} = 'ServerError';
            $Errors{ErrorType}        = $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
        }

        # Get the current user data for some checks.
        my %CurrentUserData = $CustomerUserObject->CustomerUserDataGet(
            User => $GetParam{ID},
        );

        # Check CustomerID, if CustomerCompanySupport is enabled and the UserCustomerID was changed.
        if (
            $ConfigObject->Get($Source)->{CustomerCompanySupport}
            && $GetParam{UserCustomerID}
            && $CurrentUserData{UserCustomerID} ne $GetParam{UserCustomerID}
            )
        {

            my %Company = $Kernel::OM->Get('Kernel::System::CustomerCompany')->CustomerCompanyGet(
                CustomerID => $GetParam{UserCustomerID},
            );

            if ( !%Company ) {
                $Errors{UserCustomerIDInvalid} = 'ServerError';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            my $UpdateSuccess;
            if ( !$UpdateOnlyPreferences ) {
                $UpdateSuccess = $CustomerUserObject->CustomerUserUpdate(
                    %GetParam,
                    UserID => $Self->{UserID},
                );
            }

            if ( $GetParam{UserPassword} && ( $CurrentUserData{UserPassword} // '' ) ne $GetParam{UserPassword} ) {

                $UpdateSuccess = $CustomerUserObject->DeleteOnePreference(
                    Key    => 'UserLastPwChangeTime',
                    UserID => $GetParam{ID},
                );

            }

            if ( $UpdateSuccess || $UpdateOnlyPreferences ) {

                # set dynamic field values
                ENTRY:
                for my $Entry ( @{ $ConfigObject->Get($Source)->{Map} } ) {
                    next ENTRY if $Entry->[5] ne 'dynamic_field';

                    my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                    if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                        $Note .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Dynamic field %s not found!',
                                $Entry->[2],
                            ),
                        );
                        next ENTRY;
                    }

                    my $ValueSet = $DynamicFieldBackendObject->ValueSet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ObjectName         => $GetParam{UserLogin},
                        Value              => $GetParam{ $Entry->[0] },
                        UserID             => $Self->{UserID},
                    );

                    if ( !$ValueSet ) {
                        $Note .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Unable to set value for dynamic field %s!',
                                $Entry->[2],
                            ),
                        );
                        next ENTRY;
                    }
                }

                # update preferences
                my %Preferences = %{ $ConfigObject->Get('CustomerPreferencesGroups') };
                GROUP:
                for my $Group ( sort keys %Preferences ) {
                    next GROUP if $Group eq 'Password';

                    # get user data
                    my %UserData = $CustomerUserObject->CustomerUserDataGet(
                        User => $GetParam{UserLogin}
                    );
                    my $Module = $Preferences{$Group}->{Module};
                    if ( !$MainObject->Require($Module) ) {
                        return $LayoutObject->FatalError();
                    }
                    my $Object = $Module->new(
                        %{$Self},
                        ConfigItem => $Preferences{$Group},
                        UserObject => $CustomerUserObject,
                        Debug      => $Self->{Debug},
                    );
                    my @Params = $Object->Param(
                        UserData => \%UserData,
                        Customer => 1,
                    );
                    if (@Params) {
                        my %GetParam;
                        for my $ParamItem (@Params) {
                            my @Array = $ParamObject->GetArray( Param => $ParamItem->{Name} );
                            $GetParam{ $ParamItem->{Name} } = \@Array;
                        }
                        if (
                            !$Object->Run(
                                GetParam => \%GetParam,
                                UserData => \%UserData
                            )
                            )
                        {
                            $Note .= $LayoutObject->Notify( Info => $Object->Error() );
                        }
                    }
                }

                # clear customer user cache
                $CustomerUserObject->CustomerUserCacheClear(
                    UserLogin => $GetParam{UserLogin},
                );

                # get user data and show screen again
                if ( !$Note ) {

                    # if the user would like to continue editing the priority, just redirect to the edit screen
                    if (
                        defined $ParamObject->GetParam( Param => 'ContinueAfterSave' )
                        && ( $ParamObject->GetParam( Param => 'ContinueAfterSave' ) eq '1' )
                        )
                    {
                        my $ID = $ParamObject->GetParam( Param => 'ID' ) || '';
                        return $LayoutObject->Redirect(
                            OP =>
                                "Action=$Self->{Action};Subaction=Change;ID=$ID;Search=$Search;Nav=$Nav;Notification=Update"
                        );
                    }
                    else {

                        # otherwise return to overview
                        return $LayoutObject->Redirect( OP => "Action=$Self->{Action};Notification=Update" );
                    }
                }
            }
            else {
                $Note .= $LayoutObject->Notify( Priority => 'Error' );
            }
        }

        # something has gone wrong
        my $Output = $NavBar;
        $Output .= $Note;
        $Output .= $Self->_Edit(
            Nav    => $Nav,
            Action => 'Change',
            Source => $Source,
            Search => $Search,
            Errors => \%Errors,
            %GetParam,
        );

        if ( $Nav eq 'None' ) {
            $Output .= $LayoutObject->Footer( Type => 'Small' );
        }
        else {
            $Output .= $LayoutObject->Footer();
        }

        return $Output;
    }

    # ------------------------------------------------------------ #
    # add
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Add' ) {
        my %GetParam;
        $GetParam{UserLogin}  = $ParamObject->GetParam( Param => 'UserLogin' )  || '';
        $GetParam{CustomerID} = $ParamObject->GetParam( Param => 'CustomerID' ) || '';
        my $Output = $NavBar;
        $Output .= $Self->_Edit(
            Nav    => $Nav,
            Action => 'Add',
            Source => $Source,
            Search => $Search,
            %GetParam,
        );

        if ( $Nav eq 'None' ) {
            $Output .= $LayoutObject->Footer( Type => 'Small' );
        }
        else {
            $Output .= $LayoutObject->Footer();
        }

        return $Output;
    }

    # ------------------------------------------------------------ #
    # add action
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'AddAction' ) {

        # challenge token check for write action
        $LayoutObject->ChallengeTokenCheck();

        my $Note = '';
        my ( %GetParam, %Errors );

        my $AutoLoginCreation = $ConfigObject->Get($Source)->{AutoLoginCreation};

        # Get dynamic field backend object.
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        ENTRY:
        for my $Entry ( @{ $ConfigObject->Get($Source)->{Map} } ) {

            # don't validate UserLogin if AutoLoginCreation is configured
            next ENTRY if ( $AutoLoginCreation && $Entry->[0] eq 'UserLogin' );

            # check dynamic fields
            if ( $Entry->[5] eq 'dynamic_field' ) {

                my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "DynamicField $Entry->[2] not found!",
                    );
                    next ENTRY;
                }

                my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    ParamObject        => $ParamObject,
                    Mandatory          => $Entry->[4],
                );

                if ( $ValidationResult->{ServerError} ) {
                    $Errors{ $Entry->[0] } = $ValidationResult;
                }
                else {

                    # generate storable value of dynamic field edit field
                    $GetParam{ $Entry->[0] } = $DynamicFieldBackendObject->EditFieldValueGet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ParamObject        => $ParamObject,
                        LayoutObject       => $LayoutObject,
                    );
                }
            }

            # check remaining non-dynamic-field mandatory fields
            else {
                $GetParam{ $Entry->[0] } = $ParamObject->GetParam( Param => $Entry->[0] ) || '';
                if ( !$GetParam{ $Entry->[0] } && $Entry->[4] ) {
                    $Errors{ $Entry->[0] . 'Invalid' } = 'ServerError';
                }
            }
        }

        # check email address
        if (
            $GetParam{UserEmail}
            && !$CheckItemObject->CheckEmail( Address => $GetParam{UserEmail} )
            && grep { $_ eq $GetParam{ValidID} } @ValidIDList
            )
        {
            $Errors{UserEmailInvalid} = 'ServerError';
            $Errors{ErrorType}        = $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
        }

        # Check CustomerID, if CustomerCompanySupport is enabled.
        if ( $ConfigObject->Get($Source)->{CustomerCompanySupport} && $GetParam{UserCustomerID} ) {

            my %Company = $Kernel::OM->Get('Kernel::System::CustomerCompany')->CustomerCompanyGet(
                CustomerID => $GetParam{UserCustomerID},
            );

            if ( !%Company ) {
                $Errors{UserCustomerIDInvalid} = 'ServerError';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            # add user
            my $User = $CustomerUserObject->CustomerUserAdd(
                %GetParam,
                UserID => $Self->{UserID},
                Source => $Source
            );
            if ($User) {

                ENTRY:
                for my $Entry ( @{ $ConfigObject->Get($Source)->{Map} } ) {
                    next ENTRY unless $Entry->[5] eq 'dynamic_field';

                    my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

                    if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                        $Note .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Dynamic field %s not found!',
                                $Entry->[2],
                            ),
                        );

                        next ENTRY;
                    }

                    my $ValueSet = $DynamicFieldBackendObject->ValueSet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        ObjectName         => $User,
                        Value              => $GetParam{ $Entry->[0] },
                        UserID             => $Self->{UserID},
                    );

                    if ( !$ValueSet ) {
                        $Note .= $LayoutObject->Notify(
                            Info => $LayoutObject->{LanguageObject}->Translate(
                                'Unable to set value for dynamic field %s!',
                                $Entry->[2],
                            ),
                        );

                        next ENTRY;
                    }
                }

                # update preferences
                my %Preferences = %{ $ConfigObject->Get('CustomerPreferencesGroups') };
                GROUP:
                for my $Group ( sort keys %Preferences ) {
                    next GROUP if $Group eq 'Password';

                    # get user data
                    my %UserData = $CustomerUserObject->CustomerUserDataGet(
                        User => $User,
                    );
                    my $Module = $Preferences{$Group}->{Module};
                    if ( !$MainObject->Require($Module) ) {
                        return $LayoutObject->FatalError();
                    }
                    my $Object = $Module->new(
                        %{$Self},
                        ConfigItem => $Preferences{$Group},
                        UserObject => $CustomerUserObject,
                        Debug      => $Self->{Debug},
                    );
                    my @Params = $Object->Param(
                        %{ $Preferences{$Group} },
                        UserData => \%UserData,
                        Customer => 1,
                    );
                    if (@Params) {
                        my %GetParam;
                        for my $ParamItem (@Params) {
                            my @Array = $ParamObject->GetArray( Param => $ParamItem->{Name} );
                            $GetParam{ $ParamItem->{Name} } = \@Array;
                        }
                        if (
                            !$Object->Run(
                                GetParam => \%GetParam,
                                UserData => \%UserData
                            )
                            )
                        {
                            $Note .= $LayoutObject->Notify( Info => $Object->Error() );
                        }
                    }
                }

                # get user data and show screen again
                if ( !$Note ) {

                    # in borrowed view, take the new created customer over into the new ticket
                    if ( $Nav eq 'None' ) {
                        my $Output = $NavBar;

                        $LayoutObject->AddJSData(
                            Key   => 'Customer',
                            Value => $User,
                        );
                        $LayoutObject->AddJSData(
                            Key   => 'Nav',
                            Value => $Nav,
                        );

                        $Output .= $LayoutObject->Output(
                            TemplateFile => 'AdminCustomerUser',
                            Data         => \%Param,
                        );

                        $Output .= $LayoutObject->Footer( Type => 'Small' );

                        return $Output;
                    }

                    $Self->_Overview(
                        Nav    => $Nav,
                        Search => $Search,
                    );

                    my $Output        = $NavBar . $Note;
                    my $URL           = '';
                    my $UserHTMLQuote = $LayoutObject->LinkEncode($User);
                    my $UserQuote     = $LayoutObject->Ascii2Html( Text => $User );
                    if ( $ConfigObject->Get('Frontend::Module')->{AgentTicketPhone} ) {
                        $URL
                            .= "<a href=\"$LayoutObject->{Baselink}Action=AgentTicketPhone;Subaction=StoreNew;ExpandCustomerName=2;CustomerUser=$UserHTMLQuote;$LayoutObject->{ChallengeTokenParam}\">"
                            . $LayoutObject->{LanguageObject}->Translate('New phone ticket')
                            . "</a>";
                    }
                    if ( $ConfigObject->Get('Frontend::Module')->{AgentTicketEmail} ) {
                        if ($URL) {
                            $URL .= " - ";
                        }
                        $URL
                            .= "<a href=\"$LayoutObject->{Baselink}Action=AgentTicketEmail;Subaction=StoreNew;ExpandCustomerName=2;CustomerUser=$UserHTMLQuote;$LayoutObject->{ChallengeTokenParam}\">"
                            . $LayoutObject->{LanguageObject}->Translate('New email ticket')
                            . "</a>";
                    }
                    if ($URL) {
                        $Output
                            .= $LayoutObject->Notify(
                                Data => $LayoutObject->{LanguageObject}->Translate(
                                    'Customer %s added',
                                    $UserQuote,
                                )
                                . " ( $URL )!",
                            );
                    }
                    else {
                        $Output
                            .= $LayoutObject->Notify(
                                Data => $LayoutObject->{LanguageObject}->Translate(
                                    'Customer %s added',
                                    $UserQuote,
                                )
                                . "!",
                            );
                    }
                    $Output .= $LayoutObject->Output(
                        TemplateFile => 'AdminCustomerUser',
                        Data         => \%Param,
                    );

                    if ( $Nav eq 'None' ) {
                        $Output .= $LayoutObject->Footer( Type => 'Small' );
                    }
                    else {
                        $Output .= $LayoutObject->Footer();
                    }

                    return $Output;
                }
            }
            else {
                $Note .= $LayoutObject->Notify( Priority => 'Error' );
            }
        }

        # something has gone wrong
        my $Output = $NavBar . $Note;
        $Output .= $Self->_Edit(
            Nav    => $Nav,
            Action => 'Add',
            Source => $Source,
            Search => $Search,
            Errors => \%Errors,
            %GetParam,
        );

        if ( $Nav eq 'None' ) {
            $Output .= $LayoutObject->Footer( Type => 'Small' );
        }
        else {
            $Output .= $LayoutObject->Footer();
        }

        return $Output;
    }

    # ------------------------------------------------------------ #
    # overview
    # ------------------------------------------------------------ #
    else {
        $Self->_Overview(
            Nav    => $Nav,
            Search => $Search,
        );

        my $Notification = $ParamObject->GetParam( Param => 'Notification' ) || '';
        my $Output       = $NavBar;
        $Output .= $LayoutObject->Notify( Info => Translatable('Customer user updated!') )
            if ( $Notification && $Notification eq 'Update' );

        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminCustomerUser',
            Data         => \%Param,
        );

        if ( $Nav eq 'None' ) {
            $Output .= $LayoutObject->Footer( Type => 'Small' );
        }
        else {
            $Output .= $LayoutObject->Footer();
        }

        return $Output;
    }
}

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

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    $LayoutObject->Block(
        Name => 'Overview',
        Data => \%Param,
    );

    $LayoutObject->Block(
        Name => 'IncludeInvalid',
        Data => {
            IncludeInvalid        => $Self->{IncludeInvalid},
            IncludeInvalidChecked => $Self->{IncludeInvalid} ? 'checked' : '',
        },
    );
    $LayoutObject->Block( Name => 'ActionList' );
    $LayoutObject->Block(
        Name => 'ActionSearch',
        Data => \%Param,
    );

    my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');

    # get writable data sources
    my %CustomerSource = $CustomerUserObject->CustomerSourceList(
        ReadOnly => 0,
    );

    # only show Add option if we have at least one writable backend
    if ( scalar keys %CustomerSource ) {
        $Param{SourceOption} = $LayoutObject->BuildSelection(
            Data       => { %CustomerSource, },
            Name       => 'Source',
            SelectedID => $Param{Source} || '',
            Class      => 'Modernize',
        );

        $LayoutObject->Block(
            Name => 'ActionAdd',
            Data => \%Param,
        );
    }

    if ( $Param{Search} ) {

        # get config object
        my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

        # when there is no data to show, a message is displayed on the table with this colspan
        my $ColSpan = 6;

        # same Limit as $Self->{CustomerUserMap}->{CustomerUserSearchListLimit}
        # smallest Limit from all sources
        my $Limit;
        SOURCE:
        for my $Count ( '', 1 .. 10 ) {
            next SOURCE if !$ConfigObject->Get("CustomerUser$Count");
            my $CustomerUserMap = $ConfigObject->Get("CustomerUser$Count");
            next SOURCE if !$CustomerUserMap->{CustomerUserSearchListLimit};
            if ( !defined $Limit || $CustomerUserMap->{CustomerUserSearchListLimit} < $Limit ) {
                $Limit = $CustomerUserMap->{CustomerUserSearchListLimit};
            }
        }

        # as fallback take the hardcoded limit of Kernel/System/CustomerUser/DB.pm
        $Limit //= 250;

        my %ListAllItems = $CustomerUserObject->CustomerSearch(
            Search => $Param{Search},
            Limit  => $Limit + 1,
            Valid  => 0,
        );

        if ( keys %ListAllItems <= $Limit ) {
            my $ListAllItems = keys %ListAllItems;
            $LayoutObject->Block(
                Name => 'OverviewHeader',
                Data => {
                    ListAll => $ListAllItems,
                    Limit   => $Limit,
                },
            );
        }

        my %List = $CustomerUserObject->CustomerSearch(
            Search => $Param{Search},
            Valid  => $Self->{IncludeInvalid} ? 0 : 1,
        );

        if ( keys %ListAllItems > $Limit ) {
            my $ListAllItems   = keys %ListAllItems;
            my $SearchListSize = keys %List;

            $LayoutObject->Block(
                Name => 'OverviewHeader',
                Data => {
                    SearchListSize => $SearchListSize,
                    ListAll        => $ListAllItems,
                    Limit          => $Limit,
                },
            );
        }

        $LayoutObject->Block(
            Name => 'OverviewResult',
            Data => \%Param,
        );

        if ( $ConfigObject->Get('SwitchToCustomer') && $Self->{SwitchToCustomerPermission} && $Param{Nav} ne 'None' )
        {
            $ColSpan = 7;
            $LayoutObject->Block(
                Name => 'OverviewResultSwitchToCustomer',
            );
        }

        # if there are results to show
        if (%List) {

            # get valid list
            my %ValidList = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();
            for my $ListKey ( sort { lc($a) cmp lc($b) } keys %List ) {

                my %UserData = $CustomerUserObject->CustomerUserDataGet( User => $ListKey );
                $UserData{UserFullname} = $CustomerUserObject->CustomerName(
                    UserLogin => $UserData{UserLogin},
                );

                $LayoutObject->Block(
                    Name => 'OverviewResultRow',
                    Data => {
                        Valid       => $ValidList{ $UserData{ValidID} || '' } || '-',
                        Search      => $Param{Search},
                        CustomerKey => $ListKey,
                        %UserData,
                    },
                );
                if ( $Param{Nav} eq 'None' ) {
                    $LayoutObject->Block(
                        Name => 'OverviewResultRowLinkNone',
                        Data => {
                            Search      => $Param{Search},
                            CustomerKey => $ListKey,
                            %UserData,
                        },
                    );
                }
                else {
                    $LayoutObject->Block(
                        Name => 'OverviewResultRowLink',
                        Data => {
                            Search      => $Param{Search},
                            Nav         => $Param{Nav},
                            CustomerKey => $ListKey,
                            %UserData,
                        },
                    );
                }

                if (
                    $ConfigObject->Get('SwitchToCustomer')
                    && $Self->{SwitchToCustomerPermission}
                    && $Param{Nav} ne 'None'
                    )
                {
                    $LayoutObject->Block(
                        Name => 'OverviewResultRowSwitchToCustomer',
                        Data => {
                            Search => $Param{Search},
                            %UserData,
                        },
                    );
                }
            }
        }

        # otherwise it displays a no data found message
        else {
            $LayoutObject->Block(
                Name => 'NoDataFoundMsg',
                Data => {
                    ColSpan => $ColSpan,
                },
            );
        }
    }

    # if there is nothing to search it shows a message
    else
    {
        $LayoutObject->Block(
            Name => 'NoSearchTerms',
            Data => {},
        );
    }

    $LayoutObject->AddJSData(
        Key   => 'Nav',
        Value => $Param{Nav},
    );

    return;
}

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

    # Get layout object.
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    $LayoutObject->Block(
        Name => 'Overview',
        Data => \%Param,
    );

    $LayoutObject->Block( Name => 'ActionList' );
    $LayoutObject->Block(
        Name => 'ActionOverview',
        Data => \%Param,
    );

    $LayoutObject->Block(
        Name => 'OverviewUpdate',
        Data => \%Param,
    );

    if ( $Param{Action} eq 'Change' ) {

        # shows edit header
        $LayoutObject->Block( Name => 'HeaderEdit' );

        # shows effective permissions matrix
        $Self->_EffectivePermissions(%Param);
    }

    # shows add header
    else {
        $LayoutObject->Block( Name => 'HeaderAdd' );
    }

    my $UpdateOnlyPreferences;

    # Get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # update user
    if (
        $ConfigObject->Get( $Param{Source} )->{ReadOnly}
        ||
        $ConfigObject->Get( $Param{Source} )->{Module} =~ /LDAP/i
        )
    {
        $UpdateOnlyPreferences = 1;
    }

    # Get dynamic field backend object.
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my $ParamObject               = $Kernel::OM->Get('Kernel::System::Web::Request');

    ENTRY:
    for my $Entry ( @{ $ConfigObject->Get( $Param{Source} )->{Map} } ) {
        next ENTRY if !$Entry->[0];

        # Handle dynamic fields
        if ( $Entry->[5] eq 'dynamic_field' ) {

            my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{ $Entry->[2] };

            next ENTRY if !IsHashRefWithData($DynamicFieldConfig);

            # Get HTML for dynamic field
            my $DynamicFieldHTML = $DynamicFieldBackendObject->EditFieldRender(
                DynamicFieldConfig => $DynamicFieldConfig,
                Value              => $Param{ $Entry->[0] } ? $Param{ $Entry->[0] } : undef,
                Mandatory          => $Entry->[4],
                LayoutObject       => $LayoutObject,
                ParamObject        => $ParamObject,

                # Server error, if any
                %{ $Param{Errors}->{ $Entry->[0] } },
            );

            # skip fields for which HTML could not be retrieved
            next ENTRY if !IsHashRefWithData($DynamicFieldHTML);

            $LayoutObject->Block(
                Name => 'Item',
                Data => {},
            );

            $LayoutObject->Block(
                Name => 'DynamicField',
                Data => {
                    Name  => $DynamicFieldConfig->{Name},
                    Label => $DynamicFieldHTML->{Label},
                    Field => $DynamicFieldHTML->{Field},
                },
            );

            next ENTRY;
        }

        my $Block = 'Input';

        # check input type
        if ( $Entry->[0] =~ /^UserPasswor/i ) {
            $Block = 'Password';
        }

        # check if login auto creation
        if ( $ConfigObject->Get( $Param{Source} )->{AutoLoginCreation} && $Entry->[0] eq 'UserLogin' ) {
            $Block = 'InputHidden';
        }

        if ( $Entry->[7] || $UpdateOnlyPreferences ) {
            $Param{ReadOnly} = 1;
        }
        else {
            $Param{ReadOnly} = 0;
        }

        # show required flag
        if ( $Entry->[4] ) {
            $Param{RequiredClass}          = 'Validate_Required';
            $Param{RequiredLabelClass}     = 'Mandatory';
            $Param{RequiredLabelCharacter} = '*';
        }
        else {
            $Param{RequiredClass}          = '';
            $Param{RequiredLabelClass}     = '';
            $Param{RequiredLabelCharacter} = '';
        }

        # set empty string
        $Param{Errors}->{ $Entry->[0] . 'Invalid' } ||= '';

        # add class to validate emails
        if ( $Entry->[0] eq 'UserEmail' ) {
            $Param{RequiredClass} .= ' Validate_Email';
        }

        # Build selections or input fields.
        # An explicit selection has the highest priority.
        if ( $ConfigObject->Get( $Param{Source} )->{Selections}->{ $Entry->[0] } ) {
            $Block = 'Option';

            # Change the validation class
            if ( $Param{RequiredClass} ) {
                $Param{RequiredClass} = 'Validate_Required';
            }

            # get the data of the current selection
            my $SelectionsData = $ConfigObject->Get( $Param{Source} )->{Selections}->{ $Entry->[0] };

            # make sure the encoding stamp is set
            for my $Key ( sort keys %{$SelectionsData} ) {
                $SelectionsData->{$Key} = $Kernel::OM->Get('Kernel::System::Encode')->EncodeInput( $SelectionsData->{$Key} );
            }

            # build option string
            $Param{Option} = $LayoutObject->BuildSelection(
                Data        => $SelectionsData,
                Name        => $Entry->[0],
                Translation => 1,
                SelectedID  => $Param{ $Entry->[0] },
                Class       => "$Param{RequiredClass} Modernize " . $Param{Errors}->{ $Entry->[0] . 'Invalid' },
                Disabled    => $UpdateOnlyPreferences ? 1 : 0,
            );
        }
        elsif (
            $Entry->[0] =~ m/^UserCountry/i
            &&
            $ConfigObject->Get('ReferenceData::TranslatedCountryNames')
            )
        {
            $Block = 'Option';

            my $CountryList = $Kernel::OM->Get('Kernel::System::ReferenceData')->CLDRCountryList(
                Language => $LayoutObject->{UserLanguage},
            );

            # Make sure that the previous value exists in the selection list even if isn't a country code.
            my $PreviousCountry = $Param{ $Entry->[0] };
            if ($PreviousCountry) {
                $CountryList->{$PreviousCountry} //= $PreviousCountry;
            }

            $Param{Option} = $LayoutObject->BuildSelection(
                Data         => $CountryList,
                PossibleNone => 1,
                Sort         => 'AlphanumericValue',
                Name         => $Entry->[0],
                Class        => "$Param{RequiredClass} Modernize " . $Param{Errors}->{ $Entry->[0] . 'Invalid' },
                SelectedID   => ( $Param{ $Entry->[0] } // 1 ),
            );
        }
        elsif ( $Entry->[0] =~ m/^ValidID/i ) {

            # Change the validation class
            if ( $Param{RequiredClass} ) {
                $Param{RequiredClass} = 'Validate_Required';
            }

            # build ValidID string
            $Block = 'Option';
            $Param{Option} = $LayoutObject->BuildSelection(
                Data       => { $Kernel::OM->Get('Kernel::System::Valid')->ValidList(), },
                Name       => $Entry->[0],
                SelectedID => defined( $Param{ $Entry->[0] } ) ? $Param{ $Entry->[0] } : 1,
                Class      => "$Param{RequiredClass} Modernize " . $Param{Errors}->{ $Entry->[0] . 'Invalid' },
                Disabled   => $UpdateOnlyPreferences ? 1 : 0,
            );
        }
        elsif (
            $Entry->[0] =~ m/^UserCustomerID$/i
            && $ConfigObject->Get( $Param{Source} )->{CustomerCompanySupport}
            )
        {
            my $CustomerCompanyObject = $Kernel::OM->Get('Kernel::System::CustomerCompany');
            my %CompanyList           = (
                $CustomerCompanyObject->CustomerCompanyList( Limit => 0 ),
                '' => '-',
            );
            if ( $Param{ $Entry->[0] } ) {
                my %Company = $CustomerCompanyObject->CustomerCompanyGet(
                    CustomerID => $Param{ $Entry->[0] },
                );
                if ( !%Company ) {
                    $CompanyList{ $Param{ $Entry->[0] } } = $Param{ $Entry->[0] } . ' (-)';
                }
            }
            $Block = 'Option';

            # Change the validation class
            if ( $Param{RequiredClass} ) {
                $Param{RequiredClass} = 'Validate_Required';
            }

            my $UseAutoComplete = $Kernel::OM->Get('Kernel::Config')->Get('AdminCustomerUser::UseAutoComplete');

            if ($UseAutoComplete) {

                my $Value = $Param{ $Entry->[0] } || $Param{CustomerID};
                $Value = $LayoutObject->Output(
                    Template => "[% Data.Value | html %]",
                    Data     => {
                        Value => $Value,
                    }
                );

                $Param{Option} = '<input type="text" id="UserCustomerID" name="UserCustomerID" value="' . $Value . '"
                    class="W50pc CustomerAutoCompleteSimple '
                    . $Param{RequiredClass} . ' '
                    . $Param{Errors}->{ $Entry->[0] . 'Invalid' }
                    . '" data-customer-search-type="CustomerID" />';
            }
            else {
                $Param{Option} = $LayoutObject->BuildSelection(
                    Data       => \%CompanyList,
                    Name       => $Entry->[0],
                    Max        => 80,
                    SelectedID => $Param{ $Entry->[0] } || $Param{CustomerID},
                    Class      => "$Param{RequiredClass} Modernize " . $Param{Errors}->{ $Entry->[0] . 'Invalid' },
                    Disabled   => $UpdateOnlyPreferences ? 1 : 0,
                );
            }
        }
        elsif ( $Param{Action} eq 'Add' && $Entry->[0] =~ /^UserCustomerID$/i ) {

            # Use CustomerID param if called from CIC.
            $Param{Value} = $Param{ $Entry->[0] } || $Param{CustomerID} || '';
        }
# Rother OSS / CustomerMultitenancy
        # Build the group field.
        elsif ( $Entry->[0] =~ /^UserGroupID$/i ) {
            # Check if the user has the permission to see/change the multitenancy field.
            if ( !$Self->{MultitenancyPermission} ) {
                next ENTRY;
            } else {
                # Build the field.
                $Block = 'Option';
                $Param{Option} = $LayoutObject->BuildSelection(
                    Data => {
                        $Kernel::OM->Get('Kernel::System::Group')->GroupList(
                            Valid => 1,
                        )
                    },
                    Name         => $Entry->[0],
                    PossibleNone => 1,
                    Class        => 'Modernize ' . ( $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '' ),
                    SelectedID   => $Param{ $Entry->[0] } || '',
                    Disabled     => $UpdateOnlyPreferences ? 1 : 0,
                );
            }
        }
# EO CustomerMultitenancy
        else {
            $Param{Value} = $Param{ $Entry->[0] } || '';
        }

        # add form option
        if ( $Param{Type} && $Param{Type} eq 'hidden' ) {
            $Param{Preferences} .= $Param{Value};
        }
        else {
            $LayoutObject->Block(
                Name => 'PreferencesGeneric',
                Data => {
                    Item => $Entry->[1],
                    %Param
                },
            );
            $LayoutObject->Block(
                Name => "PreferencesGeneric$Block",
                Data => {
                    Item         => $Entry->[1],
                    Name         => $Entry->[0],
                    InvalidField => $Param{Errors}->{ $Entry->[0] . 'Invalid' } || '',
                    %Param,
                },
            );

            # add the correct client side error msg
            if ( $Block eq 'Input' && $Entry->[0] eq 'UserEmail' ) {
                $LayoutObject->Block(
                    Name => 'PreferencesUserEmailErrorMsg',
                    Data => { Name => $Entry->[0] },
                );
            }
            else {
                $LayoutObject->Block(
                    Name => "PreferencesGenericErrorMsg",
                    Data => { Name => $Entry->[0] },
                );
            }

            # add the correct server error msg
            if ( $Block eq 'Input' && $Param{UserEmail} && $Entry->[0] eq 'UserEmail' ) {

                # display server error msg according with the occurred email error type
                $LayoutObject->Block(
                    Name => 'PreferencesUserEmail' . ( $Param{Errors}->{ErrorType} || '' ),
                    Data => { Name => $Entry->[0] },
                );
            }
            else {
                $LayoutObject->Block(
                    Name => "PreferencesGenericServerErrorMsg",
                    Data => { Name => $Entry->[0] },
                );
            }
        }
    }

    my $PreferencesUsed = $ConfigObject->Get( $Param{Source} )->{AdminSetPreferences};
    if ( ( defined $PreferencesUsed && $PreferencesUsed != 0 ) || !defined $PreferencesUsed ) {

        my %Data;
        my %Preferences = %{ $ConfigObject->Get('CustomerPreferencesGroups') };

        GROUP:
        for my $Group ( sort keys %Preferences ) {

            next GROUP if !$Group;

            my $PreferencesGroup = $Preferences{$Group};

            next GROUP if !$PreferencesGroup;
            next GROUP if ref $PreferencesGroup ne 'HASH';

            if ( $Data{ $PreferencesGroup->{Prio} } ) {

                COUNT:
                for ( 1 .. 151 ) {

                    $PreferencesGroup->{Prio}++;

                    if ( !$Data{ $PreferencesGroup->{Prio} } ) {
                        $Data{ $PreferencesGroup->{Prio} } = $Group;
                        last COUNT;
                    }
                }
            }

            $Data{ $PreferencesGroup->{Prio} } = $Group;
        }

        # sort
        for my $Key ( sort keys %Data ) {
            $Data{ sprintf "%07d", $Key } = $Data{$Key};
            delete $Data{$Key};
        }

        # show each preferences setting
        PRIO:
        for my $Prio ( sort keys %Data ) {

            my $Group = $Data{$Prio};
            if ( !$ConfigObject->{CustomerPreferencesGroups}->{$Group} ) {
                next PRIO;
            }

            my %Preference = %{ $ConfigObject->{CustomerPreferencesGroups}->{$Group} };
            if ( $Group eq 'Password' ) {
                next PRIO;
            }

            my $Module = $Preference{Module} || 'Kernel::Output::HTML::CustomerPreferencesGeneric';

            # load module
            if ( $Kernel::OM->Get('Kernel::System::Main')->Require($Module) ) {
                my $Object = $Module->new(
                    %{$Self},
                    ConfigItem => \%Preference,
                    UserObject => $Kernel::OM->Get('Kernel::System::CustomerUser'),
                    Debug      => $Self->{Debug},
                );
                my @Params = $Object->Param(
                    UserData => \%Param,
                    Customer => 1,
                );
                if (@Params) {
                    for my $ParamItem (@Params) {
                        $LayoutObject->Block(
                            Name => 'Item',
                            Data => {%Param},
                        );
                        if (
                            ref $ParamItem->{Data} eq 'HASH'
                            || ref $Preference{Data} eq 'HASH'
                            )
                        {
                            my %BuildSelectionParams = (
                                %Preference,
                                %{$ParamItem},
                            );
                            $BuildSelectionParams{Class} = join( ' ', $BuildSelectionParams{Class} // '', 'Modernize' );

                            $ParamItem->{Option} = $LayoutObject->BuildSelection(
                                %BuildSelectionParams,
                            );
                        }

                        $LayoutObject->Block(
                            Name => $ParamItem->{Block} || $Preference{Block} || 'Option',
                            Data => {
                                Group => $Group,
                                %Param,
                                %Data,
                                %Preference,
                                %{$ParamItem},
                            },
                        );
                    }
                }
            }
            else {
                return $LayoutObject->FatalError();
            }
        }
    }

    $LayoutObject->AddJSData(
        Key   => 'Nav',
        Value => $Param{Nav},
    );

    return $LayoutObject->Output(
        TemplateFile => 'AdminCustomerUser',
        Data         => \%Param,
    );
}

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

    # only if customer group feature is active
    if ( !$Kernel::OM->Get('Kernel::Config')->Get('CustomerGroupSupport') ) {
        return 1;
    }

    # get needed objects
    my $ConfigObject        = $Kernel::OM->Get('Kernel::Config');
    my $LayoutObject        = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $CustomerGroupObject = $Kernel::OM->Get('Kernel::System::CustomerGroup');

    # show tables
    $LayoutObject->Block(
        Name => 'EffectivePermissions',
    );

    my %Groups;
    my %Permissions;

    # go through permission types
    my @Types = @{ $ConfigObject->Get('System::Customer::Permission') };
    for my $Type (@Types) {

        # show header
        $LayoutObject->Block(
            Name => "HeaderGroupPermissionType",
            Data => {
                Type => $Type,
            },
        );

        # get groups of the user
        my %UserGroups = $CustomerGroupObject->GroupMemberList(
            UserID         => $Param{ID},
            Type           => $Type,
            Result         => 'HASH',
            RawPermissions => 0,            # get effective permissions
        );

        # store data in lookup hashes
        for my $GroupID ( sort keys %UserGroups ) {
            $Groups{$GroupID} = $UserGroups{$GroupID};
            $Permissions{$GroupID}{$Type} = 1;
        }
    }

    # show message if no permissions found
    if ( !%Permissions ) {
        $LayoutObject->Block(
            Name => 'NoGroupPermissionsFoundMsg',
        );
    }

    # go through groups, sort by name
    else {
        for my $GroupID ( sort { uc( $Groups{$a} ) cmp uc( $Groups{$b} ) } keys %Groups ) {

            # show table rows
            $LayoutObject->Block(
                Name => 'GroupPermissionTableRow',
                Data => {
                    ID   => $GroupID,
                    Name => $Groups{$GroupID},
                },
            );

            # show permission marks
            for my $Type (@Types) {
                my $PermissionMark = $Permissions{$GroupID}{$Type} ? 'On'        : 'Off';
                my $HighlightMark  = $Type eq 'rw'                 ? 'Highlight' : '';
                $LayoutObject->Block(
                    Name => 'GroupPermissionMark',
                );
                $LayoutObject->Block(
                    Name => 'GroupPermissionMark' . $PermissionMark,
                    Data => {
                        Highlight => $HighlightMark,
                    },
                );
            }
        }
    }

    # get all accessible customers of the user
    my %Customers = $CustomerGroupObject->GroupContextCustomers(
        CustomerUserID => $Param{ID},
    );

    # show message if no customers found
    if ( !%Customers ) {
        $LayoutObject->Block(
            Name => 'NoCustomerAccessFoundMsg',
        );
        return 1;
    }

    # get permission contexts
    my $ContextConfig            = $ConfigObject->Get('CustomerGroupPermissionContext');
    my $DirectAccessContextKey   = '001-CustomerID-same';
    my $IndirectAccessContextKey = '100-CustomerID-other';

    # use default context if none are found
    if ( !IsHashRefWithData($ContextConfig) ) {
        $ContextConfig = {
            $DirectAccessContextKey => {
                Name => Translatable('Same Customer'),
            },
        };
    }

    # show default and extra context headers if available
    if ( $ContextConfig->{$DirectAccessContextKey} ) {
        $LayoutObject->Block(
            Name => 'HeaderCustomerAccessContext',
            Data => {
                Name => Translatable('Direct'),
            },
        );
    }
    if ( $ContextConfig->{$IndirectAccessContextKey} ) {
        $LayoutObject->Block(
            Name => 'HeaderCustomerAccessContext',
            Data => {
                Name => Translatable('Indirect'),
            },
        );
    }

    # determine customers for direct and indirect access
    my @UserCustomerIDs = $Kernel::OM->Get('Kernel::System::CustomerUser')->CustomerIDs(
        User => $Param{ID},
    );
    my %ExtraCustomerIDs;
    if ( $ContextConfig->{$IndirectAccessContextKey} ) {
        my $ExtraContextName = $CustomerGroupObject->GroupContextNameGet(
            SysConfigName => $IndirectAccessContextKey,
        );

        # for all CustomerIDs get groups with extra access
        my %ExtraPermissionGroups;
        CUSTOMERID:
        for my $CustomerID (@UserCustomerIDs) {
            my %GroupList = $CustomerGroupObject->GroupCustomerList(
                CustomerID => $CustomerID,
                Type       => 'ro',
                Context    => $ExtraContextName,
                Result     => 'HASH',
            );
            next CUSTOMERID if !%GroupList;

            # add to groups
            %ExtraPermissionGroups = (
                %ExtraPermissionGroups,
                %GroupList,
            );
        }

        # add all unique accessible Group<->Customer combinations
        GROUPID:
        for my $GroupID ( sort keys %ExtraPermissionGroups ) {
            my @GroupCustomerIDs = $CustomerGroupObject->GroupCustomerList(
                GroupID => $GroupID,
                Type    => 'ro',
                Result  => 'ID',
            );
            next GROUPID if !@GroupCustomerIDs;

            # add to ExtraCustomerIDs
            %ExtraCustomerIDs = (
                %ExtraCustomerIDs,
                map { $_ => 1 } @GroupCustomerIDs,
            );
        }
    }

    # go through customers
    CUSTOMERID:
    for my $CustomerID ( sort keys %Customers ) {

        # show table rows
        $LayoutObject->Block(
            Name => 'CustomerAccessTableRow',
            Data => {
                ID   => $CustomerID,
                Name => $Customers{$CustomerID},
            },
        );

        # 'Same Customer'
        if ( $ContextConfig->{$DirectAccessContextKey} ) {

            # check if we should show check mark for 'Same Customer'
            my $AccessMark = ( grep { $_ eq $CustomerID } @UserCustomerIDs ) ? 'On' : 'Off';

            # show blocks
            $LayoutObject->Block(
                Name => 'CustomerAccessMark',
            );
            $LayoutObject->Block(
                Name => 'CustomerAccessMark' . $AccessMark,
            );
        }

        # 'Other Customers'
        next CUSTOMERID if !$ContextConfig->{$IndirectAccessContextKey};

        # check if we should show check mark for 'Other Customers'
        my $AccessMark = $ExtraCustomerIDs{$CustomerID} ? 'On' : 'Off';

        # show blocks
        $LayoutObject->Block(
            Name => 'CustomerAccessMark',
        );
        $LayoutObject->Block(
            Name => 'CustomerAccessMark' . $AccessMark,
        );
    }

    return 1;
}

1;
</File>
        <File Location="Custom/Kernel/System/CustomerCompany/DB.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - ab78e7007c327cb70702eff2fa3b665ad9e71fc2 - Kernel/System/CustomerCompany/DB.pm
# --
# 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::CustomerCompany::DB;

use v5.24;
use strict;
use warnings;

# core modules

# CPAN modules

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);

our @ObjectDependencies = (
    'Kernel::System::Cache',
# Rother OSS / CustomerMultitenancy
    'Kernel::Config',
# EO Rother OSS
    'Kernel::System::DB',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
# Rother OSS / CustomerMultitenancy
    'Kernel::System::Group',
# EO Rother OSS
    'Kernel::System::Log',
    'Kernel::System::Valid',
);

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

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

    # get customer company map
    # actually the parameter CustomerCompanyMap includes the complete config of the backend
    $Self->{CustomerCompanyMap} = $Param{CustomerCompanyMap} || die "Got no CustomerCompanyMap!";

    # config options
    $Self->{CustomerCompanyTable} = $Self->{CustomerCompanyMap}->{Params}->{Table}
        || die "Need CustomerCompany->Params->Table in Kernel/Config.pm!";
    $Self->{CustomerCompanyKey} = $Self->{CustomerCompanyMap}->{CustomerCompanyKey}
        || die "Need CustomerCompany->CustomerCompanyKey in Kernel/Config.pm!";
    $Self->{CustomerCompanyValid} = $Self->{CustomerCompanyMap}->{'CustomerCompanyValid'};
    $Self->{SearchListLimit}      = $Self->{CustomerCompanyMap}->{'CustomerCompanySearchListLimit'} || 50000;
    $Self->{SearchPrefix}         = $Self->{CustomerCompanyMap}->{'CustomerCompanySearchPrefix'};

    if ( !defined( $Self->{SearchPrefix} ) ) {
        $Self->{SearchPrefix} = '';
    }
    $Self->{SearchSuffix} = $Self->{CustomerCompanyMap}->{'CustomerCompanySearchSuffix'};
    if ( !defined( $Self->{SearchSuffix} ) ) {
        $Self->{SearchSuffix} = '*';
    }

    # create cache object, but only if CacheTTL is set in customer config
    if ( $Self->{CustomerCompanyMap}->{CacheTTL} ) {
        $Self->{CacheObject} = $Kernel::OM->Get('Kernel::System::Cache');
        $Self->{CacheType}   = 'CustomerCompany' . $Param{Count};
        $Self->{CacheTTL}    = $Self->{CustomerCompanyMap}->{CacheTTL} || 0;
    }

    # create new db connect if DSN is given
    if ( $Self->{CustomerCompanyMap}->{Params}->{DSN} ) {
        $Self->{DBObject} = Kernel::System::DB->new(
            DatabaseDSN             => $Self->{CustomerCompanyMap}->{Params}->{DSN},
            Attribute               => $Self->{CustomerCompanyMap}->{Params}->{Attribute},
            DatabaseUser            => $Self->{CustomerCompanyMap}->{Params}->{User},
            DatabasePw              => $Self->{CustomerCompanyMap}->{Params}->{Password},
            Type                    => $Self->{CustomerCompanyMap}->{Params}->{Type} || '',
            DisconnectOnDestruction => 1,
        ) || die('Can\'t connect to database!');

        # remember that we have the DBObject not from parent call
        $Self->{NotParentDBObject} = 1;
    }
    else {
        $Self->{DBObject} = $Kernel::OM->Get('Kernel::System::DB');
    }

    # this setting specifies if the table has the create_time,
    # create_by, change_time and change_by fields of OTOBO
    $Self->{ForeignDB} = $Self->{CustomerCompanyMap}->{Params}->{ForeignDB} ? 1 : 0;

    # defines if the database search will be performend case sensitive (1) or not (0)
    $Self->{CaseSensitive} = $Self->{CustomerCompanyMap}->{Params}->{SearchCaseSensitive}
        // $Self->{CustomerCompanyMap}->{Params}->{CaseSensitive} || 0;

    # fetch names of configured dynamic fields
    my @DynamicFieldMapEntries = grep { $_->[5] eq 'dynamic_field' } @{ $Self->{CustomerCompanyMap}->{Map} };
    $Self->{ConfiguredDynamicFieldNames} = { map { $_->[2] => 1 } @DynamicFieldMapEntries };

# Rother OSS / CustomerMultitenancy
    my $LayoutParam = $Kernel::OM->{Param}->{'Kernel::Output::HTML::Layout'};
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # Check if multitenancy is enabled and the request is coming from a user.
    if ( $LayoutParam->{UserType} && $LayoutParam->{UserType} eq 'User' && $ConfigObject->Get('Multitenancy') ) {

        # Save the UserID and all groups the user has 'ro' permission on.
        if ( $LayoutParam->{UserID} ) {
            # Only get the group list once for performance reasons.
            my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
                UserID => $LayoutParam->{UserID},
                Type   => 'ro',
            );

            # The limit does not count for members of a specific group.
            my $PermissionGroup = $ConfigObject->Get('Multitenancy::PermissionGroup') || '';
            my %GroupsReverse   = reverse %Groups;

            if ( !$GroupsReverse{$PermissionGroup} ) {
                $Self->{Multitenancy} = $LayoutParam->{UserID};
                $Self->{UserGroupIDs} = [ sort keys %Groups ];
                $Self->{UserGroups}   = [ values %Groups ];
            }
        }
    }
# EO CustomerMultitenancy

    return $Self;
}

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

    # check needed stuff
    my $Valid = 1;
    if ( !$Param{Valid} && defined( $Param{Valid} ) ) {
        $Valid = 0;
    }

    my $Limit = $Param{Limit} // $Self->{SearchListLimit};

    my $CacheType;
    my $CacheKey;

    # check cache
    if ( $Self->{CacheObject} ) {

        $CacheType = $Self->{CacheType} . '_CustomerCompanyList';
        $CacheKey  = "CustomerCompanyList::${Valid}::${Limit}::" . ( $Param{Search} || '' );
# Rother OSS / CustomerMultitenancy
        # Use cache for multitenancy.
        if ( $Self->{Multitenancy} ) {
            $CacheKey .= join '', map { '::GroupID=' . $_ } @{ $Self->{UserGroupIDs} };
        }
# EO CustomerMultitenancy
        my $Data = $Self->{CacheObject}->Get(
            Type => $CacheType,
            Key  => $CacheKey,
        );
        return %{$Data} if ref $Data eq 'HASH';
    }

    my $CustomerCompanyListFields = $Self->{CustomerCompanyMap}->{CustomerCompanyListFields};
    if ( !IsArrayRefWithData($CustomerCompanyListFields) ) {
        $CustomerCompanyListFields = [ 'customer_id', 'name', ];
    }

    # remove dynamic field names that are configured in CustomerCompanyListFields
    # as they cannot be handled here
    my @CustomerCompanyListFieldsWithoutDynamicFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerCompanyListFields};

    # what is the result
    my $What = join(
        ', ',
        @CustomerCompanyListFieldsWithoutDynamicFields
    );

    # add valid option if required
    my $SQL;
    my @Bind;
    my @Conditions;

    if ( $Valid && $Self->{CustomerCompanyValid} ) {

        # get valid object
        my $ValidObject = $Kernel::OM->Get('Kernel::System::Valid');

        push @Conditions, "$Self->{CustomerCompanyValid} IN ( ${\(join ', ', $ValidObject->ValidIDsGet())} )";
    }

    # where
    if ( $Param{Search} ) {

        # remove dynamic field names that are configured in CustomerCompanySearchFields
        # as they cannot be retrieved here
        my @CustomerCompanySearchFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} }
            @{ $Self->{CustomerCompanyMap}->{CustomerCompanySearchFields} };

        my %QueryCondition = $Self->{DBObject}->QueryCondition(
            Key           => \@CustomerCompanySearchFields,
            Value         => $Param{Search},
            SearchPrefix  => $Self->{SearchPrefix},
            SearchSuffix  => $Self->{SearchSuffix},
            CaseSensitive => $Self->{CaseSensitive},
            BindMode      => 1,
        );

        if ( $QueryCondition{SQL} ) {
            push @Conditions, " $QueryCondition{SQL}";
            push @Bind,       @{ $QueryCondition{Values} };
        }

# Rother OSS / CustomerMultitenancy
        # Don't search for customer without group permission.
        if ( $Self->{Multitenancy} ) {
            # Get the column name where the group ID is stored.
            my $GroupIDCol;
            for my $Map ( @{ $Self->{CustomerCompanyMap}->{Map} } ) {
                if ( $Map->[0] eq 'UserGroupID' ) {
                    $GroupIDCol = $Map->[2];
                }
            }

            if ($GroupIDCol) {
                my @UserGroupIDs = @{ $Self->{UserGroupIDs} };  # Not saving this in an array can cause an unkown error in a later iteration.
                my $UserGroupIDSync = $Self->{CustomerCompanyMap}->{UserGroupIDSync};
                if ( $UserGroupIDSync->{RemoteGroupToLocalGroup} ) {

                    # Check if group IDs or names should be checked.
                    if ( $UserGroupIDSync->{UseGroupNames} ) {
                        for my $UserGroupID ( @UserGroupIDs ) {
                            my $GroupName = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                                GroupID => $UserGroupID,
                            );
                            $UserGroupID = $GroupName;
                        }
                    }
                }

                # Replace the local group with the associated remote group.
                for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
                    my $LocalGroup = $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup};

                    for my $UserGroupID ( @UserGroupIDs ) {
                        if ( $UserGroupID eq $LocalGroup ) {
                            $UserGroupID = $RemoteGroup;
                        }
                    }
                }

                my $InCondition = $Self->{DBObject}->QueryInCondition(
                    Key      => $GroupIDCol,
                    Values   => \@UserGroupIDs,
                    BindMode => 0,
                );

                $InCondition .= " OR ($GroupIDCol IS NULL OR $GroupIDCol = '')";
                push @Conditions, " ($InCondition)";
            }
        }
# EO CustomerMultitenancy
    }

    # dynamic field handling
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my $DynamicFieldConfigs       = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerCompany',
        Valid      => 1,
    );
    my %DynamicFieldConfigsByName = map { $_->{Name} => $_ } @{$DynamicFieldConfigs};

    my @CustomerCompanyListFieldsDynamicFields = grep { exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerCompanyListFields};

    # sql
    my $CompleteSQL = "SELECT $Self->{CustomerCompanyKey}, $What FROM $Self->{CustomerCompanyTable}";

    if (@Conditions) {
        $SQL = join( ' AND ', @Conditions );
        $CompleteSQL .= " WHERE $SQL";
    }

    # get data from customer company table
    $Self->{DBObject}->Prepare(
        SQL   => $CompleteSQL,
        Bind  => \@Bind,
        Limit => $Limit,
    );

    my @CustomerCompanyData;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        push @CustomerCompanyData, [@Row];
    }

    my %List;

    CUSTOMERCOMPANYDATA:
    for my $CustomerCompanyData (@CustomerCompanyData) {
        my $CustomerCompanyID = shift @{$CustomerCompanyData};
        next CUSTOMERCOMPANYDATA if $List{$CustomerCompanyID};

        my %CompanyStringParts;

        my $FieldCounter = 0;
        for my $Field ( @{$CustomerCompanyData} ) {
            $CompanyStringParts{ $CustomerCompanyListFieldsWithoutDynamicFields[$FieldCounter] } = $Field;
            $FieldCounter++;
        }

        # fetch dynamic field values, if configured
        if (@CustomerCompanyListFieldsDynamicFields) {
            DYNAMICFIELDNAME:
            for my $DynamicFieldName (@CustomerCompanyListFieldsDynamicFields) {
                next DYNAMICFIELDNAME if !exists $DynamicFieldConfigsByName{$DynamicFieldName};

                my $Value = $DynamicFieldBackendObject->ValueGet(
                    DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                    ObjectName         => $CustomerCompanyID,
                );

                next DYNAMICFIELDNAME if !defined $Value;

                if ( !IsArrayRefWithData($Value) ) {
                    $Value = [$Value];
                }

                my @Values;

                VALUE:
                for my $CurrentValue ( @{$Value} ) {
                    next VALUE if !defined $CurrentValue || !length $CurrentValue;

                    my $ReadableValue = $DynamicFieldBackendObject->ReadableValueRender(
                        DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                        Value              => $CurrentValue,
                    );

                    next VALUE if !IsHashRefWithData($ReadableValue) || !defined $ReadableValue->{Value};

                    my $IsACLReducible = $DynamicFieldBackendObject->HasBehavior(
                        DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                        Behavior           => 'IsACLReducible',
                    );
                    if ($IsACLReducible) {
                        my $PossibleValues = $DynamicFieldBackendObject->PossibleValuesGet(
                            DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                        );

                        if (
                            IsHashRefWithData($PossibleValues)
                            && defined $PossibleValues->{ $ReadableValue->{Value} }
                            )
                        {
                            $ReadableValue->{Value} = $PossibleValues->{ $ReadableValue->{Value} };
                        }
                    }

                    push @Values, $ReadableValue->{Value};
                }

                $CompanyStringParts{$DynamicFieldName} = join ' ', @Values;
            }
        }

        # assemble company string
        my @CompanyStringParts;
        CUSTOMERCOMPANYLISTFIELD:
        for my $CustomerCompanyListField ( @{$CustomerCompanyListFields} ) {
            next CUSTOMERCOMPANYLISTFIELD
                if !exists $CompanyStringParts{$CustomerCompanyListField}
                || !defined $CompanyStringParts{$CustomerCompanyListField}
                || !length $CompanyStringParts{$CustomerCompanyListField};
            push @CompanyStringParts, $CompanyStringParts{$CustomerCompanyListField};
        }

        $List{$CustomerCompanyID} = join ' ', @CompanyStringParts;
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $CacheType,
            Key   => $CacheKey,
            Value => \%List,
            TTL   => $Self->{CacheTTL},
        );
    }

    return %List;
}

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

    if ( ref $Param{SearchFields} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "SearchFields must be an array reference!",
        );
        return;
    }

    my $Valid = defined $Param{Valid} ? $Param{Valid} : 1;

    $Param{Limit} //= '';

    # Split the search fields in scalar and array fields.
    my @ScalarSearchFields = grep { 'Input' eq $_->{Type} } @{ $Param{SearchFields} };
    my @ArraySearchFields  = grep { 'Selection' eq $_->{Type} } @{ $Param{SearchFields} };

    # Verify that all passed array parameters contain an arrayref.
    ARGUMENT:
    for my $Argument (@ArraySearchFields) {
        if ( !defined $Param{ $Argument->{Name} } ) {
            $Param{ $Argument->{Name} } ||= [];

            next ARGUMENT;
        }

        if ( ref $Param{ $Argument->{Name} } ne 'ARRAY' ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "$Argument->{Name} must be an array reference!",
            );
            return;
        }
    }

    # Set the default behaviour for the return type.
    my $Result = $Param{Result} || 'ARRAY';

    # Special handling if the result type is 'COUNT'.
    if ( $Result eq 'COUNT' ) {

        # Ignore the parameter 'Limit' when result type is 'COUNT'.
        $Param{Limit} = '';

        # Delete the OrderBy parameter when the result type is 'COUNT'.
        $Param{OrderBy} = [];
    }

    # Define order table from the search fields.
    my %OrderByTable = map { $_->{Name} => $_->{DatabaseField} } @{ $Param{SearchFields} };

    for my $Field (@ArraySearchFields) {

        my $SelectionsData = $Field->{SelectionsData};

        for my $SelectedValue ( @{ $Param{ $Field->{Name} } } ) {

            # Check if the selected value for the current field is valid.
            if ( !$SelectionsData->{$SelectedValue} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "The selected value $Field->{Name} is not valid!",
                );
                return;
            }
        }
    }

    my $DBObject = $Self->{DBObject};

    # Assemble the conditions used in the WHERE clause.
    my @SQLWhere;

    for my $Field (@ScalarSearchFields) {

        # Search for scalar fields (wildcards are allowed).
        if ( $Param{ $Field->{Name} } ) {

            # Get like escape string needed for some databases (e.g. oracle).
            my $LikeEscapeString = $DBObject->GetDatabaseFunction('LikeEscapeString');

            $Param{ $Field->{Name} } = $DBObject->Quote( $Param{ $Field->{Name} }, 'Like' );

            $Param{ $Field->{Name} } =~ s{ \*+ }{%}xmsg;

            # If the field contains more than only '%'.
            if ( $Param{ $Field->{Name} } !~ m{ \A %* \z }xms ) {
                push @SQLWhere,
                    "LOWER($Field->{DatabaseField}) LIKE LOWER('$Param{ $Field->{Name} }') $LikeEscapeString";
            }
        }
    }

    # dynamic field handling
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my $DynamicFieldConfigs       = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerCompany',
    );

    my $SQLDynamicFieldFrom     = '';
    my $SQLDynamicFieldWhere    = '';
    my $DynamicFieldJoinCounter = 1;

    DYNAMICFIELD:
    for my $DynamicField ( @{$DynamicFieldConfigs} ) {

        my $SearchParam = $Param{ "DynamicField_" . $DynamicField->{Name} };

        next DYNAMICFIELD if ( !$SearchParam );
        next DYNAMICFIELD if ( ref $SearchParam ne 'HASH' );

        my $NeedJoin;

        for my $Operator ( sort keys %{$SearchParam} ) {

            my @SearchParams = ( ref $SearchParam->{$Operator} eq 'ARRAY' )
                ? @{ $SearchParam->{$Operator} }
                : ( $SearchParam->{$Operator} );

            my $SQLDynamicFieldWhereSub = '';
            if ($SQLDynamicFieldWhere) {
                $SQLDynamicFieldWhereSub = ' AND (';
            }
            else {
                $SQLDynamicFieldWhereSub = ' (';
            }

            my $Counter = 0;
            TEXT:
            for my $Text (@SearchParams) {
                next TEXT if ( !defined $Text || $Text eq '' );

                $Text =~ s/\*/%/gi;

                # Check search attribute, we do not need to search for '*'.
                next TEXT if $Text =~ /^\%{1,3}$/;

                my $ValidateSuccess = $DynamicFieldBackendObject->ValueValidate(
                    DynamicFieldConfig => $DynamicField,
                    Value              => $Text,
                    UserID             => $Param{UserID} || 1,
                );
                if ( !$ValidateSuccess ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Search not executed due to invalid value '"
                            . $Text
                            . "' on field '"
                            . $DynamicField->{Name} . "'!",
                    );

                    return;
                }

                if ($Counter) {
                    $SQLDynamicFieldWhereSub .= ' OR ';
                }
                $SQLDynamicFieldWhereSub .= $DynamicFieldBackendObject->SearchSQLGet(
                    DynamicFieldConfig => $DynamicField,
                    TableAlias         => "dfv$DynamicFieldJoinCounter",
                    Operator           => $Operator,
                    SearchTerm         => $Text,
                );

                $Counter++;
            }
            $SQLDynamicFieldWhereSub .= ') ';

            if ($Counter) {
                $SQLDynamicFieldWhere .= $SQLDynamicFieldWhereSub;
                $NeedJoin = 1;
            }
        }

        if ($NeedJoin) {
            $SQLDynamicFieldFrom .= "
                INNER JOIN dynamic_field_value dfv$DynamicFieldJoinCounter
                    ON (df_obj_id_name.object_id = dfv$DynamicFieldJoinCounter.object_id
                        AND dfv$DynamicFieldJoinCounter.field_id = "
                . $DBObject->Quote( $DynamicField->{ID}, 'Integer' ) . ")
            ";

            $DynamicFieldJoinCounter++;
        }
    }

    # Execute a dynamic field search, if a dynamic field where statement exists.
    if ( $SQLDynamicFieldFrom && $SQLDynamicFieldWhere ) {

        my @DynamicFieldCustomerIDs;

        # Sql uery for the dynamic fields.
        my $SQLDynamicField = "SELECT DISTINCT(df_obj_id_name.object_name) FROM dynamic_field_obj_id_name df_obj_id_name "
            . $SQLDynamicFieldFrom
            . " WHERE "
            . $SQLDynamicFieldWhere;

        my $UsedCache;

        if ( $Self->{CacheObject} ) {

            my $DynamicFieldSearchCacheData = $Self->{CacheObject}->Get(
                Type => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
                Key  => $SQLDynamicField,
            );

            if ( defined $DynamicFieldSearchCacheData ) {
                if ( ref $DynamicFieldSearchCacheData eq 'ARRAY' ) {
                    @DynamicFieldCustomerIDs = @{$DynamicFieldSearchCacheData};

                    # Set the used cache flag.
                    $UsedCache = 1;
                }
                else {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => 'Invalid ref ' . ref($DynamicFieldSearchCacheData) . '!'
                    );
                    return;
                }
            }
        }

        # Get the data only from database, if no cache entry exists.
        if ( !$UsedCache ) {

            return if !$DBObject->Prepare(
                SQL => $SQLDynamicField,
            );

            while ( my @Row = $DBObject->FetchrowArray() ) {
                push @DynamicFieldCustomerIDs, $Row[0];
            }

            if ( $Self->{CacheObject} ) {
                $Self->{CacheObject}->Set(
                    Type  => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
                    Key   => $SQLDynamicField,
                    Value => \@DynamicFieldCustomerIDs,
                    TTL   => $Self->{CustomerCompanyMap}->{CacheTTL},
                );
            }
        }

        # Add the user logins from the dynamic fields, if a search result exists from the dynamic field search
        #   or skip the search and return a emptry array ref (or zero for the result 'COUNT', if no user logins exists
        #   from the dynamic field search.
        if (@DynamicFieldCustomerIDs) {

            my $SQLQueryInCondition = $DBObject->QueryInCondition(
                Key      => $Self->{CustomerCompanyKey},
                Values   => \@DynamicFieldCustomerIDs,
                BindMode => 0,
            );

            push @SQLWhere, $SQLQueryInCondition;
        }
        else {
            return $Result eq 'COUNT' ? 0 : [];
        }
    }

    FIELD:
    for my $Field (@ArraySearchFields) {

        next FIELD if !@{ $Param{ $Field->{Name} } };

        my $SQLQueryInCondition = $DBObject->QueryInCondition(
            Key      => $Field->{DatabaseField},
            Values   => $Param{ $Field->{Name} },
            BindMode => 0,
        );

        push @SQLWhere, $SQLQueryInCondition;
    }

    # Add the valid option if needed.
    if ( $Self->{CustomerCompanyMap}->{CustomerValid} && $Valid ) {

        my $ValidObject = $Kernel::OM->Get('Kernel::System::Valid');

        push @SQLWhere,
            "$Self->{CustomerCompanyMap}->{CustomerValid} IN (" . join( ', ', $ValidObject->ValidIDsGet() ) . ") ";
    }

    # Check if OrderBy contains only unique valid values.
    my %OrderBySeen;
    for my $OrderBy ( @{ $Param{OrderBy} } ) {

        if ( !$OrderBy || $OrderBySeen{$OrderBy} ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "OrderBy contains invalid value '$OrderBy' "
                    . 'or the value is used more than once!',
            );
            return;
        }

        # Remember the value to check if it appears more than once.
        $OrderBySeen{$OrderBy} = 1;
    }

    # Check if OrderByDirection array contains only 'Up' or 'Down'.
    DIRECTION:
    for my $Direction ( @{ $Param{OrderByDirection} } ) {

        # Only 'Up' or 'Down' allowed.
        next DIRECTION if $Direction eq 'Up';
        next DIRECTION if $Direction eq 'Down';

        # found an error
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "OrderByDirection can only contain 'Up' or 'Down'!",
        );
        return;
    }

    # Build the sql statement for the search.
    my $SQL = "SELECT DISTINCT($Self->{CustomerCompanyKey})";

    # Modify SQL when the result type is 'COUNT'.
    if ( $Result eq 'COUNT' ) {
        $SQL = "SELECT COUNT(DISTINCT($Self->{CustomerCompanyKey}))";
    }

    my @SQLOrderBy;

    # The Order by clause is not needed for the result type 'COUNT'.
    if ( $Result ne 'COUNT' ) {

        my $Count = 0;

        ORDERBY:
        for my $OrderBy ( @{ $Param{OrderBy} } ) {

            # Set the default order direction.
            my $Direction = 'DESC';

            # Add the given order direction.
            if ( $Param{OrderByDirection}->[$Count] ) {
                if ( $Param{OrderByDirection}->[$Count] eq 'Up' ) {
                    $Direction = 'ASC';
                }
                elsif ( $Param{OrderByDirection}->[$Count] eq 'Down' ) {
                    $Direction = 'DESC';
                }
            }

            $Count++;

            next ORDERBY if !$OrderByTable{$OrderBy};

            push @SQLOrderBy, "$OrderByTable{$OrderBy} $Direction";

            next ORDERBY if $OrderBy eq 'CustomerID';

            $SQL .= ", $OrderByTable{$OrderBy}";
        }

        # If there is a possibility that the ordering is not determined
        #   we add an descending ordering by id.
        if ( !grep { $_ eq 'CustomerID' } ( @{ $Param{OrderBy} } ) ) {
            push @SQLOrderBy, "$Self->{CustomerCompanyKey} DESC";
        }
    }

    # Add form to the SQL after the order by creation.
    $SQL .= " FROM $Self->{CustomerCompanyTable} ";

    if (@SQLWhere) {
        my $SQLWhereString = join ' AND ', map {"( $_ )"} @SQLWhere;
        $SQL .= "WHERE $SQLWhereString ";
    }

    if (@SQLOrderBy) {
        my $OrderByString = join ', ', @SQLOrderBy;
        $SQL .= "ORDER BY $OrderByString";
    }

    # Check if a cache exists before we ask the database.
    if ( $Self->{CacheObject} ) {

        my $CacheData = $Kernel::OM->Get('Kernel::System::Cache')->Get(
            Type => $Self->{CacheType} . '_CustomerCompanySearchDetail',
            Key  => $SQL . $Param{Limit},
        );

        if ( defined $CacheData ) {
            if ( ref $CacheData eq 'ARRAY' ) {
                return $CacheData;
            }
            elsif ( ref $CacheData eq '' ) {
                return $CacheData;
            }
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid ref ' . ref($CacheData) . '!'
            );
            return;
        }
    }

    return if !$DBObject->Prepare(
        SQL   => $SQL,
        Limit => $Param{Limit},
    );

    my @IDs;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        push @IDs, $Row[0];
    }

    # Handle the diffrent result types.
    if ( $Result eq 'COUNT' ) {

        if ( $Self->{CacheObject} ) {
            $Kernel::OM->Get('Kernel::System::Cache')->Set(
                Type  => $Self->{CacheType} . '_CustomerCompanySearchDetail',
                Key   => $SQL . $Param{Limit},
                Value => $IDs[0],
                TTL   => $Self->{CacheTTL},
            );
        }

        return $IDs[0];
    }

    else {

        if ( $Self->{CacheObject} ) {
            $Kernel::OM->Get('Kernel::System::Cache')->Set(
                Type  => $Self->{CacheType} . '_CustomerCompanySearchDetail',
                Key   => $SQL . $Param{Limit},
                Value => \@IDs,
                TTL   => $Self->{CacheTTL},
            );
        }

        return \@IDs;
    }
}

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

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

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Data = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => "CustomerCompanyGet::$Param{CustomerID}",
        );
        return %{$Data} if ref $Data eq 'HASH';
    }

    # build select
    my @Fields;
    my %FieldsMap;

    ENTRY:
    for my $Entry ( @{ $Self->{CustomerCompanyMap}->{Map} } ) {
        next ENTRY if $Entry->[5] eq 'dynamic_field';
        push @Fields, $Entry->[2];
        $FieldsMap{ $Entry->[2] } = $Entry->[0];
    }
    my $SQL = 'SELECT ' . join( ', ', @Fields );

    if ( !$Self->{ForeignDB} ) {
        $SQL .= ", create_time, create_by, change_time, change_by";
    }

    # this seems to be legacy, if Name is passed it should take precedence over CustomerID
    my $CustomerID = $Param{Name} || $Param{CustomerID};

    $SQL .= " FROM $Self->{CustomerCompanyTable} WHERE ";

    if ( $Self->{CaseSensitive} ) {
        $SQL .= "$Self->{CustomerCompanyKey} = ?";
    }
    else {
        $SQL .= "LOWER($Self->{CustomerCompanyKey}) = LOWER( ? )";
    }

    # get initial data
    return if !$Self->{DBObject}->Prepare(
        SQL  => $SQL,
        Bind => [ \$CustomerID ]
    );

    # fetch the result
    my %Data;
    ROW:
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {

        my $MapCounter = 0;

        for my $Field (@Fields) {
            $Data{ $FieldsMap{$Field} } = $Row[$MapCounter];
            $MapCounter++;
        }

        next ROW if $Self->{ForeignDB};

        for my $Key (qw(CreateTime CreateBy ChangeTime ChangeBy)) {
            $Data{$Key} = $Row[$MapCounter];
            $MapCounter++;
        }
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => "CustomerCompanyGet::$Param{CustomerID}",
            Value => \%Data,
            TTL   => $Self->{CacheTTL},
        );
    }

    # return data
    return (%Data);
}

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

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'CustomerCompany backend is read only!'
        );
        return;
    }

    my @Fields;
    my @Placeholders;
    my @Values;

    ENTRY:
    for my $Entry ( @{ $Self->{CustomerCompanyMap}->{Map} } ) {

        # ignore dynamic fields here
        next ENTRY if $Entry->[5] eq 'dynamic_field';

        push @Fields,       $Entry->[2];
        push @Placeholders, '?';
        push @Values,       \$Param{ $Entry->[0] };
    }
    if ( !$Self->{ForeignDB} ) {
        push @Fields,       qw(create_time create_by change_time change_by);
        push @Placeholders, qw(current_timestamp ? current_timestamp ?);
        push @Values, ( \$Param{UserID}, \$Param{UserID} );
    }

    # build insert
    my $SQL = "INSERT INTO $Self->{CustomerCompanyTable} (";
    $SQL .= join( ', ', @Fields ) . " ) VALUES ( " . join( ', ', @Placeholders ) . " )";

    return if !$Self->{DBObject}->Do(
        SQL  => $SQL,
        Bind => \@Values,
    );

    # log notice
    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'info',
        Message  =>
            "CustomerCompany: '$Param{CustomerCompanyName}/$Param{CustomerID}' created successfully ($Param{UserID})!",
    );

    $Self->_CustomerCompanyCacheClear( CustomerID => $Param{CustomerID} );

    return $Param{CustomerID};
}

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

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!'
        );
        return;
    }

    # check needed stuff
    FIELD:
    for my $Entry ( @{ $Self->{CustomerCompanyMap}->{Map} } ) {
        next FIELD if $Param{ $Entry->[0] };             # worry only about empty fields, '0' is considered as being empty
        next FIELD if $Entry->[5] eq 'dynamic_field';    # do not complain about missing dynamic fields
        next FIELD if !$Entry->[4];                      # complain only about missing required fields
        next FIELD if $Entry->[0] eq 'UserPassword';     # do not complain about missing password field

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Entry->[0]!"
        );

        return;
    }

    # Collect the info needed for the SQL UPDATE statement
    # The readonly flag is not honored here.
    my ( @Fields, @Values );
    FIELD:
    for my $Entry ( @{ $Self->{CustomerCompanyMap}->{Map} } ) {
        next FIELD if $Entry->[0] =~ m/^UserPassword$/i;    # skip the password field
        next FIELD if $Entry->[5] eq 'dynamic_field';       # skip dynamic fields
        push @Fields, $Entry->[2] . ' = ?';
        push @Values, \$Param{ $Entry->[0] };
    }
    if ( !$Self->{ForeignDB} ) {
        push @Fields, ( 'change_time = current_timestamp', 'change_by = ?' );
        push @Values, \$Param{UserID};
    }

    # create SQL statement
    my $SQL = "UPDATE $Self->{CustomerCompanyTable} SET ";
    $SQL .= join( ', ', @Fields );

    if ( $Self->{CaseSensitive} ) {
        $SQL .= " WHERE $Self->{CustomerCompanyKey} = ?";
    }
    else {
        $SQL .= " WHERE LOWER($Self->{CustomerCompanyKey}) = LOWER( ? )";
    }
    push @Values, \$Param{CustomerCompanyID};

    return if !$Self->{DBObject}->Do(
        SQL  => $SQL,
        Bind => \@Values,
    );

    # log notice
    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'info',
        Message  =>
            "CustomerCompany: '$Param{CustomerCompanyName}/$Param{CustomerID}' updated successfully ($Param{UserID})!",
    );

    $Self->_CustomerCompanyCacheClear( CustomerID => $Param{CustomerID} );
    if ( $Param{CustomerCompanyID} ne $Param{CustomerID} ) {
        $Self->_CustomerCompanyCacheClear( CustomerID => $Param{CustomerCompanyID} );
    }

    return 1;
}

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

    return if !$Self->{CacheObject};

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

    $Self->{CacheObject}->Delete(
        Type => $Self->{CacheType},
        Key  => "CustomerCompanyGet::$Param{CustomerID}",
    );

    # delete all search cache entries
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerCompanyList',
    );
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerCompanySearchDetail',
    );
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
    );

    for my $Function (qw(CustomerCompanyList)) {
        for my $Valid ( 0 .. 1 ) {
            $Self->{CacheObject}->Delete(
                Type => $Self->{CacheType},
                Key  => "${Function}::${Valid}",
            );
        }
    }

    return 1;
}

sub DESTROY {
    my $Self = shift;

    # disconnect if it's not a parent DBObject
    if ( $Self->{NotParentDBObject} ) {
        if ( $Self->{DBObject} ) {
            $Self->{DBObject}->Disconnect();
        }
    }

    return 1;
}

1;
</File>
        <File Location="Custom/Kernel/System/CustomerUser/DB.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 823aa25d26e191c2adf69c1cfde2f4510e24699b - Kernel/System/CustomerUser/DB.pm
# --
# 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::CustomerUser::DB;

use strict;
use warnings;

# core modules
use Digest::SHA ();

# CPAN modules
use Crypt::PasswdMD5 qw(apache_md5_crypt unix_md5_crypt);

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Language',
    'Kernel::System::Cache',
    'Kernel::System::CheckItem',
    'Kernel::System::DateTime',
    'Kernel::System::DB',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::Encode',
# Rother OSS / CustomerMultitenancy
    'Kernel::System::Group',
# EO Rother OSS
    'Kernel::System::Log',
    'Kernel::System::Main',
    'Kernel::System::Valid',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::DynamicFieldValueObjectName',
);

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

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

    # check needed data
    for my $Needed (qw( PreferencesObject CustomerUserMap )) {
        $Self->{$Needed} = $Param{$Needed} || die "Got no $Needed!";
    }

    # max shown user per search list
    $Self->{UserSearchListLimit} = $Self->{CustomerUserMap}->{CustomerUserSearchListLimit} || 250;

    # config options
    $Self->{CustomerTable} = $Self->{CustomerUserMap}->{Params}->{Table}
        || die "Need CustomerUser->Params->Table in Kernel/Config.pm!";
    $Self->{CustomerKey} = $Self->{CustomerUserMap}->{CustomerKey}
        || $Self->{CustomerUserMap}->{Key}
        || die "Need CustomerUser->CustomerKey in Kernel/Config.pm!";
    $Self->{CustomerID} = $Self->{CustomerUserMap}->{CustomerID}
        || die "Need CustomerUser->CustomerID in Kernel/Config.pm!";
    $Self->{ReadOnly}                 = $Self->{CustomerUserMap}->{ReadOnly};
    $Self->{ExcludePrimaryCustomerID} = $Self->{CustomerUserMap}->{CustomerUserExcludePrimaryCustomerID} || 0;
    $Self->{SearchPrefix}             = $Self->{CustomerUserMap}->{CustomerUserSearchPrefix};

    if ( !defined $Self->{SearchPrefix} ) {
        $Self->{SearchPrefix} = '';
    }
    $Self->{SearchSuffix} = $Self->{CustomerUserMap}->{CustomerUserSearchSuffix};
    if ( !defined $Self->{SearchSuffix} ) {
        $Self->{SearchSuffix} = '*';
    }

    # check if CustomerKey is var or int
    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        if ( $Entry->[0] eq 'UserLogin' && $Entry->[5] =~ /^int$/i ) {
            $Self->{CustomerKeyInteger} = 1;
            last ENTRY;
        }
    }

    # set cache type
    $Self->{CacheType} = 'CustomerUser' . $Param{Count};

    # create cache object, but only if CacheTTL is set in customer config
    if ( $Self->{CustomerUserMap}->{CacheTTL} ) {
        $Self->{CacheObject} = $Kernel::OM->Get('Kernel::System::Cache');
    }

    # create new db connect if DSN is given
    if ( $Self->{CustomerUserMap}->{Params}->{DSN} ) {
        $Self->{DBObject} = Kernel::System::DB->new(
            DatabaseDSN  => $Self->{CustomerUserMap}->{Params}->{DSN},
            DatabaseUser => $Self->{CustomerUserMap}->{Params}->{User},
            DatabasePw   => $Self->{CustomerUserMap}->{Params}->{Password},
            %{ $Self->{CustomerUserMap}->{Params} },
            DisconnectOnDestruction => 1,
        ) || die('Can\'t connect to database!');

        # remember that we have the DBObject not from parent call
        $Self->{NotParentDBObject} = 1;
    }
    else {
        $Self->{DBObject} = $Kernel::OM->Get('Kernel::System::DB');
    }

    # this setting specifies if the table has the create_time,
    # create_by, change_time and change_by fields of OTOBO
    $Self->{ForeignDB} = $Self->{CustomerUserMap}->{Params}->{ForeignDB} ? 1 : 0;

    # defines if the database search will be performend case sensitive (1) or not (0)
    $Self->{CaseSensitive} = $Self->{CustomerUserMap}->{Params}->{SearchCaseSensitive}
        // $Self->{CustomerUserMap}->{Params}->{CaseSensitive} || 0;

    # fetch names of configured dynamic fields
    my @DynamicFieldMapEntries = grep { $_->[5] eq 'dynamic_field' } @{ $Self->{CustomerUserMap}->{Map} };
    $Self->{ConfiguredDynamicFieldNames} = { map { $_->[2] => 1 } @DynamicFieldMapEntries };

# Rother OSS / CustomerMultitenancy
    my $LayoutParam = $Kernel::OM->{Param}->{'Kernel::Output::HTML::Layout'};
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # Check if multitenancy is enabled and the request is coming from a user.
    if ( $LayoutParam->{UserType} && $LayoutParam->{UserType} eq 'User' && $ConfigObject->Get('Multitenancy') ) {

        # Save the UserID and all groups the user has 'ro' permission on.
        if ( $LayoutParam->{UserID} ) {
            # Only get the group list once for performance reasons.
            my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
                UserID => $LayoutParam->{UserID},
                Type   => 'ro',
            );

            # The limit does not count for members of a specific group.
            my $PermissionGroup = $ConfigObject->Get('Multitenancy::PermissionGroup') || '';
            my %GroupsReverse   = reverse %Groups;

            if ( !$GroupsReverse{$PermissionGroup} ) {
                $Self->{Multitenancy} = $LayoutParam->{UserID};
                $Self->{UserGroupIDs} = [ sort keys %Groups ];
                $Self->{UserGroups}   = [ values %Groups ];
            }
        }
    }
# EO CustomerMultitenancy

    return $Self;
}

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

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

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Name = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => "CustomerName::$Param{UserLogin}",
        );
        return $Name if defined $Name;
    }

    my $CustomerUserNameFields = $Self->{CustomerUserMap}->{CustomerUserNameFields};
    if ( !IsArrayRefWithData($CustomerUserNameFields) ) {
        $CustomerUserNameFields = [ 'first_name', 'last_name', ];
    }

    # remove dynamic field names that are configured in CustomerUserNameFields
    # as they cannot be handled here
    my @CustomerUserNameFieldsWithoutDynamicFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerUserNameFields};

    # build SQL string 1/2
    my $SQL = "SELECT ";

    $SQL .= join( ", ", @CustomerUserNameFieldsWithoutDynamicFields );
    $SQL .= " FROM $Self->{CustomerTable} WHERE ";

    # check CustomerKey type
    my $UserLogin = $Param{UserLogin};
    if ( $Self->{CaseSensitive} ) {
        $SQL .= "$Self->{CustomerKey} = ?";
    }
    else {
        $SQL .= "LOWER($Self->{CustomerKey}) = LOWER(?)";
    }

    my %NameParts;

    # get data from customer user table
    return if !$Self->{DBObject}->Prepare(
        SQL   => $SQL,
        Bind  => [ \$Param{UserLogin} ],
        Limit => 1,
    );

    my $FieldCounter = 0;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        for my $Field (@Row) {
            $NameParts{ $CustomerUserNameFieldsWithoutDynamicFields[$FieldCounter] } = $Field;
            $FieldCounter++;
        }
    }

    # fetch dynamic field values, if configured
    my @DynamicFieldCustomerUserNameFields = grep { exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerUserNameFields};
    if (@DynamicFieldCustomerUserNameFields) {
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        DYNAMICFIELDNAME:
        for my $DynamicFieldName (@DynamicFieldCustomerUserNameFields) {
            my $DynamicFieldConfig = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
                Name => $DynamicFieldName,
            );
            next DYNAMICFIELDNAME if !IsHashRefWithData($DynamicFieldConfig);

            my $Value = $DynamicFieldBackendObject->ValueGet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectName         => $Param{UserLogin},
            );

            next DYNAMICFIELDNAME if !defined $Value;

            if ( !IsArrayRefWithData($Value) ) {
                $Value = [$Value];
            }

            my @RenderedValues;

            VALUE:
            for my $CurrentValue ( @{$Value} ) {
                next VALUE if !defined $CurrentValue || !length $CurrentValue;

                my $RenderedValue = $DynamicFieldBackendObject->ReadableValueRender(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    Value              => $CurrentValue,
                );

                next VALUE if !IsHashRefWithData($RenderedValue) || !defined $RenderedValue->{Value};

                push @RenderedValues, $RenderedValue->{Value};
            }

            $NameParts{$DynamicFieldName} = join ' ', @RenderedValues;
        }
    }

    # assemble name
    my @NameParts;
    CUSTOMERUSERNAMEFIELD:
    for my $CustomerUserNameField ( @{$CustomerUserNameFields} ) {
        next CUSTOMERUSERNAMEFIELD
            if !exists $NameParts{$CustomerUserNameField}
            || !defined $NameParts{$CustomerUserNameField}
            || !length $NameParts{$CustomerUserNameField};
        push @NameParts, $NameParts{$CustomerUserNameField};
    }

    my $JoinCharacter = $Self->{CustomerUserMap}->{CustomerUserNameFieldsJoin} // ' ';
    my $Name          = join $JoinCharacter, @NameParts;

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => "CustomerName::$Param{UserLogin}",
            Value => $Name,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }
    return $Name;
}

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

    my %Users;
    my $Valid = $Param{Valid} // 1;

    # check needed stuff
    if (
        !$Param{Search}
        && !$Param{UserLogin}
        && !$Param{PostMasterSearch}
        && !$Param{CustomerID}
        && !$Param{CustomerIDRaw}
        )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need Search, UserLogin, PostMasterSearch, CustomerIDRaw or CustomerID!',
        );
        return;
    }

    # check cache
    my $CacheKey = join '::', map { $_ . '=' . $Param{$_} } sort keys %Param;
# Rother OSS / CustomerMultitenancy
    # Use cache for multitenancy.
    if ( $Self->{Multitenancy} ) {
        $CacheKey .= join '', map { '::GroupID=' . $_ } @{ $Self->{UserGroupIDs} };
    }
# EO CustomerMultitenancy
    if ( $Self->{CacheObject} ) {
        my $Users = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType} . '_CustomerSearch',
            Key  => $CacheKey,
        );
        return %{$Users} if ref $Users eq 'HASH';
    }

    my $CustomerUserListFields = $Self->{CustomerUserMap}->{CustomerUserListFields};
    if ( !IsArrayRefWithData($CustomerUserListFields) ) {
        $CustomerUserListFields = [ 'first_name', 'last_name', 'email', ];
    }

    # remove dynamic field names that are configured in CustomerUserListFields
    # as they cannot be handled here
    my @CustomerUserListFieldsWithoutDynamicFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerUserListFields};

    # build SQL string 1/2
    my $SQL = "SELECT $Self->{CustomerKey} ";
    my @Bind;
    $SQL .= ', ' . ( join ', ', @CustomerUserListFieldsWithoutDynamicFields );

    # get like escape string needed for some databases (e.g. oracle)
    my $LikeEscapeString = $Self->{DBObject}->GetDatabaseFunction('LikeEscapeString');

    # build SQL string 2/2
    $SQL .= " FROM $Self->{CustomerTable} WHERE ";
    if ( $Param{Search} ) {
        if ( !$Self->{CustomerUserMap}->{CustomerUserSearchFields} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  =>
                    "Need CustomerUserSearchFields in CustomerUser config, unable to search for '$Param{Search}'!",
            );
            return;
        }

        my $Search = $Self->{DBObject}->QueryStringEscape( QueryString => $Param{Search} );

        # remove dynamic field names that are configured in CustomerUserSearchFields
        # as they cannot be retrieved here
        my @CustomerUserSearchFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} }
            @{ $Self->{CustomerUserMap}->{CustomerUserSearchFields} };

        if ( $Param{CustomerUserOnly} ) {
            @CustomerUserSearchFields = grep { $_ ne 'customer_id' } @CustomerUserSearchFields;
        }

        my %QueryCondition = $Self->{DBObject}->QueryCondition(
            Key           => \@CustomerUserSearchFields,    #$Self->{CustomerUserMap}->{CustomerUserSearchFields},
            Value         => $Search,
            SearchPrefix  => $Self->{SearchPrefix},
            SearchSuffix  => $Self->{SearchSuffix},
            CaseSensitive => $Self->{CaseSensitive},
            BindMode      => 1,
        );

        $SQL .= $QueryCondition{SQL};
        push @Bind, @{ $QueryCondition{Values} };

# Rother OSS / CustomerMultitenancy
        # Don't search for customer users without group permission.
        if ( $Self->{Multitenancy} ) {
            if ( $Self->{CustomerUserMap}->{CustomerCompanySupport} ) {
                # TODO: Workaround until the search gets changed.
                $Param{Limit} = 100000;
                # Normally, we should inner join all customer companies to see if the agent has permission to see the customer (currently working) or the customer company (iterate through all DB back-ends).
                # If the agent does not have permission on the customer company but the customer user, the query shouldn't return the customer user.
                # Currently, we are looking for everything and filtering the result, which is not optimal. The real limit should still be applied in the System/CustomerUser.pm.
                # The same applies for the CustomerUser/LDAP.pm
                # SELECT * FROM customer_user cu
                # JOIN customer_company cp ON cu.customer_id = cp.customer_id
                # WHERE ((cp.group_id IN (1) OR cp.group_id IS NULL OR cp.group_id = '')
                #     AND (cu.group_id IS NULL OR cu.group_id = ''))
                # OR (cu.group_id IN (1))
                # UNION
                # SELECT * FROM customer_user cu
                # JOIN customer_company_2 cp ON cu.customer_id = cp.customer_id
                # WHERE ((cp.group_id IN (1) OR cp.group_id IS NULL OR cp.group_id = '')
                #     AND (cu.group_id IS NULL OR cu.group_id = ''))
                # OR (cu.group_id IN (1))
            }

            # Get the column name where the group ID is stored.
            my $GroupIDCol;
            for my $Map ( @{ $Self->{CustomerUserMap}->{Map} } ) {
                if ( $Map->[0] eq 'UserGroupID' ) {
                    $GroupIDCol = $Map->[2];
                }
            }

            # The source supports multitenancy.
            if ($GroupIDCol) {
                my $UserGroupIDSync = $Self->{CustomerUserMap}->{UserGroupIDSync};
                # Check if we match for group IDs or group names.
                my @UserGroups = $UserGroupIDSync->{UseGroupNames} ? @{ $Self->{UserGroups} } : @{ $Self->{UserGroupIDs} };

                # Remap local groups to remote groups.
                for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
                    my $LocalGroup = $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup};

                    @UserGroups = map { $_ eq $LocalGroup ? $RemoteGroup : $_ } @UserGroups;
                }

                my $InCondition = $Self->{DBObject}->QueryInCondition(
                    Key      => $GroupIDCol,
                    Values   => \@UserGroups,
                    BindMode => 0,
                );

                # If the field is empty, we have access by default.
                $InCondition .= " OR ($GroupIDCol IS NULL OR $GroupIDCol = '')";
                $SQL .= " AND ($InCondition)";
            }
        }
# EO CustomerMultitenancy

        $SQL .= ' ';
    }
    elsif ( $Param{PostMasterSearch} ) {
        if ( $Self->{CustomerUserMap}->{CustomerUserPostMasterSearchFields} ) {

            # remove dynamic field names that are configured in CustomerUserPostMasterSearchFields
            # as they cannot be retrieved here
            my @CustomerUserPostMasterSearchFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} }
                @{ $Self->{CustomerUserMap}->{CustomerUserPostMasterSearchFields} };

            my $SQLExt = '';

            # for my $Field ( @{ $Self->{CustomerUserMap}->{CustomerUserPostMasterSearchFields} } ) {
            for my $Field (@CustomerUserPostMasterSearchFields) {
                if ($SQLExt) {
                    $SQLExt .= ' OR ';
                }
                my $PostMasterSearch = $Self->{DBObject}->Quote( $Param{PostMasterSearch} );
                push @Bind, \$PostMasterSearch;

                if ( $Self->{CaseSensitive} ) {
                    $SQLExt .= " $Field = ? ";
                }
                else {
                    $SQLExt .= " LOWER($Field) = LOWER(?) ";
                }
            }
            $SQL .= $SQLExt;
        }
    }
    elsif ( $Param{UserLogin} ) {

        my $UserLogin = $Param{UserLogin};

        # check CustomerKey type
        if ( $Self->{CustomerKeyInteger} ) {

            # return if login is no integer
            return if $Param{UserLogin} !~ /^(\+|\-|)\d{1,16}$/;

            $SQL .= "$Self->{CustomerKey} = ?";
            push @Bind, \$UserLogin;
        }
        else {
            $UserLogin = '%' . $Self->{DBObject}->Quote( $UserLogin, 'Like' ) . '%';
            $UserLogin =~ s/\*/%/g;
            push @Bind, \$UserLogin;
            if ( $Self->{CaseSensitive} ) {
                $SQL .= "$Self->{CustomerKey} LIKE ? $LikeEscapeString";
            }
            else {
                $SQL .= "LOWER($Self->{CustomerKey}) LIKE LOWER(?) $LikeEscapeString";
            }
        }
    }
    elsif ( $Param{CustomerID} ) {

        my $CustomerID = $Self->{DBObject}->Quote( $Param{CustomerID}, 'Like' );
        $CustomerID =~ s/\*/%/g;
        push @Bind, \$CustomerID;

        if ( $Self->{CaseSensitive} ) {
            $SQL .= "$Self->{CustomerID} LIKE ? $LikeEscapeString";
        }
        else {
            $SQL .= "LOWER($Self->{CustomerID}) LIKE LOWER(?) $LikeEscapeString";
        }
    }
    elsif ( $Param{CustomerIDRaw} ) {

        push @Bind, \$Param{CustomerIDRaw};

        if ( $Self->{CaseSensitive} ) {
            $SQL .= "$Self->{CustomerID} = ? ";
        }
        else {
            $SQL .= "LOWER($Self->{CustomerID}) = LOWER(?) ";
        }
    }

    # add valid option
    if ( $Self->{CustomerUserMap}->{CustomerValid} && $Valid ) {

        # get valid object
        my $ValidObject = $Kernel::OM->Get('Kernel::System::Valid');

        $SQL .= ' AND '
            . $Self->{CustomerUserMap}->{CustomerValid}
            . ' IN (' . join( ', ', $ValidObject->ValidIDsGet() ) . ') ';
    }

    # dynamic field handling
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    my $DynamicFieldConfigs = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
        Valid      => 1,
    );
    my %DynamicFieldConfigsByName = map { $_->{Name} => $_ } @{$DynamicFieldConfigs};

    my @CustomerUserListFieldsDynamicFields = grep { exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerUserListFields};

    # get data from customer user table
    return if !$Self->{DBObject}->Prepare(
        SQL   => $SQL,
        Bind  => \@Bind,
        Limit => $Param{Limit} || $Self->{UserSearchListLimit},
    );

    my @CustomerUserData;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        push @CustomerUserData, [@Row];
    }

    CUSTOMERUSERDATA:
    for my $CustomerUserData (@CustomerUserData) {
        my $CustomerKey = shift @{$CustomerUserData};
        next CUSTOMERUSERDATA if $Users{$CustomerKey};

        my %UserStringParts;

        my $FieldCounter = 0;
        for my $Field ( @{$CustomerUserData} ) {
            $UserStringParts{ $CustomerUserListFieldsWithoutDynamicFields[$FieldCounter] } = $Field;
            $FieldCounter++;
        }

        # fetch dynamic field values, if configured
        if (@CustomerUserListFieldsDynamicFields) {
            DYNAMICFIELDNAME:
            for my $DynamicFieldName (@CustomerUserListFieldsDynamicFields) {
                next DYNAMICFIELDNAME if !exists $DynamicFieldConfigsByName{$DynamicFieldName};

                my $Value = $DynamicFieldBackendObject->ValueGet(
                    DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                    ObjectName         => $CustomerKey,
                );

                next DYNAMICFIELDNAME if !defined $Value;

                if ( !IsArrayRefWithData($Value) ) {
                    $Value = [$Value];
                }

                my @Values;

                VALUE:
                for my $CurrentValue ( @{$Value} ) {
                    next VALUE if !defined $CurrentValue || !length $CurrentValue;

                    my $ReadableValue = $DynamicFieldBackendObject->ReadableValueRender(
                        DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                        Value              => $CurrentValue,
                    );

                    next VALUE if !IsHashRefWithData($ReadableValue) || !defined $ReadableValue->{Value};

                    my $IsACLReducible = $DynamicFieldBackendObject->HasBehavior(
                        DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                        Behavior           => 'IsACLReducible',
                    );
                    if ($IsACLReducible) {
                        my $PossibleValues = $DynamicFieldBackendObject->PossibleValuesGet(
                            DynamicFieldConfig => $DynamicFieldConfigsByName{$DynamicFieldName},
                        );

                        if (
                            IsHashRefWithData($PossibleValues)
                            && defined $PossibleValues->{ $ReadableValue->{Value} }
                            )
                        {
                            $ReadableValue->{Value} = $PossibleValues->{ $ReadableValue->{Value} };
                        }
                    }

                    push @Values, $ReadableValue->{Value};
                }

                $UserStringParts{$DynamicFieldName} = join ' ', @Values;
            }
        }

        # assemble user string
        my @UserStringParts;
        CUSTOMERUSERLISTFIELD:
        for my $CustomerUserListField ( @{$CustomerUserListFields} ) {
            next CUSTOMERUSERLISTFIELD
                if !exists $UserStringParts{$CustomerUserListField}
                || !defined $UserStringParts{$CustomerUserListField}
                || !length $UserStringParts{$CustomerUserListField};
            push @UserStringParts, $UserStringParts{$CustomerUserListField};
        }

        $Users{$CustomerKey} = join ' ', @UserStringParts;
        $Users{$CustomerKey} =~ s/^(.*)\s(.+?\@.+?\..+?)(\s|)$/"$1" <$2>/;
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType} . '_CustomerSearch',
            Key   => $CacheKey,
            Value => \%Users,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }
    return %Users;
}

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

    if ( ref $Param{SearchFields} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "SearchFields must be an array reference!",
        );
        return;
    }

    # Return only valid users per default
    my $Valid = $Param{Valid} // 1;

    $Param{Limit} //= '';

    # Split the search fields in scalar and array fields, before the diffrent handling.
    my @ScalarSearchFields = grep { 'Input' eq $_->{Type} } @{ $Param{SearchFields} };
    my @ArraySearchFields  = grep { 'Selection' eq $_->{Type} } @{ $Param{SearchFields} };

    # Verify that all passed array parameters contain an arrayref.
    ARGUMENT:
    for my $Argument (@ArraySearchFields) {
        if ( !defined $Param{ $Argument->{Name} } ) {
            $Param{ $Argument->{Name} } ||= [];

            next ARGUMENT;
        }

        if ( ref $Param{ $Argument->{Name} } ne 'ARRAY' ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "$Argument->{Name} must be an array reference!",
            );
            return;
        }
    }

    # Set the default behaviour for the return type.
    my $Result = $Param{Result} || 'ARRAY';

    # Special handling if the result type is 'COUNT'.
    if ( $Result eq 'COUNT' ) {

        # Ignore the parameter 'Limit' when result type is 'COUNT'.
        $Param{Limit} = '';

        # Delete the OrderBy parameter when the result type is 'COUNT'.
        $Param{OrderBy} = [];
    }

    # Define order table from the search fields.
    my %OrderByTable = map { $_->{Name} => $_->{DatabaseField} } @{ $Param{SearchFields} };

    for my $Field (@ArraySearchFields) {

        my $SelectionsData = $Field->{SelectionsData};

        for my $SelectedValue ( @{ $Param{ $Field->{Name} } } ) {

            # Check if the selected value for the current field is valid.
            if ( !$SelectionsData->{$SelectedValue} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "The selected value $Field->{Name} is not valid!",
                );
                return;
            }
        }
    }

    my $DBObject = $Self->{DBObject};

    # Assemble the conditions used in the WHERE clause.
    my @SQLWhere;

    for my $Field (@ScalarSearchFields) {

        # Search for scalar fields (wildcards are allowed).
        if ( $Param{ $Field->{Name} } ) {

            # Get like escape string needed for some databases (e.g. oracle).
            my $LikeEscapeString = $DBObject->GetDatabaseFunction('LikeEscapeString');

            $Param{ $Field->{Name} } = $DBObject->Quote( $Param{ $Field->{Name} }, 'Like' );

            $Param{ $Field->{Name} } =~ s{ \*+ }{%}xmsg;

            # If the field contains more than only %.
            if ( $Param{ $Field->{Name} } !~ m{ \A %* \z }xms ) {
                push @SQLWhere,
                    "LOWER($Field->{DatabaseField}) LIKE LOWER('$Param{ $Field->{Name} }') $LikeEscapeString";
            }
        }
    }

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

    # Check all configured change dynamic fields, build lookup hash by name.
    my %CustomerUserDynamicFieldName2Config;
    my $CustomerUserDynamicFields = $DynamicFieldObject->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
    );
    for my $DynamicField ( @{$CustomerUserDynamicFields} ) {
        $CustomerUserDynamicFieldName2Config{ $DynamicField->{Name} } = $DynamicField;
    }

    my $SQLDynamicFieldFrom     = '';
    my $SQLDynamicFieldWhere    = '';
    my $DynamicFieldJoinCounter = 1;

    DYNAMICFIELD:
    for my $DynamicField ( @{$CustomerUserDynamicFields} ) {

        my $SearchParam = $Param{ "DynamicField_" . $DynamicField->{Name} };

        next DYNAMICFIELD if ( !$SearchParam );
        next DYNAMICFIELD if ( ref $SearchParam ne 'HASH' );

        my $NeedJoin;

        for my $Operator ( sort keys %{$SearchParam} ) {

            my @SearchParams = ( ref $SearchParam->{$Operator} eq 'ARRAY' )
                ? @{ $SearchParam->{$Operator} }
                : ( $SearchParam->{$Operator} );

            my $SQLDynamicFieldWhereSub = '';
            if ($SQLDynamicFieldWhere) {
                $SQLDynamicFieldWhereSub = ' AND (';
            }
            else {
                $SQLDynamicFieldWhereSub = ' (';
            }

            my $Counter = 0;
            TEXT:
            for my $Text (@SearchParams) {
                next TEXT if ( !defined $Text || $Text eq '' );

                $Text =~ s/\*/%/gi;

                # Check search attribute, we do not need to search for '*'.
                next TEXT if $Text =~ /^\%{1,3}$/;

                my $ValidateSuccess = $DynamicFieldBackendObject->ValueValidate(
                    DynamicFieldConfig => $DynamicField,
                    Value              => $Text,
                    UserID             => $Param{UserID} || 1,
                );
                if ( !$ValidateSuccess ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Search not executed due to invalid value '"
                            . $Text
                            . "' on field '"
                            . $DynamicField->{Name} . "'!",
                    );
                    return;
                }

                if ($Counter) {
                    $SQLDynamicFieldWhereSub .= ' OR ';
                }
                $SQLDynamicFieldWhereSub .= $DynamicFieldBackendObject->SearchSQLGet(
                    DynamicFieldConfig => $DynamicField,
                    TableAlias         => "dfv$DynamicFieldJoinCounter",
                    Operator           => $Operator,
                    SearchTerm         => $Text,
                );

                $Counter++;
            }
            $SQLDynamicFieldWhereSub .= ') ';

            if ($Counter) {
                $SQLDynamicFieldWhere .= $SQLDynamicFieldWhereSub;
                $NeedJoin = 1;
            }
        }

        if ($NeedJoin) {
            $SQLDynamicFieldFrom .= "
                INNER JOIN dynamic_field_value dfv$DynamicFieldJoinCounter
                    ON (df_obj_id_name.object_id = dfv$DynamicFieldJoinCounter.object_id
                        AND dfv$DynamicFieldJoinCounter.field_id = "
                . $DBObject->Quote( $DynamicField->{ID}, 'Integer' ) . ")
            ";

            $DynamicFieldJoinCounter++;
        }
    }

    # Execute a dynamic field search, if a dynamic field where statement exists.
    if ($SQLDynamicFieldWhere) {

        my @DynamicFieldUserLogins;

        my $SQLDynamicField = "SELECT DISTINCT(df_obj_id_name.object_name) FROM dynamic_field_obj_id_name df_obj_id_name "
            . $SQLDynamicFieldFrom
            . " WHERE "
            . $SQLDynamicFieldWhere;

        my $UsedCache;

        if ( $Self->{CacheObject} ) {

            my $DynamicFieldSearchCacheData = $Self->{CacheObject}->Get(
                Type => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
                Key  => $SQLDynamicField,
            );

            if ( defined $DynamicFieldSearchCacheData ) {
                if ( ref $DynamicFieldSearchCacheData eq 'ARRAY' ) {
                    @DynamicFieldUserLogins = @{$DynamicFieldSearchCacheData};

                    $UsedCache = 1;
                }
                else {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => 'Invalid ref ' . ref($DynamicFieldSearchCacheData) . '!'
                    );
                    return;
                }
            }
        }

        # Get the data only from database, if no cache entry exists.
        if ( !$UsedCache ) {

            return if !$DBObject->Prepare(
                SQL => $SQLDynamicField,
            );

            while ( my @Row = $DBObject->FetchrowArray() ) {
                push @DynamicFieldUserLogins, $Row[0];
            }

            if ( $Self->{CacheObject} ) {
                $Self->{CacheObject}->Set(
                    Type  => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
                    Key   => $SQLDynamicField,
                    Value => \@DynamicFieldUserLogins,
                    TTL   => $Self->{CustomerUserMap}->{CacheTTL},
                );
            }
        }

        # Add the user logins from the dynamic fields, if a search result exists from the dynamic field search
        #   or skip the search and return a emptry array ref, if no user logins exists from the dynamic field search.
        if (@DynamicFieldUserLogins) {

            my $SQLQueryInCondition = $DBObject->QueryInCondition(
                Key      => $Self->{CustomerKey},
                Values   => \@DynamicFieldUserLogins,
                BindMode => 0,
            );

            push @SQLWhere, $SQLQueryInCondition;
        }
        else {
            return $Result eq 'COUNT' ? 0 : [];
        }
    }

    FIELD:
    for my $Field (@ArraySearchFields) {

        next FIELD if !@{ $Param{ $Field->{Name} } };

        my $SQLQueryInCondition = $DBObject->QueryInCondition(
            Key      => $Field->{DatabaseField},
            Values   => $Param{ $Field->{Name} },
            BindMode => 0,
        );

        push @SQLWhere, $SQLQueryInCondition;
    }

    # Special parameter for CustomerIDs from a customer company search result.
    if ( IsArrayRefWithData( $Param{CustomerCompanySearchCustomerIDs} ) ) {

        my $SQLQueryInCondition = $DBObject->QueryInCondition(
            Key      => $Self->{CustomerID},
            Values   => $Param{CustomerCompanySearchCustomerIDs},
            BindMode => 0,
        );

        push @SQLWhere, $SQLQueryInCondition;
    }

    # Special parameter to exclude some user logins from the search result.
    if ( IsArrayRefWithData( $Param{ExcludeUserLogins} ) ) {

        my $SQLQueryInCondition = $DBObject->QueryInCondition(
            Key      => $Self->{CustomerKey},
            Values   => $Param{ExcludeUserLogins},
            BindMode => 0,
            Negate   => 1,
        );

        push @SQLWhere, $SQLQueryInCondition;
    }

    # Add the valid option if needed.
    if ( $Self->{CustomerUserMap}->{CustomerValid} && $Valid ) {

        my $ValidObject = $Kernel::OM->Get('Kernel::System::Valid');

        push @SQLWhere,
            "$Self->{CustomerUserMap}->{CustomerValid} IN (" . join( ', ', $ValidObject->ValidIDsGet() ) . ") ";
    }

    # Check if OrderBy contains only unique valid values.
    my %OrderBySeen;
    for my $OrderBy ( @{ $Param{OrderBy} } ) {

        if ( !$OrderBy || $OrderBySeen{$OrderBy} ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "OrderBy contains invalid value '$OrderBy' or the value is used more than once!",
            );
            return;
        }

        # Remember the value to check if it appears more than once.
        $OrderBySeen{$OrderBy} = 1;
    }

    # Check if OrderByDirection array contains only 'Up' or 'Down'.
    DIRECTION:
    for my $Direction ( @{ $Param{OrderByDirection} } ) {

        # Only 'Up' or 'Down' allowed.
        next DIRECTION if $Direction eq 'Up';
        next DIRECTION if $Direction eq 'Down';

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "OrderByDirection can only contain 'Up' or 'Down'!",
        );
        return;
    }

    # Build the sql statement for the search.
    my $SQL = "SELECT DISTINCT($Self->{CustomerKey})";

    # Modify SQL when the result type is 'COUNT'.
    if ( $Result eq 'COUNT' ) {
        $SQL = "SELECT COUNT(DISTINCT($Self->{CustomerKey}))";
    }

    # Assemble the ORDER BY clause.
    my @SQLOrderBy;

    # The Order by clause is not needed for the result type 'COUNT'.
    if ( $Result ne 'COUNT' ) {

        my $Count = 0;

        ORDERBY:
        for my $OrderBy ( @{ $Param{OrderBy} } ) {

            my $Direction = 'DESC';

            if ( $Param{OrderByDirection}->[$Count] ) {
                if ( $Param{OrderByDirection}->[$Count] eq 'Up' ) {
                    $Direction = 'ASC';
                }
                elsif ( $Param{OrderByDirection}->[$Count] eq 'Down' ) {
                    $Direction = 'DESC';
                }
            }

            $Count++;

            next ORDERBY if !$OrderByTable{$OrderBy};

            push @SQLOrderBy, "$OrderByTable{$OrderBy} $Direction";

            next ORDERBY if $OrderBy eq 'UserLogin';

            $SQL .= ", $OrderByTable{$OrderBy}";
        }

        # If there is a possibility that the ordering is not determined
        #   we add an descending ordering by id.
        if ( !grep { $_ eq 'UserLogin' } ( @{ $Param{OrderBy} } ) ) {
            push @SQLOrderBy, "$Self->{CustomerKey} DESC";
        }
    }

    $SQL .= " FROM $Self->{CustomerTable} ";

    if (@SQLWhere) {
        my $SQLWhereString = join ' AND ', map {"( $_ )"} @SQLWhere;
        $SQL .= "WHERE $SQLWhereString ";
    }
    if (@SQLOrderBy) {
        my $OrderByString = join ', ', @SQLOrderBy;
        $SQL .= "ORDER BY $OrderByString";
    }

    # Check if a cache exists before we ask the database.
    if ( $Self->{CacheObject} ) {

        my $CacheData = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType} . '_CustomerSearchDetail',
            Key  => $SQL . $Param{Limit},
        );

        if ( defined $CacheData ) {
            if ( ref $CacheData eq 'ARRAY' ) {
                return $CacheData;
            }
            elsif ( ref $CacheData eq '' ) {
                return $CacheData;
            }
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid ref ' . ref($CacheData) . '!'
            );
            return;
        }
    }

    return if !$DBObject->Prepare(
        SQL   => $SQL,
        Limit => $Param{Limit},
    );

    my @IDs;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        push @IDs, $Row[0];
    }

    # Handle the diffrent result types.
    if ( $Result eq 'COUNT' ) {

        if ( $Self->{CacheObject} ) {
            $Self->{CacheObject}->Set(
                Type  => $Self->{CacheType} . '_CustomerSearchDetail',
                Key   => $SQL . $Param{Limit},
                Value => $IDs[0],
                TTL   => $Self->{CustomerUserMap}->{CacheTTL},
            );
        }

        return $IDs[0];
    }
    else {

        if ( $Self->{CacheObject} ) {
            $Self->{CacheObject}->Set(
                Type  => $Self->{CacheType} . '_CustomerSearchDetail',
                Key   => $SQL . $Param{Limit},
                Value => \@IDs,
                TTL   => $Self->{CustomerUserMap}->{CacheTTL},
            );
        }

        return \@IDs;
    }
}

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

    my $Valid      = defined $Param{Valid} ? $Param{Valid} : 1;
    my $SearchTerm = $Param{SearchTerm} || '';

    my $CacheType = $Self->{CacheType} . '_CustomerIDList';
    my $CacheKey  = "CustomerIDList::${Valid}::$SearchTerm";

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Result = $Self->{CacheObject}->Get(
            Type => $CacheType,
            Key  => $CacheKey,
        );
        return @{$Result} if ref $Result eq 'ARRAY';
    }

    my $SQL = "
        SELECT DISTINCT($Self->{CustomerID})
        FROM $Self->{CustomerTable}
        WHERE 1 = 1 ";
    my @Bind;

    # add valid option
    if ( $Self->{CustomerUserMap}->{CustomerValid} && $Valid ) {

        # get valid object
        my $ValidObject = $Kernel::OM->Get('Kernel::System::Valid');

        my $ValidIDs = join( ', ', $ValidObject->ValidIDsGet() );
        $SQL .= "
            AND $Self->{CustomerUserMap}->{CustomerValid} IN ($ValidIDs) ";
    }

    # add search term
    if ($SearchTerm) {
        my $SearchTermEscaped = $Self->{DBObject}->QueryStringEscape( QueryString => $SearchTerm );

        $SQL .= ' AND ';
        my %QueryCondition = $Self->{DBObject}->QueryCondition(
            Key           => $Self->{CustomerID},
            Value         => $SearchTermEscaped,
            SearchPrefix  => $Self->{SearchPrefix},
            SearchSuffix  => $Self->{SearchSuffix},
            CaseSensitive => $Self->{CaseSensitive},
            BindMode      => 1,
        );
        $SQL .= $QueryCondition{SQL};
        push @Bind, @{ $QueryCondition{Values} };

        $SQL .= ' ';
    }

    return if !$Self->{DBObject}->Prepare(
        SQL  => $SQL,
        Bind => \@Bind,
    );

    my @Result;

    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        push @Result, $Row[0];
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $CacheType,
            Key   => $CacheKey,
            Value => \@Result,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }
    return @Result;
}

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

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

    # check cache
    if ( $Self->{CacheObject} ) {
        my $CustomerIDs = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => "CustomerIDs::$Param{User}",
        );
        return @{$CustomerIDs} if ref $CustomerIDs eq 'ARRAY';
    }

    # get customer data
    my %Data = $Self->CustomerUserDataGet(
        User => $Param{User},
    );

    # there are multi customer ids
    my @CustomerIDs;
    if ( $Data{UserCustomerIDs} ) {

        # used separators
        SEPARATOR:
        for my $Separator ( ';', ',', '|' ) {

            next SEPARATOR if $Data{UserCustomerIDs} !~ /\Q$Separator\E/;

            # split it
            my @IDs = split /\Q$Separator\E/, $Data{UserCustomerIDs};

            for my $ID (@IDs) {
                $ID =~ s/^\s+//g;
                $ID =~ s/\s+$//g;
                push @CustomerIDs, $ID;
            }

            last SEPARATOR;
        }

        # fallback if no separator got found
        if ( !@CustomerIDs ) {
            $Data{UserCustomerIDs} =~ s/^\s+//g;
            $Data{UserCustomerIDs} =~ s/\s+$//g;
            push @CustomerIDs, $Data{UserCustomerIDs};
        }
    }

    # use also the primary customer id
    if ( $Data{UserCustomerID} && !$Self->{ExcludePrimaryCustomerID} ) {
        push @CustomerIDs, $Data{UserCustomerID};
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => 'CustomerIDs::' . $Param{User},
            Value => \@CustomerIDs,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return @CustomerIDs;
}

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

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

    # build select
    my $SQL = 'SELECT ';
    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {

        next ENTRY if $Entry->[5] eq 'dynamic_field';

        $SQL .= " $Entry->[2], ";
    }

    if ( !$Self->{ForeignDB} ) {
        $SQL .= "create_time, create_by, change_time, change_by, ";
    }

    $SQL .= $Self->{CustomerKey} . " FROM $Self->{CustomerTable} WHERE ";

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Data = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => "CustomerUserDataGet::$Param{User}",
        );
        return %{$Data} if ref $Data eq 'HASH';
    }

    # check customer key type
    my $User = $Param{User};

    if ( $Self->{CaseSensitive} ) {
        $SQL .= "$Self->{CustomerKey} = ?";
    }
    else {
        $SQL .= "LOWER($Self->{CustomerKey}) = LOWER(?)";
    }

    # ask the database
    return if !$Self->{DBObject}->Prepare(
        SQL   => $SQL,
        Bind  => [ \$User ],
        Limit => 1,
    );

    # fetch the result
    my %Data;
    ROW:
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {

        my $MapCounter = 0;

        ENTRY:
        for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {

            next ENTRY if $Entry->[5] eq 'dynamic_field';

            $Data{ $Entry->[0] } = $Row[$MapCounter];
            $MapCounter++;
        }

        next ROW if $Self->{ForeignDB};

        for my $Key (qw(CreateTime CreateBy ChangeTime ChangeBy)) {
            $Data{$Key} = $Row[$MapCounter];
            $MapCounter++;
        }
    }

    # check data
    if ( !$Data{UserLogin} ) {

        # cache request
        if ( $Self->{CacheObject} ) {
            $Self->{CacheObject}->Set(
                Type  => $Self->{CacheType},
                Key   => "CustomerUserDataGet::$Param{User}",
                Value => {},
                TTL   => $Self->{CustomerUserMap}->{CacheTTL},
            );
        }
        return;
    }

    my $CustomerUserListFieldsMap = $Self->{CustomerUserMap}->{CustomerUserListFields};
    if ( !IsArrayRefWithData($CustomerUserListFieldsMap) ) {
        $CustomerUserListFieldsMap = [ 'first_name', 'last_name', 'email', ];
    }

    # Order fields by CustomerUserListFields (see bug#13821).
    my @CustomerUserListFields;
    for my $Field ( @{$CustomerUserListFieldsMap} ) {
        my @FieldNames = map { $_->[0] } grep { $_->[2] eq $Field } @{ $Self->{CustomerUserMap}->{Map} };
        push @CustomerUserListFields, $FieldNames[0];
    }

    my $UserMailString = '';
    my @UserMailStringParts;

    FIELD:
    for my $Field (@CustomerUserListFields) {
        next FIELD if !$Data{$Field};

        push @UserMailStringParts, $Data{$Field};
    }
    $UserMailString = join ' ', @UserMailStringParts;
    $UserMailString =~ s/^(.*)\s(.+?\@.+?\..+?)(\s|)$/"$1" <$2>/;

    # add the UserMailString to the data hash
    $Data{UserMailString} = $UserMailString;

    # compat!
    $Data{UserID} = $Data{UserLogin};

    # get preferences
    my %Preferences = $Self->GetPreferences( UserID => $Data{UserID} );

    # add last login timestamp
    if ( $Preferences{UserLastLogin} ) {

        my $DateTimeObject = $Kernel::OM->Create(
            'Kernel::System::DateTime',
            ObjectParams => {
                Epoch => $Preferences{UserLastLogin},
            },
        );

        $Preferences{UserLastLoginTimestamp} = $DateTimeObject->ToString();

    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => "CustomerUserDataGet::$Param{User}",
            Value => { %Data, %Preferences },
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return ( %Data, %Preferences );
}

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

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!',
        );
        return;
    }

    # check needed stuff
    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        if ( !$Param{ $Entry->[0] } && $Entry->[4] ) {

            # skip UserLogin, will be checked later
            next ENTRY if ( $Entry->[0] eq 'UserLogin' );

            # ignore dynamic fields here
            next ENTRY if $Entry->[5] eq 'dynamic_field';

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Entry->[0]!",
            );
            return;
        }
    }
    if ( !$Param{UserID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need UserID!',
        );
        return;
    }

    # if no UserLogin is given
    if ( !$Param{UserLogin} && $Self->{CustomerUserMap}->{AutoLoginCreation} ) {

        # get time object
        my $DateTimeObject = $Kernel::OM->Create('Kernel::System::DateTime');
        my $DateTimeString = $DateTimeObject->Format( Format => '%Y%m%d%H%M' );

        my $Prefix = $Self->{CustomerUserMap}->{AutoLoginCreationPrefix} || 'auto';
        $Param{UserLogin} = "$Prefix-$DateTimeString" . int( rand(99) );
    }

    # check if user login exists
    if ( !$Param{UserLogin} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need UserLogin!',
        );
        return;
    }

    # check email address if already exists
    if ( $Param{UserEmail} && $Self->{CustomerUserMap}->{CustomerUserEmailUniqCheck} ) {
        my %Result = $Self->CustomerSearch(
            Valid            => 0,
            PostMasterSearch => $Param{UserEmail},
        );
        if (%Result) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => $Kernel::OM->Get('Kernel::Language')->Translate('This email address is already in use for another customer user.'),
            );
            return;
        }
    }

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    # check email address mx
    if (
        $Param{UserEmail}
        && !$CheckItemObject->CheckEmail( Address => $Param{UserEmail} )
        && grep { $_ eq $Param{ValidID} } $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet()
        )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Email address ($Param{UserEmail}) not valid ("
                . $CheckItemObject->CheckError() . ")!",
        );
        return;
    }

    # quote values
    my %Value;
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        if ( $Entry->[5] =~ /^int$/i ) {
            if ( $Param{ $Entry->[0] } ) {
                $Value{ $Entry->[0] } = $Param{ $Entry->[0] };
            }
            else {
                $Value{ $Entry->[0] } = 0;
            }
        }
        else {
            if ( $Param{ $Entry->[0] } ) {
                $Value{ $Entry->[0] } = $Param{ $Entry->[0] };
            }
            else {
                $Value{ $Entry->[0] } = '';
            }
        }
    }

    # build insert
    my $SQL = "INSERT INTO $Self->{CustomerTable} (";
    my @Bind;
    my %SeenKey;    # If the map contains duplicated field names, insert only once.
    my @ColumnNames;

    MAPENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        next MAPENTRY if $Entry->[5] eq 'dynamic_field';            # skip dynamic fields
        next MAPENTRY if ( lc( $Entry->[0] ) eq "userpassword" );
        next MAPENTRY if $SeenKey{ $Entry->[2] }++;
        push @ColumnNames, $Entry->[2];
    }

    $SQL .= join ', ', @ColumnNames;

    if ( !$Self->{ForeignDB} ) {
        $SQL .= ', create_time, create_by, change_time, change_by';
    }

    $SQL .= ') VALUES (';

    my %SeenValue;
    my $BindColumns = 0;

    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        next ENTRY if $Entry->[5] eq 'dynamic_field';            # skip dynamic fields
        next ENTRY if ( lc( $Entry->[0] ) eq "userpassword" );
        next ENTRY if $SeenValue{ $Entry->[2] }++;
        $BindColumns++;
        push @Bind, \$Value{ $Entry->[0] };
    }

    $SQL .= join ', ', ('?') x $BindColumns;

    if ( !$Self->{ForeignDB} ) {
        $SQL .= ', current_timestamp, ?, current_timestamp, ?';
        push @Bind, \$Param{UserID};
        push @Bind, \$Param{UserID};
    }

    $SQL .= ')';

    return if !$Self->{DBObject}->Do(
        SQL  => $SQL,
        Bind => \@Bind
    );

    # log notice
    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'info',
        Message  => "CustomerUser: '$Param{UserLogin}' created successfully ($Param{UserID})!",
    );

    # set password
    if ( $Param{UserPassword} ) {
        $Self->SetPassword(
            UserLogin => $Param{UserLogin},
            PW        => $Param{UserPassword}
        );
    }

    $Self->_CustomerUserCacheClear( UserLogin => $Param{UserLogin} );

    return $Param{UserLogin};
}

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

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!',
        );
        return;
    }

    # check needed stuff
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        if (
            !$Param{ $Entry->[0] }
            && $Entry->[5] ne 'dynamic_field'    # ignore dynamic fields here
            && $Entry->[4]                       # only check required fields
            && $Entry->[0] ne 'UserPassword'     # ignore UserPassword field
            )
        {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Entry->[0]!",
            );
            return;
        }
    }

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    # check email address
    if (
        $Param{UserEmail}
        && !$CheckItemObject->CheckEmail( Address => $Param{UserEmail} )
        && grep { $_ eq $Param{ValidID} } $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet()
        )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Email address ($Param{UserEmail}) not valid ("
                . $CheckItemObject->CheckError() . ")!",
        );
        return;
    }

    # get old user data (pw)
    my %UserData = $Self->CustomerUserDataGet( User => $Param{ID} );

    # if we update the email address, check if it already exists
    if (
        $Param{UserEmail}
        && $Self->{CustomerUserMap}->{CustomerUserEmailUniqCheck}
        && lc $Param{UserEmail} ne lc $UserData{UserEmail}
        )
    {
        my %Result = $Self->CustomerSearch(
            Valid            => 0,
            PostMasterSearch => $Param{UserEmail},
        );
        if (%Result) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => $Kernel::OM->Get('Kernel::Language')->Translate('This email address is already in use for another customer user.'),
            );
            return;
        }
    }

    # quote values
    my %Value;
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        if ( $Entry->[5] =~ /^int$/i ) {
            if ( $Param{ $Entry->[0] } ) {
                $Value{ $Entry->[0] } = $Param{ $Entry->[0] };
            }
            else {
                $Value{ $Entry->[0] } = 0;
            }
        }
        else {
            if ( $Param{ $Entry->[0] } ) {
                $Value{ $Entry->[0] } = $Param{ $Entry->[0] };
            }
            else {
                $Value{ $Entry->[0] } = "";
            }
        }
    }

    # update db
    my $SQL = "UPDATE $Self->{CustomerTable} SET ";
    my @Bind;

    my %SeenKey;    # If the map contains duplicated field names, insert only once.
    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        next ENTRY if $Entry->[5] eq 'dynamic_field';            # skip dynamic fields
        next ENTRY if $Entry->[7];                               # skip readonly fields
        next ENTRY if ( lc( $Entry->[0] ) eq "userpassword" );
        next ENTRY if $SeenKey{ $Entry->[2] }++;
        $SQL .= " $Entry->[2] = ?, ";
        push @Bind, \$Value{ $Entry->[0] };
    }

    if ( !$Self->{ForeignDB} ) {
        $SQL .= 'change_time = current_timestamp, change_by = ?';
        push @Bind, \$Param{UserID};
    }
    else {
        chop $SQL;
        chop $SQL;
    }

    $SQL .= ' WHERE ';

    if ( $Self->{CaseSensitive} ) {
        $SQL .= "$Self->{CustomerKey} = ?";
    }
    else {
        $SQL .= "LOWER($Self->{CustomerKey}) = LOWER(?)";
    }
    push @Bind, \$Param{ID};

    return if !$Self->{DBObject}->Do(
        SQL  => $SQL,
        Bind => \@Bind
    );

    # check if we need to update Customer Preferences
    if ( $Param{UserLogin} ne $UserData{UserLogin} ) {

        # update the preferences
        $Self->{PreferencesObject}->RenamePreferences(
            NewUserID => $Param{UserLogin},
            OldUserID => $UserData{UserLogin},
        );
    }

    # log notice
    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'info',
        Message  => "CustomerUser: '$Param{UserLogin}' updated successfully ($Param{UserID})!",
    );

    # check pw
    if ( $Param{UserPassword} ) {
        $Self->SetPassword(
            UserLogin => $Param{UserLogin},
            PW        => $Param{UserPassword}
        );
    }

    $Self->_CustomerUserCacheClear( UserLogin => $Param{UserLogin} );
    if ( $Param{UserLogin} ne $UserData{UserLogin} ) {
        $Self->_CustomerUserCacheClear( UserLogin => $UserData{UserLogin} );
    }

    return 1;
}

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

    # This method is similar to Kernel::System::User::SetPassword()

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!',
        );

        return;
    }

    my $Login = $Param{UserLogin};
    my $Pw    = $Param{PW} || '';

    # check needed stuff
    if ( !$Login ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need UserLogin!',
        );

        return;
    }

    # TODO: add check whether the CustomerUser exists

    my $CryptedPw = '';

    my $ConfigObject  = $Kernel::OM->Get('Kernel::Config');
    my $ConfigSection = 'Customer::AuthModule::DB';

    my $CryptType = $ConfigObject->Get("${ConfigSection}::CryptType") || 'sha2';

    my $EncodeObject = $Kernel::OM->Get('Kernel::System::Encode');

    # crypt plain (no crypt at all)
    if ( $CryptType eq 'plain' ) {
        $CryptedPw = $Pw;
    }

    # crypt with UNIX crypt
    elsif ( $CryptType eq 'crypt' ) {

        # encode output, needed by crypt() only non utf8 signs
        $EncodeObject->EncodeOutput( \$Pw );
        $EncodeObject->EncodeOutput( \$Login );

        $CryptedPw = crypt( $Pw, $Login );
        $EncodeObject->EncodeInput( \$CryptedPw );
    }

    # crypt with unix_md5_crypt
    elsif ( $CryptType eq 'md5' || !$CryptType ) {

        # encode output, needed by unix_md5_crypt() only non utf8 signs
        $EncodeObject->EncodeOutput( \$Pw );
        $EncodeObject->EncodeOutput( \$Login );

        $CryptedPw = unix_md5_crypt( $Pw, $Login );
        $EncodeObject->EncodeInput( \$CryptedPw );
    }

    # crypt with md5 (compatible with Apache's .htpasswd files)
    elsif ( $CryptType eq 'apr1' ) {

        # encode output, needed by apache_md5_crypt() only non utf8 signs
        $EncodeObject->EncodeOutput( \$Pw );
        $EncodeObject->EncodeOutput( \$Login );

        $CryptedPw = apache_md5_crypt( $Pw, $Login );
        $EncodeObject->EncodeInput( \$CryptedPw );
    }

    # crypt with sha1
    elsif ( $CryptType eq 'sha1' ) {

        my $SHAObject = Digest::SHA->new('sha1');
        $EncodeObject->EncodeOutput( \$Pw );
        $SHAObject->add($Pw);
        $CryptedPw = $SHAObject->hexdigest();
    }

    # crypt with sha512
    elsif ( $CryptType eq 'sha512' ) {

        my $SHAObject = Digest::SHA->new('sha512');
        $EncodeObject->EncodeOutput( \$Pw );
        $SHAObject->add($Pw);
        $CryptedPw = $SHAObject->hexdigest();
    }

    # bcrypt
    elsif ( $CryptType eq 'bcrypt' ) {

        my $MainObject = $Kernel::OM->Get('Kernel::System::Main');

        if ( !$MainObject->Require('Crypt::Eksblowfish::Bcrypt') ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "CustomerUser: '$Login' tried to store password with bcrypt but 'Crypt::Eksblowfish::Bcrypt' is not installed!",
            );
            return;
        }

        my $Cost = $ConfigObject->Get("${ConfigSection}::bcryptCost") // 12;

        # Don't allow values smaller than 9 for security.
        $Cost = 9 if $Cost < 9;

        # Current Crypt::Eksblowfish::Bcrypt limit is 31.
        $Cost = 31 if $Cost > 31;

        my $Salt = $MainObject->GenerateRandomString( Length => 16 );

        # remove UTF8 flag, required by Crypt::Eksblowfish::Bcrypt
        $EncodeObject->EncodeOutput( \$Pw );

        # calculate password hash
        my $Octets = Crypt::Eksblowfish::Bcrypt::bcrypt_hash(
            {
                key_nul => 1,
                cost    => $Cost,
                salt    => $Salt,
            },
            $Pw
        );

        # We will store cost and salt in the password string so that it can be decoded
        #   in future even if we use a higher cost by default.
        $CryptedPw = "BCRYPT:$Cost:$Salt:" . Crypt::Eksblowfish::Bcrypt::en_base64($Octets);
    }

    # crypt with sha256 as fallback
    else {

        my $SHAObject = Digest::SHA->new('sha256');

        # encode output, needed by sha256_hex() only non utf8 signs
        $EncodeObject->EncodeOutput( \$Pw );

        $SHAObject->add($Pw);
        $CryptedPw = $SHAObject->hexdigest();
    }

    # update db
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        if ( $Entry->[0] =~ /^UserPassword$/i ) {
            $Param{PasswordCol} = $Entry->[2];
        }
        if ( $Entry->[0] =~ /^UserLogin$/i ) {
            $Param{LoginCol} = $Entry->[2];
        }
    }

    # check if needed pw col. exists (else there is no pw col.)
    if ( $Param{PasswordCol} && $Param{LoginCol} ) {
        my $SQL = "UPDATE $Self->{CustomerTable} SET "
            . " $Param{PasswordCol} = ? "
            . " WHERE ";

        if ( $Self->{CaseSensitive} ) {
            $SQL .= "$Param{LoginCol} = ?";
        }
        else {
            $SQL .= "LOWER($Param{LoginCol}) = LOWER(?)";
        }

        return if !$Self->{DBObject}->Do(
            SQL  => $SQL,
            Bind => [ \$CryptedPw, \$Param{UserLogin} ],
        );

        # log notice
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'notice',
            Message  => "CustomerUser: '$Param{UserLogin}' changed password successfully!",
        );

        $Self->_CustomerUserCacheClear( UserLogin => $Param{UserLogin} );

        return 1;
    }

    # need no pw to set
    return 1;
}

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

    # generated passwords are eight characters long by default
    my $Size = $Param{Size} || 8;

    my $Password = $Kernel::OM->Get('Kernel::System::Main')->GenerateRandomString(
        Length => $Size,
    );

    return $Password;
}

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

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

    $Self->_CustomerUserCacheClear( UserLogin => $Param{UserID} );

    return $Self->{PreferencesObject}->SetPreferences(%Param);
}

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

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

    return $Self->{PreferencesObject}->GetPreferences(%Param);
}

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

    return $Self->{PreferencesObject}->SearchPreferences(%Param);
}

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

    return if !$Self->{CacheObject};

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

    $Self->{CacheObject}->Delete(
        Type => $Self->{CacheType},
        Key  => "CustomerUserDataGet::$Param{UserLogin}",
    );
    $Self->{CacheObject}->Delete(
        Type => $Self->{CacheType},
        Key  => "CustomerName::$Param{UserLogin}",
    );
    $Self->{CacheObject}->Delete(
        Type => $Self->{CacheType},
        Key  => "CustomerIDs::$Param{UserLogin}",
    );

    # delete all search cache entries
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerIDList',
    );
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerSearch',
    );
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerSearchDetail',
    );
    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
    );

    $Self->{CacheObject}->CleanUp(
        Type => 'CustomerGroup',
    );

    for my $Function (qw(CustomerUserList)) {
        for my $Valid ( 0 .. 1 ) {
            $Self->{CacheObject}->Delete(
                Type => $Self->{CacheType},
                Key  => "${Function}::${Valid}",
            );
        }
    }

    return 1;
}

sub DESTROY {
    my $Self = shift;

    # disconnect if it's not a parent DBObject
    if ( $Self->{NotParentDBObject} ) {
        if ( $Self->{DBObject} ) {
            $Self->{DBObject}->Disconnect();
        }
    }

    return 1;
}

1;
</File>
        <File Location="Custom/Kernel/System/CustomerUser/LDAP.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/System/CustomerUser/LDAP.pm
# --
# 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::CustomerUser::LDAP;

use strict;
use warnings;

use Net::LDAP;
use Net::LDAP::Util qw(escape_filter_value);

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Cache',
    'Kernel::System::DateTime',
    'Kernel::System::DB',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::Encode',
# Rother OSS / CustomerMultitenancy
    'Kernel::System::Group',
# EO Rother OSS
    'Kernel::System::Log',
    'Kernel::System::Main',
);

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

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

    # check needed data
    for my $Needed (qw( PreferencesObject CustomerUserMap )) {
        $Self->{$Needed} = $Param{$Needed} || die "Got no $Needed!";
    }

    # max shown user a search list
    $Self->{UserSearchListLimit} = $Self->{CustomerUserMap}->{CustomerUserSearchListLimit} || 200;

    # get ldap preferences
    $Self->{Die} = 0;
    if ( defined $Self->{CustomerUserMap}->{Params}->{Die} ) {
        $Self->{Die} = $Self->{CustomerUserMap}->{Params}->{Die};
    }

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # params
    if ( $Self->{CustomerUserMap}->{Params}->{Params} ) {
        $Self->{Params} = $Self->{CustomerUserMap}->{Params}->{Params};
    }

    # Net::LDAP new params
    elsif ( $ConfigObject->Get( 'AuthModule::LDAP::Params' . $Param{Count} ) ) {
        $Self->{Params} = $ConfigObject->Get( 'AuthModule::LDAP::Params' . $Param{Count} );
    }
    else {
        $Self->{Params} = {};
    }

    $Self->{StartTLS} = $ConfigObject->Get( 'AuthModule::LDAP::StartTLS' . $Param{Count} ) || '';

    # host
    if ( $Self->{CustomerUserMap}->{Params}->{Host} ) {
        $Self->{Host} = $Self->{CustomerUserMap}->{Params}->{Host};
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need CustomerUser->Params->Host in Kernel/Config.pm',
        );
        return;
    }

    # base dn
    if ( defined $Self->{CustomerUserMap}->{Params}->{BaseDN} ) {
        $Self->{BaseDN} = $Self->{CustomerUserMap}->{Params}->{BaseDN};
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need CustomerUser->Params->BaseDN in Kernel/Config.pm',
        );
        return;
    }

    # scope
    if ( $Self->{CustomerUserMap}->{Params}->{SSCOPE} ) {
        $Self->{SScope} = $Self->{CustomerUserMap}->{Params}->{SSCOPE};
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need CustomerUser->Params->SSCOPE in Kernel/Config.pm',
        );
        return;
    }

    # search user
    $Self->{SearchUserDN} = $Self->{CustomerUserMap}->{Params}->{UserDN} || '';
    $Self->{SearchUserPw} = $Self->{CustomerUserMap}->{Params}->{UserPw} || '';

    # group dn
    $Self->{GroupDN} = $Self->{CustomerUserMap}->{Params}->{GroupDN} || '';

    # customer key
    if ( $Self->{CustomerUserMap}->{CustomerKey} ) {
        $Self->{CustomerKey} = $Self->{CustomerUserMap}->{CustomerKey};
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need CustomerUser->CustomerKey in Kernel/Config.pm',
        );
        return;
    }

    # customer id
    if ( $Self->{CustomerUserMap}->{CustomerID} ) {
        $Self->{CustomerID} = $Self->{CustomerUserMap}->{CustomerID};
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need CustomerUser->CustomerID in Kernel/Config.pm',
        );
        return;
    }

    # ldap filter always used
    $Self->{AlwaysFilter} = $Self->{CustomerUserMap}->{Params}->{AlwaysFilter} || '';

    $Self->{ExcludePrimaryCustomerID} = $Self->{CustomerUserMap}->{CustomerUserExcludePrimaryCustomerID} || 0;
    $Self->{SearchPrefix}             = $Self->{CustomerUserMap}->{CustomerUserSearchPrefix};
    if ( !defined $Self->{SearchPrefix} ) {
        $Self->{SearchPrefix} = '';
    }
    $Self->{SearchSuffix} = $Self->{CustomerUserMap}->{CustomerUserSearchSuffix};
    if ( !defined $Self->{SearchSuffix} ) {
        $Self->{SearchSuffix} = '*';
    }

    # charset settings
    $Self->{SourceCharset} = $Self->{CustomerUserMap}->{Params}->{SourceCharset} || '';

    # set cache type
    $Self->{CacheType} = 'CustomerUser' . $Param{Count};

    # create cache object, but only if CacheTTL is set in customer config
    if ( $Self->{CustomerUserMap}->{CacheTTL} ) {
        $Self->{CacheObject} = $Kernel::OM->Get('Kernel::System::Cache');
    }

    # get valid filter if used
    $Self->{ValidFilter} = $Self->{CustomerUserMap}->{CustomerUserValidFilter} || '';

    # connect first if Die is enabled, make sure that connection is possible, else die
    if ( $Self->{Die} ) {
        return if !$Self->_Connect();
    }

    # fetch names of configured dynamic fields
    my @DynamicFieldMapEntries = grep { $_->[5] eq 'dynamic_field' } @{ $Self->{CustomerUserMap}->{Map} };
    $Self->{ConfiguredDynamicFieldNames} = { map { $_->[2] => 1 } @DynamicFieldMapEntries };

# Rother OSS / CustomerMultitenancy
    my $LayoutParam = $Kernel::OM->{Param}->{'Kernel::Output::HTML::Layout'};

    # Check if multitenancy is enabled and the request is coming from a user.
    if ( $LayoutParam->{UserType} && $LayoutParam->{UserType} eq 'User' && $ConfigObject->Get('Multitenancy') ) {

        # Save the UserID and all groups the user has 'ro' permission on.
        if ( $LayoutParam->{UserID} ) {
            # Only get the group list once for performance reasons.
            my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
                UserID => $LayoutParam->{UserID},
                Type   => 'ro',
            );

            # The limit does not count for members of a specific group.
            my $PermissionGroup = $ConfigObject->Get('Multitenancy::PermissionGroup') || '';
            my %GroupsReverse   = reverse %Groups;

            if ( !$GroupsReverse{$PermissionGroup} ) {
                $Self->{Multitenancy} = $LayoutParam->{UserID};
                $Self->{UserGroupIDs} = [ sort keys %Groups ];
                $Self->{UserGroups}   = [ values %Groups ];
            }
        }
    }
# EO CustomerMultitenancy

    return $Self;
}

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

    # return if connection is already open
    return 1 if $Self->{LDAP};

    # ldap connect and bind (maybe with SearchUserDN and SearchUserPw)
    $Self->{LDAP} = Net::LDAP->new( $Self->{Host}, %{ $Self->{Params} } );

    if ( !$Self->{LDAP} ) {
        if ( $Self->{Die} ) {
            die "Can't connect to $Self->{Host}: $@";
        }
        else {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Can't connect to $Self->{Host}: $@",
            );
            return;
        }
    }
    if ( $Self->{StartTLS} ) {
        my $Started = $Self->{LDAP}->start_tls( verify => $Self->{StartTLS} );
        if ( !$Started ) {
            if ( $Self->{Die} ) {
                die "start_tls on $Self->{Host} failed: $@";
            }
            else {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "start_tls: '$Self->{StartTLS}' on $Self->{Host} failed: $@",
                );
                $Self->{LDAP}->disconnect();
                return;
            }
        }
    }

    my $Result;
    if ( $Self->{SearchUserDN} && $Self->{SearchUserPw} ) {
        $Result = $Self->{LDAP}->bind(
            dn       => $Self->{SearchUserDN},
            password => $Self->{SearchUserPw},
        );
    }
    else {
        $Result = $Self->{LDAP}->bind();
    }

    if ( $Result->code() ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'First bind failed! ' . $Result->error(),
        );
        $Self->{LDAP}->disconnect();
        return;
    }

    return 1;
}

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

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

    # build filter
    my $Filter = "($Self->{CustomerKey}=" . escape_filter_value( $Param{UserLogin} ) . ')';

    # prepare filter
    if ( $Self->{AlwaysFilter} ) {
        $Filter = "(&$Filter$Self->{AlwaysFilter})";
    }

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Name = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => 'CustomerName::' . $Param{UserLogin},
        );
        return $Name if defined $Name;
    }

    # create ldap connect
    return if !$Self->_Connect();

    # perform user search
    my $Result = $Self->{LDAP}->search(
        base      => $Self->{BaseDN},
        scope     => $Self->{SScope},
        filter    => $Filter,
        sizelimit => $Self->{UserSearchListLimit},
        attrs     => $Self->{CustomerUserMap}->{CustomerUserNameFields},
    );

    if ( $Result->code() ) {
        if ( $Result->code() == 4 ) {

            # Result code 4 (LDAP_SIZELIMIT_EXCEEDED) is normal if there
            # are more items in LDAP than search limit defined in OTOBO or
            # in LDAP server. Avoid spamming logs with such errors.
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'debug',
                Message  => 'LDAP size limit exceeded (' . $Result->error() . ').',
            );
        }
        else {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Search failed! ' . $Result->error(),
            );
        }
        return;
    }

    my %NameParts;

    for my $Entry ( $Result->all_entries() ) {

        for my $Field ( @{ $Self->{CustomerUserMap}->{CustomerUserNameFields} } ) {

            if ( defined $Entry->get_value($Field) ) {
                $NameParts{$Field} = $Self->_ConvertFrom( $Entry->get_value($Field) );
            }
        }
    }

    # fetch dynamic field values, if configured
    my @DynamicFieldCustomerUserNameFields = grep { exists $Self->{ConfiguredDynamicFieldNames}->{$_} }
        @{ $Self->{CustomerUserMap}->{CustomerUserNameFields} };
    if (@DynamicFieldCustomerUserNameFields) {
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        DYNAMICFIELDNAME:
        for my $DynamicFieldName (@DynamicFieldCustomerUserNameFields) {
            my $DynamicFieldConfig = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
                Name => $DynamicFieldName,
            );
            next DYNAMICFIELDNAME if !IsHashRefWithData($DynamicFieldConfig);

            my $Value = $DynamicFieldBackendObject->ValueGet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectName         => $Param{UserLogin},
            );

            next DYNAMICFIELDNAME if !defined $Value;

            if ( !IsArrayRefWithData($Value) ) {
                $Value = [$Value];
            }

            my @RenderedValues;

            VALUE:
            for my $CurrentValue ( @{$Value} ) {
                next VALUE if !defined $CurrentValue || !length $CurrentValue;

                my $RenderedValue = $DynamicFieldBackendObject->ReadableValueRender(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    Value              => $CurrentValue,
                );

                next VALUE if !IsHashRefWithData($RenderedValue) || !defined $RenderedValue->{Value};

                push @RenderedValues, $RenderedValue->{Value};
            }

            $NameParts{$DynamicFieldName} = join ' ', @RenderedValues;
        }
    }

    # assemble name
    my @NameParts;
    CUSTOMERUSERNAMEFIELD:
    for my $CustomerUserNameField ( @{ $Self->{CustomerUserMap}->{CustomerUserNameFields} } ) {
        next CUSTOMERUSERNAMEFIELD
            if !exists $NameParts{$CustomerUserNameField}
            || !defined $NameParts{$CustomerUserNameField}
            || !length $NameParts{$CustomerUserNameField};
        push @NameParts, $NameParts{$CustomerUserNameField};
    }

    my $JoinCharacter = $Self->{CustomerUserMap}->{CustomerUserNameFieldsJoin} // ' ';
    my $Name          = join $JoinCharacter, @NameParts;

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => 'CustomerName::' . $Param{UserLogin},
            Value => $Name,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return $Name;
}

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

    if ( $Param{CustomerIDRaw} ) {
        $Param{CustomerID} = $Param{CustomerIDRaw};
    }

    # check needed stuff
    if ( !$Param{Search} && !$Param{UserLogin} && !$Param{PostMasterSearch} && !$Param{CustomerID} )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need Search, UserLogin, PostMasterSearch or CustomerID!'
        );
        return;
    }

    # build filter
    my $Filter = '';
    if ( $Param{Search} ) {

        my $Count = 0;
        my @Parts = split( /\+/, $Param{Search}, 6 );
        for my $Part (@Parts) {

            $Part = $Self->{SearchPrefix} . $Part . $Self->{SearchSuffix};
            $Part =~ s/(\%+)/\%/g;
            $Part =~ s/(\*+)\*/*/g;
            $Count++;

            # remove dynamic field names that are configured in CustomerUserSearchFields
            # as they cannot be retrieved here
            my @CustomerUserSearchFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} }
                @{ $Self->{CustomerUserMap}->{CustomerUserSearchFields} };

            if (@CustomerUserSearchFields) {

                # quote LDAP filter value but keep asterisks unescaped (wildcard)
                $Part =~ s/\*/encodedasterisk20160930/g;
                $Part = escape_filter_value( $Self->_ConvertTo($Part) );
                $Part =~ s/encodedasterisk20160930/*/g;

                $Filter .= '(|';
                for my $Field (@CustomerUserSearchFields) {
                    $Filter .= "($Field=" . $Part . ')';
                }
                $Filter .= ')';
            }
            else {

                # quote LDAP filter value but keep asterisks unescaped (wildcard)
                $Part =~ s/\*/encodedasterisk20160930/g;
                $Part = escape_filter_value($Part);
                $Part =~ s/encodedasterisk20160930/*/g;

                $Filter .= "($Self->{CustomerKey}=" . $Part . ')';
            }
        }

        if ( $Count > 1 ) {
            $Filter = "(&$Filter)";
        }
    }
    elsif ( $Param{PostMasterSearch} ) {

        # remove dynamic field names that are configured in CustomerUserPostMasterSearchFields
        # as they cannot be retrieved here
        my @CustomerUserPostMasterSearchFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} }
            @{ $Self->{CustomerUserMap}->{CustomerUserPostMasterSearchFields} };

        if (@CustomerUserPostMasterSearchFields) {

            # quote LDAP filter value but keep asterisks unescaped (wildcard)
            $Param{PostMasterSearch} =~ s/\*/encodedasterisk20160930/g;
            $Param{PostMasterSearch} = escape_filter_value( $Param{PostMasterSearch} );
            $Param{PostMasterSearch} =~ s/encodedasterisk20160930/*/g;

            $Filter = '(|';
            for my $Field (@CustomerUserPostMasterSearchFields) {
                $Filter .= "($Field=$Param{PostMasterSearch})";
            }
            $Filter .= ')';
        }
    }
    elsif ( $Param{UserLogin} ) {
        $Filter = "($Self->{CustomerKey}=" . escape_filter_value( $Param{UserLogin} ) . ')';
    }
    elsif ( $Param{CustomerID} ) {
        $Filter = "($Self->{CustomerID}=" . escape_filter_value( $Param{CustomerID} ) . ')';
    }

    # prepare filter
    if ( $Self->{AlwaysFilter} ) {
        $Filter = "(&$Filter$Self->{AlwaysFilter})";
    }

    # add valid filter
    if ( $Self->{ValidFilter} ) {
        $Filter = "(&$Filter$Self->{ValidFilter})";
    }

    # check cache
    my $CacheKey = join '::', map { $_ . '=' . $Param{$_} } sort keys %Param;
# Rother OSS / CustomerMultitenancy
    # Use cache for multitenancy.
    if ( $Self->{Multitenancy} ) {
        $CacheKey .= join '', map { '::GroupID=' . $_ } @{ $Self->{UserGroupIDs} };
    }
# EO CustomerMultitenancy
    if ( $Self->{CacheObject} ) {
        my $Users = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType} . '_CustomerSearch',
            Key  => $CacheKey,
        );
        return %{$Users} if ref $Users eq 'HASH';
    }

    # create ldap connect
    return if !$Self->_Connect();

    my $CustomerUserListFields = $Self->{CustomerUserMap}->{CustomerUserListFields};

    # remove dynamic field names that are configured in CustomerUserListFields
    # as they cannot be handled here
    my @CustomerUserListFieldsWithoutDynamicFields = grep { !exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerUserListFields};

    # combine needed attrs
    my @Attributes = ( @CustomerUserListFieldsWithoutDynamicFields, $Self->{CustomerKey} );

# Rother OSS / CustomerMultitenancy
    # Don't search for customer users without group permission.
    if ( $Self->{Multitenancy} ) {
        my $GroupIDAttr;
        for my $Map ( @{ $Self->{CustomerUserMap}->{Map} } ) {
            if ( $Map->[0] eq 'UserGroupID' ) {
                $GroupIDAttr = $Map->[2];
            }
        }

        # The source supports multitenancy.
        if ($GroupIDAttr) {
            if ( $Self->{CustomerUserMap}->{CustomerCompanySupport} ) {
                # TODO: Workaround until the search gets changed. See CustomerUser/LDAP.pm for more information.
                $Param{Limit} = 100000;
            }

            my $UserGroupIDSync = $Self->{CustomerUserMap}->{UserGroupIDSync};
            # Check if we match for group IDs or group names.
            my @UserGroups = $UserGroupIDSync->{UseGroupNames} ? @{ $Self->{UserGroups} } : @{ $Self->{UserGroupIDs} };

            # Remap local groups to remote groups.
            for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
                my $LocalGroup = $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup};

                @UserGroups = map { $_ eq $LocalGroup ? $RemoteGroup : $_ } @UserGroups;
            }

            # Build filter string. If the field is empty, we have access by default.
            my $AdditionalFilter = "(|(!($GroupIDAttr=*))";
            for my $UserGroup (@UserGroups) {
                $AdditionalFilter .= "($GroupIDAttr=$UserGroup)";
            }
            $AdditionalFilter .= '))';

            # Remove the last closing parenthesis and add the custom filter to the search.
            $Filter = substr $Filter, 0, -1;
            $Filter .= $AdditionalFilter;
        }
    }
# EO CustomerMultitenancy

    # perform user search
    my $Result = $Self->{LDAP}->search(
        base      => $Self->{BaseDN},
        scope     => $Self->{SScope},
        filter    => $Filter,
        sizelimit => $Param{Limit} || $Self->{UserSearchListLimit},
        attrs     => \@Attributes,
    );

    # log ldap errors
    if ( $Result->code() ) {
        if ( $Result->code() == 4 ) {

            # Result code 4 (LDAP_SIZELIMIT_EXCEEDED) is normal if there
            # are more items in LDAP than search limit defined in OTOBO or
            # in LDAP server. Avoid spamming logs with such errors.
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'debug',
                Message  => 'LDAP size limit exceeded (' . $Result->error() . ').',
            );
        }
        else {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Search failed! ' . $Result->error(),
            );
        }
    }

    # dynamic field handling
    my @CustomerUserListFieldsDynamicFields = grep { exists $Self->{ConfiguredDynamicFieldNames}->{$_} } @{$CustomerUserListFields};
    my %CustomerUserListFieldsDynamicFields = map  { $_ => 1 } @CustomerUserListFieldsDynamicFields;

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

    my $DynamicFieldConfigs = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
        Valid      => 1,
    );
    my %DynamicFieldConfigsByName = map { $_->{Name} => $_ } @{$DynamicFieldConfigs};

    my %Users;
    for my $Entry ( $Result->all_entries() ) {

        my $CustomerString = '';

        my $CustomerKey;
        if ( defined $Entry->get_value( $Self->{CustomerKey} ) ) {
            $CustomerKey = $Self->_ConvertFrom( $Entry->get_value( $Self->{CustomerKey} ) );
        }

        FIELD:
        for my $Field ( @{$CustomerUserListFields} ) {

            # dynamic field value
            if ( $CustomerUserListFieldsDynamicFields{$Field} ) {
                next FIELD if !defined $CustomerKey;
                next FIELD if !exists $DynamicFieldConfigsByName{$Field};

                my $Value = $DynamicFieldBackendObject->ValueGet(
                    DynamicFieldConfig => $DynamicFieldConfigsByName{$Field},
                    ObjectName         => $CustomerKey,
                );

                next FIELD if !defined $Value;

                if ( !IsArrayRefWithData($Value) ) {
                    $Value = [$Value];
                }

                my @Values;

                VALUE:
                for my $CurrentValue ( @{$Value} ) {
                    next VALUE if !defined $CurrentValue || !length $CurrentValue;

                    my $ReadableValue = $DynamicFieldBackendObject->ReadableValueRender(
                        DynamicFieldConfig => $DynamicFieldConfigsByName{$Field},
                        Value              => $CurrentValue,
                    );

                    next VALUE if !IsHashRefWithData($ReadableValue) || !defined $ReadableValue->{Value};

                    my $IsACLReducible = $DynamicFieldBackendObject->HasBehavior(
                        DynamicFieldConfig => $DynamicFieldConfigsByName{$Field},
                        Behavior           => 'IsACLReducible',
                    );
                    if ($IsACLReducible) {
                        my $PossibleValues = $DynamicFieldBackendObject->PossibleValuesGet(
                            DynamicFieldConfig => $DynamicFieldConfigsByName{$Field},
                        );

                        if (
                            IsHashRefWithData($PossibleValues)
                            && defined $PossibleValues->{ $ReadableValue->{Value} }
                            )
                        {
                            $ReadableValue->{Value} = $PossibleValues->{ $ReadableValue->{Value} };
                        }
                    }

                    push @Values, $ReadableValue->{Value};
                }

                $CustomerString .= ( join ' ', @Values ) . ' ';

                next FIELD;
            }

            my $Value = $Self->_ConvertFrom( $Entry->get_value($Field) );

            if ($Value) {
                if ( $Field =~ /^targetaddress$/i ) {
                    $Value =~ s/SMTP:(.*)/$1/;
                }
                $CustomerString .= $Value . ' ';
            }
        }

        $CustomerString =~ s/^(.*)\s(.+?\@.+?\..+?)(\s|)$/"$1" <$2>/;

        if ( defined $CustomerKey ) {
            $Users{$CustomerKey} = $CustomerString;
        }
    }

    # check if user need to be in a group!
    if ( $Self->{GroupDN} ) {

        for my $Filter2 ( sort keys %Users ) {

            my $Result2 = $Self->{LDAP}->search(
                base      => $Self->{GroupDN},
                scope     => $Self->{SScope},
                filter    => 'memberUid=' . escape_filter_value($Filter2),
                sizelimit => $Param{Limit} || $Self->{UserSearchListLimit},
                attrs     => ['1.1'],
            );

            if ( !$Result2->all_entries() ) {
                delete $Users{$Filter2};
            }
        }
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType} . '_CustomerSearch',
            Key   => $CacheKey,
            Value => \%Users,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return %Users;
}

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

    if ( ref $Param{SearchFields} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "SearchFields must be an array reference!",
        );
        return;
    }

    # Return only valid users per default
    my $Valid = $Param{Valid} // 1;

    $Param{Limit} //= '';

    # Split the search fields in scalar and array fields, before the diffrent handling.
    my @ScalarSearchFields = grep { 'Input' eq $_->{Type} } @{ $Param{SearchFields} };
    my @ArraySearchFields  = grep { 'Selection' eq $_->{Type} } @{ $Param{SearchFields} };

    # Verify that all passed array parameters contain an arrayref.
    ARGUMENT:
    for my $Argument (@ArraySearchFields) {
        if ( !defined $Param{ $Argument->{Name} } ) {
            $Param{ $Argument->{Name} } ||= [];

            next ARGUMENT;
        }

        if ( ref $Param{ $Argument->{Name} } ne 'ARRAY' ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "$Argument->{Name} must be an array reference!",
            );
            return;
        }
    }

    # Set the default behaviour for the return type.
    my $Result = $Param{Result} || 'ARRAY';

    # Special handling if the result type is 'COUNT'.
    if ( $Result eq 'COUNT' ) {

        # Ignore the parameter 'Limit' when result type is 'COUNT'.
        $Param{Limit} = '';

        # Delete the OrderBy parameter when the result type is 'COUNT'.
        $Param{OrderBy} = [];
    }

    # Define order table from the search fields.
    my %OrderByTable = map { $_->{Name} => $_->{DatabaseField} } @{ $Param{SearchFields} };

    for my $Field (@ArraySearchFields) {

        my $SelectionsData = $Field->{SelectionsData};

        for my $SelectedValue ( @{ $Param{ $Field->{Name} } } ) {

            # Check if the selected value for the current field is valid.
            if ( !$SelectionsData->{$SelectedValue} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "The selected value $Field->{Name} is not valid!",
                );
                return;
            }
        }
    }

    # Build the ldap filter for the diffrent search params.
    my $Filter = '';

    for my $Field (@ScalarSearchFields) {

        # Search for scalar fields (wildcards are allowed).
        if ( $Param{ $Field->{Name} } ) {

            $Param{ $Field->{Name} } =~ s/(\%+)/\%/g;
            $Param{ $Field->{Name} } =~ s/(\*+)\*/*/g;

            $Filter .= "($Field->{DatabaseField}=" . $Self->_ConvertTo( $Param{ $Field->{Name} } ) . ")";
        }
    }

    if ($Filter) {
        $Filter = "(&$Filter)";
    }

    my $ArrayFilter = '';

    # Special parameter for CustomerIDs from a customer company search result.
    if ( IsArrayRefWithData( $Param{CustomerCompanySearchCustomerIDs} ) ) {
        $ArrayFilter .= '(|';
        for my $OneParam ( @{ $Param{CustomerCompanySearchCustomerIDs} } ) {
            $ArrayFilter .= "($Self->{CustomerID}=" . $Self->_ConvertTo($OneParam) . ")";
        }
        $ArrayFilter .= ')';
    }

    FIELD:
    for my $Field (@ArraySearchFields) {

        # Ignore empty lists.
        next FIELD if !@{ $Param{ $Field->{Name} } };

        $ArrayFilter .= '(|';
        for my $OneParam ( @{ $Param{ $Field->{Name} } } ) {
            $ArrayFilter .= "($Field->{DatabaseField}=" . $Self->_ConvertTo($OneParam) . ")";
        }
        $ArrayFilter .= ')';
    }

    # Add the array filter fields to the ldap filter.
    if ($ArrayFilter) {
        $Filter = "(&$Filter$ArrayFilter)";
    }

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

    # Check all configured change dynamic fields, build lookup hash by name.
    my %CustomerUserDynamicFieldName2Config;
    my $CustomerUserDynamicFields = $DynamicFieldObject->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
    );
    for my $DynamicField ( @{$CustomerUserDynamicFields} ) {
        $CustomerUserDynamicFieldName2Config{ $DynamicField->{Name} } = $DynamicField;
    }

    my $SQLDynamicFieldFrom     = '';
    my $SQLDynamicFieldWhere    = '';
    my $DynamicFieldJoinCounter = 1;

    DYNAMICFIELD:
    for my $DynamicField ( @{$CustomerUserDynamicFields} ) {

        my $SearchParam = $Param{ "DynamicField_" . $DynamicField->{Name} };

        next DYNAMICFIELD if ( !$SearchParam );
        next DYNAMICFIELD if ( ref $SearchParam ne 'HASH' );

        my $NeedJoin;

        for my $Operator ( sort keys %{$SearchParam} ) {

            my @SearchParams = ( ref $SearchParam->{$Operator} eq 'ARRAY' )
                ? @{ $SearchParam->{$Operator} }
                : ( $SearchParam->{$Operator} );

            my $SQLDynamicFieldWhereSub = '';
            if ($SQLDynamicFieldWhere) {
                $SQLDynamicFieldWhereSub = ' AND (';
            }
            else {
                $SQLDynamicFieldWhereSub = ' (';
            }

            my $Counter = 0;
            TEXT:
            for my $Text (@SearchParams) {
                next TEXT if ( !defined $Text || $Text eq '' );

                $Text =~ s/\*/%/gi;

                # Check search attribute, we do not need to search for '*'.
                next TEXT if $Text =~ /^\%{1,3}$/;

                my $ValidateSuccess = $DynamicFieldBackendObject->ValueValidate(
                    DynamicFieldConfig => $DynamicField,
                    Value              => $Text,
                    UserID             => $Param{UserID} || 1,
                );
                if ( !$ValidateSuccess ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Search not executed due to invalid value '"
                            . $Text
                            . "' on field '"
                            . $DynamicField->{Name} . "'!",
                    );
                    return;
                }

                if ($Counter) {
                    $SQLDynamicFieldWhereSub .= ' OR ';
                }
                $SQLDynamicFieldWhereSub .= $DynamicFieldBackendObject->SearchSQLGet(
                    DynamicFieldConfig => $DynamicField,
                    TableAlias         => "dfv$DynamicFieldJoinCounter",
                    Operator           => $Operator,
                    SearchTerm         => $Text,
                );

                $Counter++;
            }
            $SQLDynamicFieldWhereSub .= ') ';

            if ($Counter) {
                $SQLDynamicFieldWhere .= $SQLDynamicFieldWhereSub;
                $NeedJoin = 1;
            }
        }

        if ($NeedJoin) {
            $SQLDynamicFieldFrom .= "
                INNER JOIN dynamic_field_value dfv$DynamicFieldJoinCounter
                    ON (df_obj_id_name.object_id = dfv$DynamicFieldJoinCounter.object_id
                        AND dfv$DynamicFieldJoinCounter.field_id = "
                . $DBObject->Quote( $DynamicField->{ID}, 'Integer' ) . ")
            ";

            $DynamicFieldJoinCounter++;
        }
    }

    # Execute a dynamic field search, if a dynamic field where statement exists.
    if ($SQLDynamicFieldWhere) {

        my @DynamicFieldUserLogins;

        # sql uery for the dynamic fields
        my $SQLDynamicField = "SELECT DISTINCT(df_obj_id_name.object_name) FROM dynamic_field_obj_id_name df_obj_id_name "
            . $SQLDynamicFieldFrom
            . " WHERE "
            . $SQLDynamicFieldWhere;

        my $UsedCache;

        if ( $Self->{CacheObject} ) {

            my $DynamicFieldSearchCacheData = $Self->{CacheObject}->Get(
                Type => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
                Key  => $SQLDynamicField,
            );

            if ( defined $DynamicFieldSearchCacheData ) {
                if ( ref $DynamicFieldSearchCacheData eq 'ARRAY' ) {
                    @DynamicFieldUserLogins = @{$DynamicFieldSearchCacheData};

                    # set the used cache flag
                    $UsedCache = 1;
                }
                else {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => 'Invalid ref ' . ref($DynamicFieldSearchCacheData) . '!'
                    );
                    return;
                }
            }
        }

        # Get the data only from database, if no cache entry exists.
        if ( !$UsedCache ) {

            return if !$DBObject->Prepare(
                SQL => $SQLDynamicField,
            );

            while ( my @Row = $DBObject->FetchrowArray() ) {
                push @DynamicFieldUserLogins, $Row[0];
            }

            if ( $Self->{CacheObject} ) {
                $Self->{CacheObject}->Set(
                    Type  => $Self->{CacheType} . '_CustomerSearchDetailDynamicFields',
                    Key   => $SQLDynamicField,
                    Value => \@DynamicFieldUserLogins,
                    TTL   => $Self->{CustomerUserMap}->{CacheTTL},
                );
            }
        }

        # Add the user logins from the dynamic fields, if a search result exists from the dynamic field search
        #   or skip the search and return a emptry array ref, if no user logins exists from the dynamic field search.
        if (@DynamicFieldUserLogins) {

            my $DynamicFieldUserLoginsFilter = '(|';
            for my $OneParam (@DynamicFieldUserLogins) {
                $DynamicFieldUserLoginsFilter .= "($Self->{CustomerKey}=" . $Self->_ConvertTo($OneParam) . ")";
            }
            $DynamicFieldUserLoginsFilter .= ')';

            # Add the dynamic field user logins filter to the ldap filter.
            $Filter = "(&$Filter$DynamicFieldUserLoginsFilter)";
        }
        else {
            return $Result eq 'COUNT' ? 0 : [];
        }
    }

    # Special parameter to exclude some user logins from the search result.
    if ( IsArrayRefWithData( $Param{ExcludeUserLogins} ) ) {
        my $ExcludeUserLoginsFilter = '(&';
        for my $OneParam ( @{ $Param{ExcludeUserLogins} } ) {
            $ExcludeUserLoginsFilter .= "(!($Self->{CustomerKey}=" . $Self->_ConvertTo($OneParam) . "))";
        }
        $ExcludeUserLoginsFilter .= ')';

        # Add the exclude user logins filter to the ldap filter.
        $Filter = "(&$Filter$ExcludeUserLoginsFilter)";
    }

    if ( $Self->{AlwaysFilter} ) {
        $Filter = "(&$Filter$Self->{AlwaysFilter})";
    }

    if ( $Self->{ValidFilter} && $Valid ) {
        $Filter = "(&$Filter$Self->{ValidFilter})";
    }

    # Default filter for the search, if no filter exists.
    if ( !$Filter ) {
        $Filter = "($Self->{CustomerKey}=*)";
    }

    # Check if OrderBy contains only unique valid values.
    my %OrderBySeen;
    for my $OrderBy ( @{ $Param{OrderBy} } ) {

        if ( !$OrderBy || $OrderBySeen{$OrderBy} ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "OrderBy contains invalid value '$OrderBy' or the value is used more than once!",
            );
            return;
        }

        # Remember the value to check if it appears more than once.
        $OrderBySeen{$OrderBy} = 1;
    }

    # Check if OrderByDirection array contains only 'Up' or 'Down'.
    DIRECTION:
    for my $Direction ( @{ $Param{OrderByDirection} } ) {

        # Only 'Up' or 'Down' allowed.
        next DIRECTION if $Direction eq 'Up';
        next DIRECTION if $Direction eq 'Down';

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "OrderByDirection can only contain 'Up' or 'Down'!",
        );
        return;
    }

    # Assemble the ORDER BY clause.
    my @OrderByFields;
    my $OrderByString = '';
    my $Count         = 0;

    ORDERBY:
    for my $OrderBy ( @{ $Param{OrderBy} } ) {
        next ORDERBY if !$OrderByTable{$OrderBy};

        my $Direction = $Param{OrderByDirection}->[$Count] || 'Down';

        $OrderByString .= $OrderByTable{$OrderBy} . $Direction;

        push @OrderByFields, {
            Name          => $OrderBy,
            DatabaseField => $OrderByTable{$OrderBy},
            Direction     => $Direction,
        };
    }
    continue {
        $Count++;
    }

    # If there is a possibility that the ordering is not determined
    #   we add an descending ordering by id.
    if ( !grep { $_ eq 'UserLogin' } ( @{ $Param{OrderBy} } ) ) {
        push @OrderByFields, {
            Name          => 'UserLogin',
            DatabaseField => "$Self->{CustomerKey}",
            Direction     => 'Down',
        };
    }

    my $CacheKey = 'CustomerSearchDetail::' . $Result . $Filter . $Param{Limit} . $OrderByString;

    if ( $Self->{CacheObject} ) {
        my $CacheData = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );

        if ( defined $CacheData ) {
            if ( ref $CacheData eq 'ARRAY' ) {
                return $CacheData;
            }
            elsif ( ref $CacheData eq '' ) {
                return $CacheData;
            }
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid ref ' . ref($CacheData) . '!'
            );
            return;
        }
    }

    return if !$Self->_Connect();

    # cCmbine needed attributes.
    my @Attributes = map { $_->{DatabaseField} } @OrderByFields;

    # Perform the ldap user search.
    my $ResultSearch = $Self->{LDAP}->search(
        base      => $Self->{BaseDN},
        scope     => $Self->{SScope},
        filter    => $Filter,
        sizelimit => $Self->{UserSearchListLimit},
        attrs     => \@Attributes,
    );

    if ( $ResultSearch->code() ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => $ResultSearch->error(),
        );
    }

    my @TmpCustomerUsers;
    for my $Entry ( $ResultSearch->all_entries() ) {

        my %Data;
        for my $OrderBy (@OrderByFields) {

            my $FieldValue = $Entry->get_value( $OrderBy->{DatabaseField} );
            $FieldValue = $Self->_ConvertFrom($FieldValue);

            $Data{ $OrderBy->{Name} } = $FieldValue;
        }

        push @TmpCustomerUsers, \%Data;
    }

    # Sort the customer users.
    @TmpCustomerUsers = sort { $Self->_SearchResultSort(@OrderByFields) } @TmpCustomerUsers;

    my @IDs;

    # Check if user need to be in a group!
    if ( $Self->{GroupDN} ) {

        FILTERID:
        for my $Data (@TmpCustomerUsers) {

            my $ResultGroupDN = $Self->{LDAP}->search(
                base      => $Self->{GroupDN},
                scope     => $Self->{SScope},
                filter    => 'memberUid=' . $Data->{UserLogin},
                sizelimit => $Self->{UserSearchListLimit},
                attrs     => ['1.1'],
            );

            next FILTERID if !$ResultGroupDN->all_entries();

            push @IDs, $Data->{UserLogin};
        }
    }
    else {
        @IDs = map { $_->{UserLogin} } @TmpCustomerUsers;
    }

    if ( $Param{Limit} ) {
        splice @IDs, $Param{Limit};
    }

    if ( $Result eq 'COUNT' ) {

        if ( $Self->{CacheObject} ) {
            $Self->{CacheObject}->Set(
                Type  => $Self->{CacheType},
                Key   => $CacheKey,
                Value => scalar @IDs,
                TTL   => $Self->{CustomerUserMap}->{CacheTTL},
            );
        }
        return scalar @IDs;
    }
    else {

        if ( $Self->{CacheObject} ) {
            $Self->{CacheObject}->Set(
                Type  => $Self->{CacheType},
                Key   => $CacheKey,
                Value => \@IDs,
                TTL   => $Self->{CustomerUserMap}->{CacheTTL},
            );
        }
        return \@IDs;
    }
}

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

    my $Valid      = defined $Param{Valid} ? $Param{Valid} : 1;
    my $SearchTerm = $Param{SearchTerm} || '';

    my $CacheKey = "CustomerIDList::${Valid}::$SearchTerm";

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Result = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return @{$Result} if ref $Result eq 'ARRAY';
    }

    # prepare filter
    my $Filter = "($Self->{CustomerID}=*)";
    if ($SearchTerm) {

        my $SearchFilter = $Self->{SearchPrefix} . $SearchTerm . $Self->{SearchSuffix};
        $SearchFilter =~ s/(\%+)/\%/g;
        $SearchFilter =~ s/(\*+)\*/*/g;

        # quote LDAP filter value but keep asterisks unescaped (wildcard)
        $SearchFilter =~ s/\*/encodedasterisk20160930/g;
        $SearchFilter = escape_filter_value($SearchFilter);
        $SearchFilter =~ s/encodedasterisk20160930/*/g;

        $Filter = "($Self->{CustomerID}=$SearchFilter)";

    }

    if ( $Self->{AlwaysFilter} ) {
        $Filter = "(&$Filter$Self->{AlwaysFilter})";
    }

    # add valid filter
    if ( $Self->{ValidFilter} && $Valid ) {
        $Filter = "(&$Filter$Self->{ValidFilter})";
    }

    # create ldap connect
    return if !$Self->_Connect();

    # combine needed attrs
    my @Attributes = ( $Self->{CustomerKey}, $Self->{CustomerID} );

    # perform user search
    my $Result = $Self->{LDAP}->search(
        base      => $Self->{BaseDN},
        scope     => $Self->{SScope},
        filter    => $Filter,
        sizelimit => $Self->{UserSearchListLimit},
        attrs     => \@Attributes,
    );

    # log ldap errors
    if ( $Result->code() ) {

        if ( $Result->code() == 4 ) {

            # Result code 4 (LDAP_SIZELIMIT_EXCEEDED) is normal if there
            # are more items in LDAP than search limit defined in OTOBO or
            # in LDAP server. Avoid spamming logs with such errors.
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'debug',
                Message  => 'LDAP size limit exceeded (' . $Result->error() . ').',
            );
        }
        else {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Search failed! ' . $Result->error(),
            );
        }
    }

    my %Users;
    for my $Entry ( $Result->all_entries() ) {

        my $FieldValue = $Entry->get_value( $Self->{CustomerID} );
        $FieldValue = defined $FieldValue ? $FieldValue : '';

        my $KeyValue = $Entry->get_value( $Self->{CustomerKey} );
        $KeyValue = defined $KeyValue ? $KeyValue : '';
        $Users{ $Self->_ConvertFrom($KeyValue) } = $Self->_ConvertFrom($FieldValue);
    }

    # check if user need to be in a group!
    if ( $Self->{GroupDN} ) {
        for my $Filter2 ( sort keys %Users ) {
            my $Result2 = $Self->{LDAP}->search(
                base      => $Self->{GroupDN},
                scope     => $Self->{SScope},
                filter    => 'memberUid=' . escape_filter_value($Filter2),
                sizelimit => $Self->{UserSearchListLimit},
                attrs     => ['1.1'],
            );
            if ( !$Result2->all_entries() ) {
                delete $Users{$Filter2};
            }
        }
    }

    # make CustomerIDs unique
    my %Tmp;
    @Tmp{ values %Users } = undef;
    my @Result = keys %Tmp;

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => $CacheKey,
            Value => \@Result,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return @Result;
}

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

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

    # check cache
    if ( $Self->{CacheObject} ) {
        my $CustomerIDs = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => "CustomerIDs::$Param{User}",
        );
        return @{$CustomerIDs} if ref $CustomerIDs eq 'ARRAY';
    }

    # get customer data
    my %Data = $Self->CustomerUserDataGet(
        User => $Param{User},
    );

    # there are multi customer ids
    my @CustomerIDs;
    if ( $Data{UserCustomerIDs} ) {

        # used separators
        SEPARATOR:
        for my $Separator ( ';', ',', '|' ) {

            next SEPARATOR if $Data{UserCustomerIDs} !~ /\Q$Separator\E/;

            # split it
            my @IDs = split /\Q$Separator\E/, $Data{UserCustomerIDs};

            for my $ID (@IDs) {
                $ID =~ s/^\s+//g;
                $ID =~ s/\s+$//g;
                push @CustomerIDs, $ID;
            }

            last SEPARATOR;
        }

        # fallback if no separator got found
        if ( !@CustomerIDs ) {
            $Data{UserCustomerIDs} =~ s/^\s+//g;
            $Data{UserCustomerIDs} =~ s/\s+$//g;
            push @CustomerIDs, $Data{UserCustomerIDs};
        }
    }

    # use also the primary customer id
    if ( $Data{UserCustomerID} && !$Self->{ExcludePrimaryCustomerID} ) {
        push @CustomerIDs, $Data{UserCustomerID};
    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => 'CustomerIDs::' . $Param{User},
            Value => \@CustomerIDs,
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return @CustomerIDs;
}

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

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

    # perform user search
    my @Attributes;
    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        next ENTRY if $Entry->[5] eq 'dynamic_field';
        push( @Attributes, $Entry->[2] );
    }
    my $Filter = "($Self->{CustomerKey}=" . escape_filter_value( $Param{User} ) . ')';

    # prepare filter
    if ( $Self->{AlwaysFilter} ) {
        $Filter = "(&$Filter$Self->{AlwaysFilter})";
    }

    # check cache
    if ( $Self->{CacheObject} ) {
        my $Data = $Self->{CacheObject}->Get(
            Type => $Self->{CacheType},
            Key  => 'CustomerUserDataGet::' . $Param{User},
        );
        return %{$Data} if ref $Data eq 'HASH';
    }

    # create ldap connect
    return if !$Self->_Connect();

    # perform search
    my $Result = $Self->{LDAP}->search(
        base   => $Self->{BaseDN},
        scope  => $Self->{SScope},
        filter => $Filter,
        attrs  => \@Attributes,
    );

    # log ldap errors
    if ( $Result->code() ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => $Result->error(),
        );
        return;
    }

    # get first entry
    my $Result2 = $Result->entry(0);
    if ( !$Result2 ) {
        return;
    }

    # get customer user info
    my %Data;
    ENTRY:
    for my $Entry ( @{ $Self->{CustomerUserMap}->{Map} } ) {
        next ENTRY if $Entry->[5] eq 'dynamic_field';

        my $Value = $Self->_ConvertFrom( $Result2->get_value( $Entry->[2] ) ) || '';

        if ( $Value && $Entry->[2] =~ /^targetaddress$/i ) {
            $Value =~ s/SMTP:(.*)/$1/;
        }

        if ( IsStringWithData( $Self->{CustomerUserMap}->{TranslateManagerTo} ) ) {
            if ( $Value && $Entry->[2] =~ /^manager$/i ) {

                # We need to check if we need to translate the manager flag
                my $TranslateTo = $Self->{CustomerUserMap}->{TranslateManagerTo};

                # perform search
                my $ResultManager = $Self->{LDAP}->search(
                    base   => $Value,
                    scope  => 'base',
                    filter => "(objectClass=*)",
                );

                # get first entry
                my $ResultManager2 = $ResultManager->entry(0);
                $Value = $Self->_ConvertFrom( $ResultManager2->get_value($TranslateTo) ) || '';
            }
        }

        $Data{ $Entry->[0] } = $Value;
    }

    return if !$Data{UserLogin};

    # to build the UserMailString
    my $UserMailString = '';
    my @UserMailStringParts;

    my $CustomerUserListFieldsMap = $Self->{CustomerUserMap}->{CustomerUserListFields};
    if ( !IsArrayRefWithData($CustomerUserListFieldsMap) ) {
        $CustomerUserListFieldsMap = [ 'first_name', 'last_name', 'email', ];
    }

    for my $Field ( @{$CustomerUserListFieldsMap} ) {

        my $Value = $Self->_ConvertFrom( $Result2->get_value($Field) ) || '';

        if ($Value) {
            if ( $Field =~ /^targetaddress$/i ) {
                $Value =~ s/SMTP:(.*)/$1/;
            }
            push @UserMailStringParts, $Value;
        }
    }
    $UserMailString = join ' ', @UserMailStringParts;
    $UserMailString =~ s/^(.*)\s(.+?\@.+?\..+?)(\s|)$/"$1" <$2>/;

    # add the UserMailString to the data hash
    $Data{UserMailString} = $UserMailString;

    # compat!
    $Data{UserID} = $Data{UserLogin};

    # get preferences
    my %Preferences = $Self->GetPreferences( UserID => $Data{UserLogin} );

    # add last login timestamp
    if ( $Preferences{UserLastLogin} ) {

        my $DateTimeObject = $Kernel::OM->Create(
            'Kernel::System::DateTime',
            ObjectParams => {
                Epoch => $Preferences{UserLastLogin},
            },
        );

        $Preferences{UserLastLoginTimestamp} = $DateTimeObject->ToString();

    }

    # cache request
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Set(
            Type  => $Self->{CacheType},
            Key   => 'CustomerUserDataGet::' . $Param{User},
            Value => { %Data, %Preferences },
            TTL   => $Self->{CustomerUserMap}->{CacheTTL},
        );
    }

    return ( %Data, %Preferences );
}

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

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!'
        );
        return;
    }

    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'error',
        Message  => 'Not supported for this module!'
    );

    return;
}

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

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!'
        );
        return;
    }

    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'error',
        Message  => 'Not supported for this module!'
    );

    return;
}

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

    my $Pw = $Param{PW} || '';

    # check ro/rw
    if ( $Self->{ReadOnly} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Customer backend is read only!'
        );
        return;
    }

    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'error',
        Message  => 'Not supported for this module!'
    );

    return;
}

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

    return $Kernel::OM->Get('Kernel::System::Main')->GenerateRandomString(
        Length     => $Param{Size} || 8,
        Dictionary => [ 0 .. 9, 'A' .. 'Z', 'a' .. 'z', '-', '_', '!', '@', '#', '$', '%', '^', '&', '*' ],
    );
}

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

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

    # cache reset
    if ( $Self->{CacheObject} ) {
        $Self->{CacheObject}->Delete(
            Type => $Self->{CacheType},
            Key  => "CustomerUserDataGet::$Param{UserID}",
        );
    }
    return $Self->{PreferencesObject}->SetPreferences(%Param);
}

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

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

    return $Self->{PreferencesObject}->GetPreferences(%Param);
}

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

    return $Self->{PreferencesObject}->SearchPreferences(%Param);
}

sub _ConvertFrom {
    my ( $Self, $Text ) = @_;

    return if !defined $Text;

    if ( !$Self->{SourceCharset} ) {
        return $Text;
    }

    return $Kernel::OM->Get('Kernel::System::Encode')->Convert(
        Text => $Text,
        From => $Self->{SourceCharset},
        To   => 'utf-8',
    );
}

sub _ConvertTo {
    my ( $Self, $Text ) = @_;

    return if !defined $Text;

    # get encode object
    my $EncodeObject = $Kernel::OM->Get('Kernel::System::Encode');

    if ( !$Self->{SourceCharset} ) {
        $EncodeObject->EncodeInput( \$Text );
        return $Text;
    }

    return $EncodeObject->Convert(
        Text => $Text,
        To   => $Self->{SourceCharset},
        From => 'utf-8',
    );
}

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

    return if !$Self->{CacheObject};

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

    $Self->{CacheObject}->Delete(
        Type => $Self->{CacheType},
        Key  => "CustomerUserDataGet::$Param{UserLogin}",
    );
    $Self->{CacheObject}->Delete(
        Type => $Self->{CacheType},
        Key  => "CustomerName::$Param{UserLogin}",
    );

    $Self->{CacheObject}->CleanUp(
        Type => $Self->{CacheType} . '_CustomerSearch',
    );

    $Self->{CacheObject}->CleanUp(
        Type => 'CustomerGroup',
    );

    return 1;
}

sub _SearchResultSort {
    my ( $Self, @OrderByFields ) = @_;

    for my $OrderBy (@OrderByFields) {
        my $Compare;

        if ( $OrderBy->{Direction} && $OrderBy->{Direction} eq 'Up' ) {
            $Compare = lc( $a->{ $OrderBy->{Name} } ) cmp lc( $b->{ $OrderBy->{Name} } );
        }
        else {
            $Compare = lc( $b->{ $OrderBy->{Name} } ) cmp lc( $a->{ $OrderBy->{Name} } );
        }
        return $Compare if $Compare;
    }
    return 0;
}

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

    # take down session
    if ( $Self->{LDAP} ) {
        $Self->{LDAP}->unbind();
    }

    return 1;
}

1;
</File>
        <File Location="Custom/Kernel/System/CustomerCompany.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/System/CustomerCompany.pm
# --
# 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::CustomerCompany;

use strict;
use warnings;

use parent qw(Kernel::System::EventHandler);

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::Encode',
# Rother OSS / CustomerMultitenancy
    'Kernel::System::Group',
# EO Rother OSS
    'Kernel::System::Log',
    'Kernel::System::Main',
    'Kernel::System::ReferenceData',
    'Kernel::System::Valid',
);

=head1 NAME

Kernel::System::CustomerCompany - customer company lib

=head1 DESCRIPTION

All Customer functions. E.g. to add and update customer companies.

=head1 PUBLIC INTERFACE

=head2 new()

Don't use the constructor directly, use the ObjectManager instead:

    my $CustomerCompanyObject = $Kernel::OM->Get('Kernel::System::CustomerCompany');

=cut

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

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

    # get needed objects
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    my $MainObject   = $Kernel::OM->Get('Kernel::System::Main');

# Rother OSS / CustomerMultitenancy
    my $LayoutParam = $Kernel::OM->{Param}->{'Kernel::Output::HTML::Layout'};

    # Check if multitenancy is enabled and the request is coming from a user.
    if ( $LayoutParam->{UserType} && $LayoutParam->{UserType} eq 'User' && $ConfigObject->Get('Multitenancy') ) {

        # Save the UserID and all groups the user has 'ro' permission on.
        if ( $LayoutParam->{UserID} ) {
            # Only get the group list once for performance reasons.
            my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
                UserID => $LayoutParam->{UserID},
                Type   => 'ro',
            );

            # The limit does not count for members of a specific group.
            my $PermissionGroup = $ConfigObject->Get('Multitenancy::PermissionGroup') || '';
            my %GroupsReverse   = reverse %Groups;

            if ( !$GroupsReverse{$PermissionGroup} ) {
                $Self->{Multitenancy} = $LayoutParam->{UserID};
                $Self->{UserGroupIDs} = [ keys %Groups ];
                $Self->{UserGroups}   = [ values %Groups ];
            }
        }
    }
# EO CustomerMultitenancy

    # load customer company backend modules
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$ConfigObject->Get("CustomerCompany$Count");
# Rother OSS / CustomerMultitenancy
        # Check if the user has permission to access the source.
        my $CustomerCompanyUserGroup = $ConfigObject->Get("CustomerCompany$Count")->{CustomerCompanyUserGroup};

        # The user does not have permission to get information from this source.
        if ( $Self->{Multitenancy} && $CustomerCompanyUserGroup ) {
            if ( !grep { $_ =~ /^$CustomerCompanyUserGroup$/ } @{ $Self->{UserGroups} } ) {
                next SOURCE;
            }
        }
# EO CustomerMultitenancy

        my $GenericModule = $ConfigObject->Get("CustomerCompany$Count")->{Module}
            || 'Kernel::System::CustomerCompany::DB';
        if ( !$MainObject->Require($GenericModule) ) {
            $MainObject->Die("Can't load backend module $GenericModule! $@");
        }
        $Self->{"CustomerCompany$Count"} = $GenericModule->new(
            Count              => $Count,
            CustomerCompanyMap => $ConfigObject->Get("CustomerCompany$Count"),
        );
    }

    # init of event handler
    $Self->EventHandlerInit(
        Config => 'CustomerCompany::EventModulePost',
    );

    return $Self;
}

=head2 CustomerCompanyAdd()

add a new customer company

    my $ID = $CustomerCompanyObject->CustomerCompanyAdd(
        CustomerID              => 'example.com',
        CustomerCompanyName     => 'New Customer Inc.',
        CustomerCompanyStreet   => '5201 Blue Lagoon Drive',
        CustomerCompanyZIP      => '33126',
        CustomerCompanyCity     => 'Miami',
        CustomerCompanyCountry  => 'USA',
        CustomerCompanyURL      => 'http://www.example.org',
        CustomerCompanyComment  => 'some comment',
        ValidID                 => 1,
        UserID                  => 123,
    );

NOTE: Actual fields accepted by this API call may differ based on
CustomerCompany mapping in your system configuration.

=cut

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

    # set defaults
    $Param{Source} ||= 'CustomerCompany';

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

    # store customer company data
    my $CustomerID = $Self->{ $Param{Source} }->CustomerCompanyAdd(%Param);

    return unless $CustomerID;

    # trigger event
    $Self->EventHandler(
        Event => 'CustomerCompanyAdd',
        Data  => {
            CustomerID => $Param{CustomerID},
            NewData    => \%Param,
        },
        UserID => $Param{UserID},
    );

    return $CustomerID;
}

=head2 CustomerCompanyGet()

get customer company attributes

    my %CustomerCompany = $CustomerCompanyObject->CustomerCompanyGet(
        CustomerID => 123,
    );

Returns:

    %CustomerCompany = (
        'CustomerCompanyName'    => 'Customer Inc.',
        'CustomerID'             => 'example.com',
        'CustomerCompanyStreet'  => '5201 Blue Lagoon Drive',
        'CustomerCompanyZIP'     => '33126',
        'CustomerCompanyCity'    => 'Miami',
        'CustomerCompanyCountry' => 'United States',
        'CustomerCompanyURL'     => 'http://example.com',
        'CustomerCompanyComment' => 'Some Comments',
        'ValidID'                => '1',
        'CreateTime'             => '2010-10-04 16:35:49',
        'ChangeTime'             => '2010-10-04 16:36:12',
    );

NOTE: Actual fields returned by this API call may differ based on
CustomerCompany mapping in your system configuration.

=cut

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

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

        return;
    }

    # Fetch dynamic field configurations for CustomerCompany.
    my $DynamicFieldConfigs = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerCompany',
        Valid      => 1,
    );

    my %DynamicFieldLookup = map { $_->{Name} => $_ } $DynamicFieldConfigs->@*;

    # get needed objects
    my $ConfigObject              = $Kernel::OM->Get('Kernel::Config');
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    # search for the company in the configured sources
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE unless $Self->{"CustomerCompany$Count"};

        my %Company = $Self->{"CustomerCompany$Count"}->CustomerCompanyGet( %Param, );

        next SOURCE unless %Company;

        # fetch dynamic field values
        if ( IsArrayRefWithData( $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Map} ) ) {
            CUSTOMERCOMPANYFIELD:
            for my $CustomerCompanyField ( @{ $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Map} } ) {
                next CUSTOMERCOMPANYFIELD unless $CustomerCompanyField->[5] eq 'dynamic_field';
                next CUSTOMERCOMPANYFIELD unless $DynamicFieldLookup{ $CustomerCompanyField->[2] };

                my $Value = $DynamicFieldBackendObject->ValueGet(
                    DynamicFieldConfig => $DynamicFieldLookup{ $CustomerCompanyField->[2] },
                    ObjectName         => $Company{CustomerID},
                );

                $Company{ $CustomerCompanyField->[0] } = $Value;
            }
        }

# Rother OSS / CustomerMultitenancy
        # Check permission.
        my $UserGroupIDSync = $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{UserGroupIDSync};

        if ( $Company{UserGroupID} && $UserGroupIDSync->{RemoteGroupToLocalGroup} ) {
            # Replace the remote group with the associated local group.
            for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
                if ( $Company{UserGroupID} eq $RemoteGroup ) {
                    my $LocalGroup = $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup};
                    $Company{UserGroupID} = $LocalGroup;

                    # Check if group ID or group names should be used.
                    if ( $UserGroupIDSync->{UseGroupNames} ) {
                        my $GroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                            Group => $LocalGroup,
                        );

                        $Company{UserGroupID} = $GroupID;
                    }
                }
            }
        }

        if ( $Self->{Multitenancy} && $Company{UserGroupID} ) {
            # Check if the user has permission to access the information.
            if ( !grep { $_ =~ /^$Company{UserGroupID}$/ } @{ $Self->{UserGroupIDs} } ) {
                return;
            }
        }
# EO CustomerMultitenancy

        # return data for the first found company
        return (
            %Company,
            Source => "CustomerCompany$Count",
            Config => $ConfigObject->Get("CustomerCompany$Count"),
        );
    }

    # company was not found
    return;
}

=head2 CustomerCompanyUpdate()

update customer company attributes

    $CustomerCompanyObject->CustomerCompanyUpdate(
        CustomerCompanyID       => 'oldexample.com', # required for CustomerCompanyID-update
        CustomerID              => 'example.com',
        CustomerCompanyName     => 'New Customer Inc.',
        CustomerCompanyStreet   => '5201 Blue Lagoon Drive',
        CustomerCompanyZIP      => '33126',
        CustomerCompanyLocation => 'Miami',
        CustomerCompanyCountry  => 'USA',
        CustomerCompanyURL      => 'http://example.com',
        CustomerCompanyComment  => 'some comment',
        ValidID                 => 1,
        UserID                  => 123,
    );

=cut

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

    $Param{CustomerCompanyID} ||= $Param{CustomerID};

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

    # check if company exists
    my %Company = $Self->CustomerCompanyGet( CustomerID => $Param{CustomerCompanyID} );
    if ( !%Company ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such company '$Param{CustomerCompanyID}'!",
        );

        return;
    }

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to change the UserGroupID.
    if ( $Self->{Multitenancy} ) {
        # Set the UserGroupID to the current UserGroupID.
        if ( $Company{UserGroupID} ) {
            $Param{UserGroupID} = $Company{UserGroupID};
        }
    }

    my $UserGroupIDSync = $Self->{$Company{Source}}->{CustomerCompanyMap}->{UserGroupIDSync};
    if ( $Param{UserGroupID} ) {
        # Check if group ID or group names should be used.
        if ( $UserGroupIDSync->{UseGroupNames} ) {
            $Param{UserGroupID} = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                GroupID => $Param{UserGroupID},
            );
        }

        # Replace the local group with the associated remote group.
        for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
            if ( $Param{UserGroupID} eq $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup} ) {
                $Param{UserGroupID} = $RemoteGroup;
            }
        }
    }
# EO CustomerMultitenancy

    my $Result = $Self->{ $Company{Source} }->CustomerCompanyUpdate(%Param);

    return unless $Result;

    # trigger event
    $Self->EventHandler(
        Event => 'CustomerCompanyUpdate',
        Data  => {
            CustomerID    => $Param{CustomerID},
            OldCustomerID => $Param{CustomerCompanyID},
            NewData       => \%Param,
            OldData       => \%Company,
        },
        UserID => $Param{UserID},
    );

    return $Result;
}

=head2 CustomerCompanySourceList()

return customer company source list

    my %List = $CustomerCompanyObject->CustomerCompanySourceList(
        ReadOnly => 0 # optional, 1 returns only RO backends, 0 returns writable, if not passed returns all backends
    );

=cut

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

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my %Data;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$ConfigObject->Get("CustomerCompany$Count");

        if ( defined $Param{ReadOnly} ) {
            my $BackendConfig = $ConfigObject->Get("CustomerCompany$Count");
            if ( $Param{ReadOnly} ) {
                next SOURCE if !$BackendConfig->{ReadOnly};
            }
            else {
                next SOURCE if $BackendConfig->{ReadOnly};
            }
        }

        $Data{"CustomerCompany$Count"} = $ConfigObject->Get("CustomerCompany$Count")->{Name}
            || "No Name $Count";
    }

    return %Data;
}

=head2 CustomerCompanyList()

get list of customer companies.

    my %List = $CustomerCompanyObject->CustomerCompanyList();

    my %List = $CustomerCompanyObject->CustomerCompanyList(
        Valid => 0,
        Limit => 0,     # optional, override configured search result limit (0 means unlimited)
    );

    my %List = $CustomerCompanyObject->CustomerCompanyList(
        Search => 'somecompany',
    );

Returns:

    %List = {
        'example.com' => 'example.com Customer Inc.',
        'acme.com'    => 'acme.com Acme, Inc.'
    };

=cut

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

    # Get dynamic field object.
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    my $DynamicFieldConfigs = $DynamicFieldObject->DynamicFieldListGet(
        ObjectType => 'CustomerCompany',
        Valid      => 1,
    );

    my %DynamicFieldLookup = map { $_->{Name} => $_ } @{$DynamicFieldConfigs};

    # Get dynamic field backend object.
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    my %Data;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerCompany$Count"};

        # search dynamic field values, if configured
        my $Map = $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Map};
        if ( IsArrayRefWithData($Map) ) {

            # fetch dynamic field names that are configured in Map
            # only these will be considered for any other search config
            # [ 'DynamicField_Name_Y', undef, 'Name_Y', 0, 0, 'dynamic_field', undef, 0,],
            my %DynamicFieldNames = map { $_->[2] => 1 } grep { $_->[5] eq 'dynamic_field' } @{$Map};

            if (%DynamicFieldNames) {
                my $FoundDynamicFieldObjectIDs;
                my $SearchFields;
                my $SearchParam;

                # check which of the dynamic fields configured in Map are also
                # configured in SearchFields

                # param Search
                if ( defined $Param{Search} && length $Param{Search} ) {
                    $SearchFields = $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{CustomerCompanySearchFields};
                    $SearchParam  = $Param{Search};
                }

                # search dynamic field values
                if ( IsArrayRefWithData($SearchFields) ) {
                    my @SearchDynamicFieldNames = grep { exists $DynamicFieldNames{$_} } @{$SearchFields};

                    my %FoundDynamicFieldObjectIDs;
                    FIELDNAME:
                    for my $FieldName (@SearchDynamicFieldNames) {

                        my $DynamicFieldConfig = $DynamicFieldLookup{$FieldName};

                        next FIELDNAME if !IsHashRefWithData($DynamicFieldConfig);

                        my $DynamicFieldValues = $DynamicFieldBackendObject->ValueSearch(
                            DynamicFieldConfig => $DynamicFieldConfig,
                            Search             => $SearchParam,
                        );

                        if ( IsArrayRefWithData($DynamicFieldValues) ) {
                            for my $DynamicFieldValue ( @{$DynamicFieldValues} ) {
                                $FoundDynamicFieldObjectIDs{ $DynamicFieldValue->{ObjectID} } = 1;
                            }
                        }
                    }

                    $FoundDynamicFieldObjectIDs = [ keys %FoundDynamicFieldObjectIDs ];
                }

                # execute backend search for found object IDs
                # this data is being merged with the following CustomerCompanyList call
                if ( IsArrayRefWithData($FoundDynamicFieldObjectIDs) ) {

                    my $ObjectNames = $DynamicFieldObject->ObjectMappingGet(
                        ObjectID   => $FoundDynamicFieldObjectIDs,
                        ObjectType => 'CustomerCompany',
                    );

                    my %SearchParam = %Param;
                    delete $SearchParam{Search};
                    my %CompanyList = $Self->{"CustomerCompany$Count"}->CustomerCompanyList(%SearchParam);

                    OBJECTNAME:
                    for my $ObjectName ( values %{$ObjectNames} ) {
                        next OBJECTNAME if exists $Data{$ObjectName};

                        if ( IsHashRefWithData( \%CompanyList ) && exists $CompanyList{$ObjectName} ) {
                            %Data = (
                                $ObjectName => $CompanyList{$ObjectName},
                                %Data
                            );
                        }
                    }
                }
            }
        }

        # get company list result of backend and merge it
        my %SubData = $Self->{"CustomerCompany$Count"}->CustomerCompanyList(%Param);
        %Data = ( %Data, %SubData );
    }

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to see the customer user data.
    # Improve: Check every company one by one as CustomerCompanyList does sometimes return a non filtered list (CustomerCompany add)
    if ( $Self->{Multitenancy} ) {
        for my $CustomerID ( keys %Data ) {
            my %Company = $Self->CustomerCompanyGet(
                CustomerID => $CustomerID,
            );

            if ( !%Company || !$Company{CustomerID} ) {
                delete $Data{$CustomerID};
            }
        }
    }
# EO CustomerMultitenancy

    return %Data;
}

=head2 CustomerCompanySearchDetail()

To find customer companies in the system.

The search criteria are logically AND connected.
When a list is passed as criteria, the individual members are OR connected.
When an undef or a reference to an empty array is passed, then the search criteria
is ignored.

Returns either a list, as an arrayref, or a count of found customer company ids.
The count of results is returned when the parameter C<Result = 'COUNT'> is passed.

    my $CustomerCompanyIDsRef = $CustomerCompanyObject->CustomerCompanySearchDetail(

        # all fields in a CustomerCompanyMap are searchable
        CustomerID          => 'example*',                                  # (optional)
        CustomerCompanyName => 'Name*',                                     # (optional)

        # array parameters are used with logical OR operator (all values are possible which
        are defined in the config selection hash for the field)
        CustomerCompanyCountry => [ 'Austria', 'Germany', ],                # (optional)

        # DynamicFields
        #   At least one operator must be specified. Operators will be connected with AND,
        #       values in an operator with OR.
        #   You can also pass more than one argument to an operator: ['value1', 'value2']
        DynamicField_FieldNameX => {
            Equals            => 123,
            Like              => 'value*',                # "equals" operator with wildcard support
            GreaterThan       => '2001-01-01 01:01:01',
            GreaterThanEquals => '2001-01-01 01:01:01',
            SmallerThan       => '2002-02-02 02:02:02',
            SmallerThanEquals => '2002-02-02 02:02:02',
        }

        OrderBy => [ 'CustomerID', 'CustomerCompanyCountry' ],              # (optional)
        # ignored if the result type is 'COUNT'
        # default: [ 'CustomerID' ]
        # (all fields which are in a CustomerCompanyMap can be used for ordering)

        # Additional information for OrderBy:
        # The OrderByDirection can be specified for each OrderBy attribute.
        # The pairing is made by the array indices.

        OrderByDirection => [ 'Down', 'Up' ],                               # (optional)
        # ignored if the result type is 'COUNT'
        # (Down | Up) Default: [ 'Down' ]

        Result => 'ARRAY' || 'COUNT',                                       # (optional)
        # default: ARRAY, returns an array of change ids
        # COUNT returns a scalar with the number of found changes

        Limit => 100,                                                       # (optional)
        # ignored if the result type is 'COUNT'
    );

Returns:

Result: 'ARRAY'

    @CustomerIDs = ( 1, 2, 3 );

Result: 'COUNT'

    $CustomerIDs = 10;

=cut

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

    # Get all general search fields (without a restriction to a source).
    my @AllSearchFields = $Self->CustomerCompanySearchFields();

    # Generate a hash with the customer company sources which must be searched.
    my %SearchCustomerCompanySources;

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerCompany$Count"};

        # Get the search fields for the current source.
        my @SourceSearchFields = $Self->CustomerCompanySearchFields(
            Source => "CustomerCompany$Count",
        );
        my %LookupSourceSearchFields = map { $_->{Name} => 1 } @SourceSearchFields;

        # Check if all search param exists in the search fields from the current source.
        SEARCHFIELD:
        for my $SearchField (@AllSearchFields) {

            next SEARCHFIELD if !$Param{ $SearchField->{Name} };

            next SOURCE if !$LookupSourceSearchFields{ $SearchField->{Name} };
        }
        $SearchCustomerCompanySources{"CustomerCompany$Count"} = \@SourceSearchFields;
    }

    # Set the default behaviour for the return type.
    $Param{Result} ||= 'ARRAY';

    if ( $Param{Result} eq 'COUNT' ) {

        my $IDsCount = 0;

        SOURCE:
        for my $Source ( sort keys %SearchCustomerCompanySources ) {
            next SOURCE if !$Self->{$Source};

            my $SubIDsCount = $Self->{$Source}->CustomerCompanySearchDetail(
                %Param,
                SearchFields => $SearchCustomerCompanySources{$Source},
            );

            return if !defined $SubIDsCount;

            $IDsCount += $SubIDsCount || 0;
        }
        return $IDsCount;
    }
    else {

        my @IDs;

        my $ResultCount = 0;

        SOURCE:
        for my $Source ( sort keys %SearchCustomerCompanySources ) {
            next SOURCE if !$Self->{$Source};

            my $SubIDs = $Self->{$Source}->CustomerCompanySearchDetail(
                %Param,
                SearchFields => $SearchCustomerCompanySources{$Source},
            );

            return if !defined $SubIDs;

            next SOURCE if !IsArrayRefWithData($SubIDs);

            push @IDs, @{$SubIDs};

            $ResultCount++;
        }

        # If we have more then one search results from diffrent sources, we need a resorting
        #   and splice (for the limit) because of the merged single results.
        if ( $ResultCount > 1 ) {

            my @CustomerCompanyataList;

            for my $ID (@IDs) {

                my %CustomerCompanyData = $Self->CustomerCompanyGet(
                    CustomerID => $ID,
                );
                push @CustomerCompanyataList, \%CustomerCompanyData;
            }

            my $OrderBy = 'CustomerID';
            if ( IsArrayRefWithData( $Param{OrderBy} ) ) {
                $OrderBy = $Param{OrderBy}->[0];
            }

            if ( IsArrayRefWithData( $Param{OrderByDirection} ) && $Param{OrderByDirection}->[0] eq 'Up' ) {
                @CustomerCompanyataList = sort { lc( $a->{$OrderBy} ) cmp lc( $b->{$OrderBy} ) } @CustomerCompanyataList;
            }
            else {
                @CustomerCompanyataList = sort { lc( $b->{$OrderBy} ) cmp lc( $a->{$OrderBy} ) } @CustomerCompanyataList;
            }

            if ( $Param{Limit} && scalar @CustomerCompanyataList > $Param{Limit} ) {
                splice @CustomerCompanyataList, $Param{Limit};
            }

            @IDs = map { $_->{CustomerID} } @CustomerCompanyataList;
        }

        return \@IDs;
    }
}

=head2 CustomerCompanySearchFields()

Get a list of defined search fields (optional only the relevant fields for the given source).

    my @SeachFields = $CustomerCompanyObject->CustomerCompanySearchFields(
        Source => 'CustomerCompany', # optional, but important in the CustomerCompanySearchDetail to get the right database fields
    );

Returns an array of hash references.

    @SeachFields = (
        {
            Name  => 'CustomerID',
            Label => 'CustomerID',
            Type  => 'Input',
        },
        {
            Name           => 'CustomerCompanyCountry',
            Label          => 'Country',
            Type           => 'Selection',
            SelectionsData => {
                'Germany'        => 'Germany',
                'United Kingdom' => 'United Kingdom',
                'United States'  => 'United States',
                ...
            },
        },
        {
            Name          => 'DynamicField_Branch',
            Label         => '',
            Type          => 'DynamicField',
            DatabaseField => 'Branch',
        },
    );

=cut

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

    # Get the search fields from all customer company maps (merge from all maps together).
    my @SearchFields;

    my %SearchFieldsExists;

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerCompany$Count"};
        next SOURCE if $Param{Source} && $Param{Source} ne "CustomerCompany$Count";

        ENTRY:
        for my $Entry ( @{ $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Map} } ) {

            my $SearchFieldName = $Entry->[0];

            next ENTRY if $SearchFieldsExists{$SearchFieldName};

            # Remeber the already collected search field name.
            $SearchFieldsExists{$SearchFieldName} = 1;

            my %FieldConfig = $Self->GetFieldConfig(
                FieldName => $SearchFieldName,
                Source    => $Param{Source},     # to get the right database field for the given source
            );

            next ENTRY if !%FieldConfig;

            my %SearchFieldData = (
                %FieldConfig,
                Name => $SearchFieldName,
            );

            my %SelectionsData = $Self->GetFieldSelections(
                FieldName => $SearchFieldName,
            );

            if ( $SearchFieldData{StorageType} eq 'dynamic_field' ) {
                $SearchFieldData{Type} = 'DynamicField';
            }
            elsif (%SelectionsData) {
                $SearchFieldData{Type}           = 'Selection';
                $SearchFieldData{SelectionsData} = \%SelectionsData;
            }
            else {
                $SearchFieldData{Type} = 'Input';
            }

            push @SearchFields, \%SearchFieldData;
        }
    }

    return @SearchFields;
}

=head2 GetFieldConfig()

This function collect some field config information from the customer user map.

    my %FieldConfig = $CustomerCompanyObject->GetFieldConfig(
        FieldName => 'CustomerCompanyName',
        Source    => 'CustomerCompany', # optional
    );

Returns some field config information:

    my %FieldConfig = (
        Label         => 'Name',
        DatabaseField => 'name',
        StorageType   => 'var',
    );

=cut

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

    if ( !$Param{FieldName} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need FieldName!"
        );
        return;
    }

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerCompany$Count"};
        next SOURCE if $Param{Source} && $Param{Source} ne "CustomerCompany$Count";

        # Search the right field and return the label.
        ENTRY:
        for my $Entry ( @{ $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Map} } ) {
            next ENTRY if $Param{FieldName} ne $Entry->[0];

            my %FieldConfig = (
                Label         => $Entry->[1],
                DatabaseField => $Entry->[2],
                StorageType   => $Entry->[5],
            );

            return %FieldConfig;
        }
    }

    return;
}

=head2 GetFieldSelections()

This function collect the selections for the given field name, if the field has some selections.

    my %SelectionsData = $CustomerCompanyObject->GetFieldSelections(
        FieldName => 'CustomerCompanyCountry',
    );

Returns the selections for the given field name (merged from all sources) or a empty hash:

    my %SelectionData = (
        'Germany'        => 'Germany',
        'United Kingdom' => 'United Kingdom',
        'United States'  => 'United States',
    );

=cut

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

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

    my %SelectionsData;

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerCompany$Count"};
        next SOURCE if !$Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Selections}->{ $Param{FieldName} };

        %SelectionsData = (
            %SelectionsData,
            %{ $Self->{"CustomerCompany$Count"}->{CustomerCompanyMap}->{Selections}->{ $Param{FieldName} } }
        );
    }

    # Make sure the encoding stamp is set.
    for my $Key ( sort keys %SelectionsData ) {
        $SelectionsData{$Key} = $Kernel::OM->Get('Kernel::System::Encode')->EncodeInput( $SelectionsData{$Key} );
    }

    # Default handling for field 'CustomerCompanyCountry'.
    if ( !%SelectionsData && $Param{FieldName} =~ /^CustomerCompanyCountry/i ) {
        %SelectionsData = %{ $Kernel::OM->Get('Kernel::System::ReferenceData')->CountryList() };
    }

    # Default handling for field 'ValidID'.
    elsif ( !%SelectionsData && $Param{FieldName} =~ /^ValidID/i ) {
        %SelectionsData = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();
    }

    return %SelectionsData;
}

sub DESTROY {
    my $Self = shift;

    # execute all transaction events
    $Self->EventHandlerTransaction();

    return 1;
}

1;
</File>
        <File Location="Custom/Kernel/System/CustomerUser.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/System/CustomerUser.pm
# --
# 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::CustomerUser;

use strict;
use warnings;

use parent qw(Kernel::System::EventHandler);

# core modules

# CPAN modules

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Language',
    'Kernel::System::Cache',
    'Kernel::System::CustomerCompany',
    'Kernel::System::DB',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::Encode',
# Rother OSS / CustomerMultitenancy
    'Kernel::System::Group',
# EO Rother OSS
    'Kernel::System::Log',
    'Kernel::System::Main',
# Rother OSS / CustomerMultitenancy
    'Kernel::System::User',
# EO Rother OSS
    'Kernel::System::Valid',
);

=head1 NAME

Kernel::System::CustomerUser - customer user lib

=head1 DESCRIPTION

All customer user functions. E. g. to add and update customer users.

=head1 PUBLIC INTERFACE

=head2 new()

Don't use the constructor directly, use the ObjectManager instead:

    my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');

=cut

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

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

    $Self->{CacheType} = 'CustomerUser';
    $Self->{CacheTTL}  = 60 * 60 * 24 * 20;

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # load generator customer preferences module
    my $GeneratorModule = $ConfigObject->Get('CustomerPreferences')->{Module}
        || 'Kernel::System::CustomerUser::Preferences::DB';

    # get main object
    my $MainObject = $Kernel::OM->Get('Kernel::System::Main');

    if ( $MainObject->Require($GeneratorModule) ) {
        $Self->{PreferencesObject} = $GeneratorModule->new();
    }

# Rother OSS / CustomerMultitenancy
    my $LayoutParam = $Kernel::OM->{Param}->{'Kernel::Output::HTML::Layout'};

    # Check if multitenancy is enabled and the request is coming from a user.
    if ( $LayoutParam->{UserType} && $LayoutParam->{UserType} eq 'User' && $ConfigObject->Get('Multitenancy') ) {

        # Save the UserID and all groups the user has 'ro' permission on.
        if ( $LayoutParam->{UserID} ) {
            # Only get the group list once for performance reasons.
            my %Groups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
                UserID => $LayoutParam->{UserID},
                Type   => 'ro',
            );

            # The limit does not count for members of a specific group.
            my $PermissionGroup = $ConfigObject->Get('Multitenancy::PermissionGroup') || '';
            my %GroupsReverse   = reverse %Groups;

            if ( !$GroupsReverse{$PermissionGroup} ) {
                $Self->{Multitenancy} = $LayoutParam->{UserID};
                $Self->{UserGroupIDs} = [ keys %Groups ];
                $Self->{UserGroups}   = [ values %Groups ];
            }
        }
    }
# EO CustomerMultitenancy

    # load customer user backend module
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE unless $ConfigObject->Get("CustomerUser$Count");
# Rother OSS / CustomerMultitenancy
        # Check if the user has permission to access the source.
        my $CustomerUserGroup = $ConfigObject->Get("CustomerUser$Count")->{CustomerUserGroup};

        # The user does not have permission to get information from this source.
        if ( $Self->{Multitenancy} && $CustomerUserGroup ) {
            if ( !grep { $_ =~ /^$CustomerUserGroup$/ } @{ $Self->{UserGroups} } ) {
                next SOURCE;
            }
        }
# EO CustomerMultitenancy

        my $GenericModule = $ConfigObject->Get("CustomerUser$Count")->{Module};
        if ( !$MainObject->Require($GenericModule) ) {
            $MainObject->Die("Can't load backend module $GenericModule! $@");
        }

        $Self->{"CustomerUser$Count"} = $GenericModule->new(
            Count             => $Count,
            PreferencesObject => $Self->{PreferencesObject},
            CustomerUserMap   => $ConfigObject->Get("CustomerUser$Count"),
        );
    }

    # init of event handler
    $Self->EventHandlerInit(
        Config => 'CustomerUser::EventModulePost',
    );

    return $Self;
}

=head2 CustomerSourceList()

return customer source list

    my %List = $CustomerUserObject->CustomerSourceList(
        ReadOnly => 0 # optional, 1 returns only RO backends, 0 returns writable, if not passed returns all backends
    );

=cut

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

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my %Data;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$ConfigObject->Get("CustomerUser$Count");
        if ( defined $Param{ReadOnly} ) {
            my $CustomerBackendConfig = $ConfigObject->Get("CustomerUser$Count");
            if ( $Param{ReadOnly} ) {
                next SOURCE if !$CustomerBackendConfig->{ReadOnly} || $CustomerBackendConfig->{Module} !~ /LDAP/i;
            }
            else {
                next SOURCE if $CustomerBackendConfig->{ReadOnly} || $CustomerBackendConfig->{Module} =~ /LDAP/i;
            }
        }
        $Data{"CustomerUser$Count"} = $ConfigObject->Get("CustomerUser$Count")->{Name}
            || "No Name $Count";
    }
    return %Data;
}

=head2 CustomerSearch()

to search users

    # text search
    my %List = $CustomerUserObject->CustomerSearch(
        Search => '*some*', # also 'hans+huber' possible
        Valid  => 1,        # (optional) default 1
        Limit  => 100,      # (optional) overrides limit of the config
    );

    # username search
    my %List = $CustomerUserObject->CustomerSearch(
        UserLogin => '*some*',
        Valid     => 1,         # (optional) default 1
    );

    # email search
    my %List = $CustomerUserObject->CustomerSearch(
        PostMasterSearch => 'email@example.com',
        Valid            => 1,                    # (optional) default 1
    );

    # search by CustomerID
    my %List = $CustomerUserObject->CustomerSearch(
        CustomerID       => 'CustomerID123',
        Valid            => 1,                # (optional) default 1
    );

=cut

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

    # remove leading and ending spaces
    if ( $Param{Search} ) {
        $Param{Search} =~ s/^\s+//;
        $Param{Search} =~ s/\s+$//;
    }

    # Get dynamic field object.
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    my $DynamicFieldConfigs = $DynamicFieldObject->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
        Valid      => 1,
    );

    my %DynamicFieldLookup = map { $_->{Name} => $_ } @{$DynamicFieldConfigs};

    # Get dynamic field backend object.
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    my %Data;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};

        # search dynamic field values, if configured
        my $Map = $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Map};
        if ( IsArrayRefWithData($Map) ) {

            # fetch dynamic field names that are configured in Map
            # only these will be considered for any other search config
            # [ 'DynamicField_Name_X', undef, 'Name_X', 0, 0, 'dynamic_field', undef, 0, undef, undef, ],
            my %DynamicFieldNames = map { $_->[2] => 1 } grep { $_->[5] eq 'dynamic_field' } @{$Map};

            if ( IsHashRefWithData( \%DynamicFieldNames ) ) {
                my $FoundDynamicFieldObjectIDs;
                my $SearchFields;
                my $SearchParam;

                # check which of the dynamic fields configured in Map are also
                # configured in SearchFields

                # param Search
                if ( defined $Param{Search} && length $Param{Search} ) {
                    $SearchFields = $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{CustomerUserSearchFields};
                    $SearchParam  = $Param{Search};
                }

                # param PostMasterSearch
                elsif ( defined $Param{PostMasterSearch} && length $Param{PostMasterSearch} ) {
                    $SearchFields = $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{CustomerUserPostMasterSearchFields};
                    $SearchParam  = $Param{PostMasterSearch};
                }

                # search dynamic field values
                if ( IsArrayRefWithData($SearchFields) ) {
                    my @SearchDynamicFieldNames = grep { exists $DynamicFieldNames{$_} } @{$SearchFields};

                    my %FoundDynamicFieldObjectIDs;
                    FIELDNAME:
                    for my $FieldName (@SearchDynamicFieldNames) {

                        my $DynamicFieldConfig = $DynamicFieldLookup{$FieldName};

                        next FIELDNAME if !IsHashRefWithData($DynamicFieldConfig);

                        my $DynamicFieldValues = $DynamicFieldBackendObject->ValueSearch(
                            DynamicFieldConfig => $DynamicFieldConfig,
                            Search             => $SearchParam,
                        );

                        if ( IsArrayRefWithData($DynamicFieldValues) ) {
                            for my $DynamicFieldValue ( @{$DynamicFieldValues} ) {
                                $FoundDynamicFieldObjectIDs{ $DynamicFieldValue->{ObjectID} } = 1;
                            }
                        }
                    }

                    $FoundDynamicFieldObjectIDs = [ keys %FoundDynamicFieldObjectIDs ];
                }

                # execute backend search for found object IDs
                # this data is being merged with the following CustomerSearch call
                if ( IsArrayRefWithData($FoundDynamicFieldObjectIDs) ) {

                    my $ObjectNames = $DynamicFieldObject->ObjectMappingGet(
                        ObjectID   => $FoundDynamicFieldObjectIDs,
                        ObjectType => 'CustomerUser',
                    );

                    OBJECTNAME:
                    for my $ObjectName ( values %{$ObjectNames} ) {
                        next OBJECTNAME if exists $Data{$ObjectName};

                        my %SearchParam = %Param;
                        delete $SearchParam{Search};
                        delete $SearchParam{PostMasterSearch};

                        $SearchParam{UserLogin} = $ObjectName;

                        my %SubData = $Self->{"CustomerUser$Count"}->CustomerSearch(%SearchParam);

                        # UserLogin search does a wild-card search, but in this case only the
                        # exact matching user login is relevant
                        if ( IsHashRefWithData( \%SubData ) && exists $SubData{$ObjectName} ) {
                            %Data = (
                                $ObjectName => $SubData{$ObjectName},
                                %Data
                            );
                        }
                    }
                }
            }
        }

        # get customer search result of backend and merge it
        my %SubData = $Self->{"CustomerUser$Count"}->CustomerSearch(%Param);

        %Data = ( %SubData, %Data );
    }

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to see this customer user.
    # TODO: Remove this if the DB/LDAP function CustomerSearch is fully implemented.
    if ( $Self->{Multitenancy} ) {
        CUSTOMERLOGIN:
        for my $CustomerUserLogin ( keys %Data ) {
            my %UserData = $Self->CustomerUserDataGet(
                User => $CustomerUserLogin,
            );

            if ( !%UserData ) {
                delete $Data{$CustomerUserLogin};
                next CUSTOMERLOGIN;
            }
        }
    }
# EO CustomerMultitenancy

    return %Data;
}

=head2 CustomerSearchDetail()

To find customer users in the system.

The search criteria are logically AND connected.
When a list is passed as criteria, the individual members are OR connected.
When an undef or a reference to an empty array is passed, then the search criteria
is ignored.

Returns either a list, as an arrayref, or a count of found customer user ids.
The count of results is returned when the parameter C<Result = 'COUNT'> is passed.

    my $CustomerUserIDsRef = $CustomerUserObject->CustomerSearchDetail(

        # all fields which are defined in a CustomerUserMap, except password fields, are searchable
        UserLogin     => 'example*',                                    # (optional)
        UserFirstname => 'Firstn*',                                     # (optional)

        # search for valid users only per default,
        # pass 0 in order to also search for invalid users
        Valid     => 1,                                                 # (optional) default 1

        # special parameters
        CustomerCompanySearchCustomerIDs => [ 'example.com' ],          # (optional)
        ExcludeUserLogins                => [ 'example', 'doejohn' ],   # (optional)

        # array parameters are used with logical OR operator (all values are possible which
        # are defined in the config selection hash for the field)
        UserCountry              => [ 'Austria', 'Germany', ],          # (optional)

        # DynamicFields
        #   At least one operator must be specified. Operators will be connected with AND,
        #       values in an operator with OR.
        #   You can also pass more than one argument to an operator: ['value1', 'value2']
        DynamicField_FieldNameX => {
            Equals            => 123,
            Like              => 'value*',                # "equals" operator with wildcard support
            GreaterThan       => '2001-01-01 01:01:01',
            GreaterThanEquals => '2001-01-01 01:01:01',
            SmallerThan       => '2002-02-02 02:02:02',
            SmallerThanEquals => '2002-02-02 02:02:02',
        }

        OrderBy => [ 'UserLogin', 'UserCustomerID' ],                   # (optional)
        # ignored if the result type is 'COUNT'
        # default: [ 'UserLogin' ]
        # (all fields which are defined in a CustomerUserMap can be used for ordering)

        # Additional information for OrderBy:
        # The OrderByDirection can be specified for each OrderBy attribute.
        # The pairing is made by the array indices.

        OrderByDirection => [ 'Down', 'Up' ],                          # (optional)
        # ignored if the result type is 'COUNT'
        # (Down | Up) Default: [ 'Down' ]

        Result => 'ARRAY' || 'COUNT',                                  # (optional)
        # default: ARRAY, returns an array of change ids
        # COUNT returns a scalar with the number of found changes

        Limit => 100,                                                  # (optional)
        # ignored if the result type is 'COUNT'
    );

Returns a list of customer users when $Result => 'ARRAY' was passed:

    $CustomerUserIDs = [ 'adaldrida', 'adamanta', 'adalgrim ' ];
    $CustomerUserIDs = []; # when no customer users had been found

Returns a count of customer users when $Result => 'COUNT' was passed:

    $CustomerUserIDs = 3;
    $CustomerUserIDs = 0; # when no customer users had been found

=cut

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

    # get all general search fields (without a restriction to a source)
    my @AllSearchFields = $Self->CustomerUserSearchFields();

    # generate a hash with the customer user sources which must be searched
    my %SearchCustomerUserSources;

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerUser$Count"};

        # get the search fields for the current source
        my @SourceSearchFields = $Self->CustomerUserSearchFields(
            Source => "CustomerUser$Count",
        );
        my %LookupSourceSearchFields = map { $_->{Name} => 1 } @SourceSearchFields;

        # check if all search param exists in the search fields from the current source
        SEARCHFIELD:
        for my $SearchField (@AllSearchFields) {

            next SEARCHFIELD if !$Param{ $SearchField->{Name} };

            next SOURCE if !$LookupSourceSearchFields{ $SearchField->{Name} };
        }
        $SearchCustomerUserSources{"CustomerUser$Count"} = \@SourceSearchFields;
    }

    # set the default behaviour for the return type
    $Param{Result} ||= 'ARRAY';

    if ( $Param{Result} eq 'COUNT' ) {

        my $IDsCount = 0;

        SOURCE:
        for my $Source ( sort keys %SearchCustomerUserSources ) {
            next SOURCE if !$Self->{$Source};

            my $SubIDsCount = $Self->{$Source}->CustomerSearchDetail(
                %Param,
                SearchFields => $SearchCustomerUserSources{$Source},
            );

            return if !defined $SubIDsCount;

            $IDsCount += $SubIDsCount || 0;
        }
        return $IDsCount;
    }
    else {

        my @IDs;

        my $ResultCount = 0;

        SOURCE:
        for my $Source ( sort keys %SearchCustomerUserSources ) {
            next SOURCE if !$Self->{$Source};

            my $SubIDs = $Self->{$Source}->CustomerSearchDetail(
                %Param,
                SearchFields => $SearchCustomerUserSources{$Source},
            );

            return if !defined $SubIDs;

            next SOURCE if !IsArrayRefWithData($SubIDs);

            push @IDs, @{$SubIDs};

            $ResultCount++;
        }

        # if we have more then one search results from diffrent sources, we need a resorting
        # because of the merged single results
        if ( $ResultCount > 1 ) {

            my @UserDataList;

            for my $ID (@IDs) {

                my %UserData = $Self->CustomerUserDataGet(
                    User => $ID,
                );
                push @UserDataList, \%UserData;
            }

            my $OrderBy = 'UserLogin';
            if ( IsArrayRefWithData( $Param{OrderBy} ) ) {
                $OrderBy = $Param{OrderBy}->[0];
            }

            if ( IsArrayRefWithData( $Param{OrderByDirection} ) && $Param{OrderByDirection}->[0] eq 'Up' ) {
                @UserDataList = sort { lc( $a->{$OrderBy} ) cmp lc( $b->{$OrderBy} ) } @UserDataList;
            }
            else {
                @UserDataList = sort { lc( $b->{$OrderBy} ) cmp lc( $a->{$OrderBy} ) } @UserDataList;
            }

            if ( $Param{Limit} && scalar @UserDataList > $Param{Limit} ) {
                splice @UserDataList, $Param{Limit};
            }

            @IDs = map { $_->{UserLogin} } @UserDataList;
        }

# Rother OSS / CustomerMultitenancy
        # Check permission for every single customer user.
        if ( $Self->{Multitenancy} ) {
            my @NewIDS = @IDs;
            @IDs = ();

            for my $ID (@NewIDS) {
                my %UserData = $Self->CustomerUserDataGet(
                    User => $ID,
                );

                if (%UserData) {
                    push @IDs, $ID;
                }
            }
        }
# EO CustomerMultitenancy

        return \@IDs;
    }
}

=head2 CustomerUserSearchFields()

Get a list of the defined search fields (optional only the relevant fields for the given source).

    my @SeachFields = $CustomerUserObject->CustomerUserSearchFields(
        Source => 'CustomerUser', # optional, but important in the CustomerSearchDetail to get the right database fields
    );

Returns an array of hash references.

    @SeachFields = (
        {
            Name          => 'UserEmail',
            Label         => 'Email',
            Type          => 'Input',
            DatabaseField => 'mail',
        },
        {
            Name           => 'UserCountry',
            Label          => 'Country',
            Type           => 'Selection',
            SelectionsData => {
                'Germany'        => 'Germany',
                'United Kingdom' => 'United Kingdom',
                'United States'  => 'United States',
                ...
            },
            DatabaseField => 'country',
        },
        {
            Name          => 'DynamicField_SkypeAccountName',
            Label         => '',
            Type          => 'DynamicField',
            DatabaseField => 'SkypeAccountName',
        },
    );

=cut

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

    # Get the search fields from all customer user maps (merge from all maps together).
    my @SearchFields;

    my %SearchFieldsExists;

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerUser$Count"};
        next SOURCE if $Param{Source} && $Param{Source} ne "CustomerUser$Count";

        ENTRY:
        for my $Entry ( @{ $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Map} } ) {

            my $SearchFieldName = $Entry->[0];

            next ENTRY if $SearchFieldsExists{$SearchFieldName};
            next ENTRY if $SearchFieldName =~ m{(Password|Pw)\d*$}smxi;

            # Remember the already collected search field name.
            $SearchFieldsExists{$SearchFieldName} = 1;

            my %FieldConfig = $Self->GetFieldConfig(
                FieldName => $SearchFieldName,
                Source    => $Param{Source},     # to get the right database field for the given source
            );

            next ENTRY if !%FieldConfig;

            my %SearchFieldData = (
                %FieldConfig,
                Name => $SearchFieldName,
            );

            my %SelectionsData = $Self->GetFieldSelections(
                FieldName => $SearchFieldName,
            );

            if ( $SearchFieldData{StorageType} eq 'dynamic_field' ) {
                $SearchFieldData{Type} = 'DynamicField';
            }
            elsif (%SelectionsData) {
                $SearchFieldData{Type}           = 'Selection';
                $SearchFieldData{SelectionsData} = \%SelectionsData;
            }
            else {
                $SearchFieldData{Type} = 'Input';
            }

            push @SearchFields, \%SearchFieldData;
        }
    }

    return @SearchFields;
}

=head2 GetFieldConfig()

This function collect some field config information from the customer user map.

    my %FieldConfig = $CustomerUserObject->GetFieldConfig(
        FieldName => 'UserEmail',
        Source    => 'CustomerUser', # optional
    );

Returns some field config information:

    my %FieldConfig = (
        Label         => 'Email',
        DatabaseField => 'email',
        StorageType   => 'var',
    );

=cut

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

    if ( !$Param{FieldName} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need FieldName!"
        );
        return;
    }

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerUser$Count"};
        next SOURCE if $Param{Source} && $Param{Source} ne "CustomerUser$Count";

        # Search the right field and return some config information from the field.
        ENTRY:
        for my $Entry ( @{ $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Map} } ) {
            next ENTRY if $Param{FieldName} ne $Entry->[0];

            my %FieldConfig = (
                Label         => $Entry->[1],
                DatabaseField => $Entry->[2],
                StorageType   => $Entry->[5],
            );

            return %FieldConfig;
        }
    }

    return;
}

=head2 GetFieldSelections()

This function collect the selections for the given field name, if the field has some selections.

    my %SelectionsData = $CustomerUserObject->GetFieldSelections(
        FieldName => 'UserTitle',
    );

Returns the selections for the given field name (merged from all sources) or a empty hash:

    my %SelectionData = (
        'Mr.'  => 'Mr.',
        'Mrs.' => 'Mrs.',
    );

=cut

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

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

    my %SelectionsData;

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {
        next SOURCE if !$Self->{"CustomerUser$Count"};
        next SOURCE if !$Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Selections}->{ $Param{FieldName} };

        %SelectionsData = (
            %SelectionsData, %{ $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Selections}->{ $Param{FieldName} } }
        );
    }

    # Make sure the encoding stamp is set.
    for my $Key ( sort keys %SelectionsData ) {
        $SelectionsData{$Key} = $Kernel::OM->Get('Kernel::System::Encode')->EncodeInput( $SelectionsData{$Key} );
    }

    # Default handling for field 'ValidID'.
    if ( !%SelectionsData && $Param{FieldName} =~ /^ValidID/i ) {
        %SelectionsData = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();
    }

    return %SelectionsData;
}

=head2 CustomerIDList()

return a list of with all known unique CustomerIDs of the registered customers users (no SearchTerm),
or a filtered list where the CustomerIDs must contain a search term.

    my @CustomerIDs = $CustomerUserObject->CustomerIDList(
        SearchTerm  => 'somecustomer',    # optional
        Valid       => 1,                 # optional
    );

=cut

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

    my @Data;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};

        # get customer list result of backend and merge it
        push @Data, $Self->{"CustomerUser$Count"}->CustomerIDList(%Param);
    }

    # make entries unique
    my %Tmp;
    @Tmp{@Data} = undef;
    @Data = sort { lc $a cmp lc $b } keys %Tmp;

# Rother OSS / CustomerMultitenancy
    # Don't return customer IDs if the agent does not have permission to view.
    if ( $Self->{Multitenancy} ) {
        my @CleanedCustomerIDs;

        for my $CustomerID ( @Data ) {
            my %Company = $Kernel::OM->Get('Kernel::System::CustomerCompany')->CustomerCompanyGet(
                CustomerID => $CustomerID,
            );

            if ( %Company && $Company{CustomerID} ) {
                push @CleanedCustomerIDs, $CustomerID;
            }
        }

        @Data = @CleanedCustomerIDs;
    }
# EO CustomerMultitenancy

    return @Data;
}

=head2 CustomerName()

get customer user name

    my $Name = $CustomerUserObject->CustomerName(
        UserLogin => 'some-login',
    );

=cut

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

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};

        # Get customer name and return it.
        my $Name = $Self->{"CustomerUser$Count"}->CustomerName(%Param);
        if ($Name) {
            return $Name;
        }
    }
    return;
}

=head2 CustomerIDs()

get customer user customer ids

    my @CustomerIDs = $CustomerUserObject->CustomerIDs(
        User => 'some-login',
    );

=cut

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

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

    # get customer ids (stop after first source with results)
    my @CustomerIDs;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};

        # get customer ids from source
        my @SourceCustomerIDs = $Self->{"CustomerUser$Count"}->CustomerIDs(%Param);
        next SOURCE if !@SourceCustomerIDs;

        @CustomerIDs = @SourceCustomerIDs;
        last SOURCE;
    }

    # create hash with existing customer ids
    my %CustomerIDs = map { $_ => 1 } @CustomerIDs;

    # get related customer ids
    my @RelatedCustomerIDs = $Self->CustomerUserCustomerMemberList(
        CustomerUserID => $Param{User},
    );

    # add related customer ids if not found in source
    RELATEDCUSTOMERID:
    for my $RelatedCustomerID (@RelatedCustomerIDs) {
        next RELATEDCUSTOMERID if $CustomerIDs{$RelatedCustomerID};

        push @CustomerIDs, $RelatedCustomerID;
    }

# Rother OSS / CustomerMultitenancy
    # Don't return customer IDs if the agent does not have permission to view.
    if ( $Self->{Multitenancy} ) {
        my @CleanedCustomerIDs;

        for my $CustomerID ( @CustomerIDs ) {
            my %Company = $Kernel::OM->Get('Kernel::System::CustomerCompany')->CustomerCompanyGet(
                CustomerID => $CustomerID,
            );

            if ( %Company && $Company{CustomerID} ) {
                push @CleanedCustomerIDs, $CustomerID;
            }
        }

        @CustomerIDs = @CleanedCustomerIDs;
    }
# EO CustomerMultitenancy

    # return customer ids
    return @CustomerIDs;
}

=head2 CustomerUserDataGet()

get user data (UserLogin, UserFirstname, UserLastname, UserEmail, ...)

    my %CustomerUserData = $CustomerUserObject->CustomerUserDataGet(
        User => 'franz',
    );

When there are multiple backends then only the data from the first backend where the customer user is found.
An empty list is returned when the customer user was't found in any backend.

=cut

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

    return unless $Param{User};

    # fetch dynamic field configurations for CustomerUser.
    my $DynamicFieldConfigs = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        ObjectType => 'CustomerUser',
        Valid      => 1,
    );

    my %DynamicFieldLookup = map { $_->{Name} => $_ } @{$DynamicFieldConfigs};

    # Get needed objects.
    my $ConfigObject              = $Kernel::OM->Get('Kernel::Config');
    my $CustomerCompanyObject     = $Kernel::OM->Get('Kernel::System::CustomerCompany');
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};

        my %Customer = $Self->{"CustomerUser$Count"}->CustomerUserDataGet(%Param);
        next SOURCE if !%Customer;

        # generate the full name and save it in the hash
        my $UserFullname = $Self->CustomerName(%Customer);

        # save the generated fullname in the hash.
        $Customer{UserFullname} = $UserFullname;

        # add preferences defaults
        my $Config = $ConfigObject->Get('CustomerPreferencesGroups');
        if ($Config) {
            KEY:
            for my $Key ( sort keys %{$Config} ) {

                next KEY if !defined $Config->{$Key}->{DataSelected};
                next KEY if defined $Customer{ $Config->{$Key}->{PrefKey} };

                # set default data
                $Customer{ $Config->{$Key}->{PrefKey} } = $Config->{$Key}->{DataSelected};
            }
        }

        # check if customer company support is enabled and get company data
        my %Company;
        if (
            $ConfigObject->Get("CustomerCompany")
            && $ConfigObject->Get("CustomerUser$Count")->{CustomerCompanySupport}
            )
        {
            %Company = $CustomerCompanyObject->CustomerCompanyGet(
                CustomerID => $Customer{UserCustomerID},
            );

            $Company{CustomerCompanyValidID} = $Company{ValidID};
        }

        # fetch dynamic field values
        if ( IsArrayRefWithData( $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Map} ) ) {
            CUSTOMERUSERFIELD:
            for my $CustomerUserField ( @{ $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{Map} } ) {
                next CUSTOMERUSERFIELD if $CustomerUserField->[5] ne 'dynamic_field';
                next CUSTOMERUSERFIELD if !$DynamicFieldLookup{ $CustomerUserField->[2] };

                my $Value = $DynamicFieldBackendObject->ValueGet(
                    DynamicFieldConfig => $DynamicFieldLookup{ $CustomerUserField->[2] },
                    ObjectName         => $Customer{UserID},
                );

                $Customer{ $CustomerUserField->[0] } = $Value;
            }
        }

# Rother OSS / CustomerMultitenancy
        # Check permission.
        if ( $Customer{UserGroupID} ) {
            my $UserGroupIDSync = $Self->{"CustomerUser$Count"}->{CustomerUserMap}->{UserGroupIDSync};

            # Replace the remote group with the associated local group.
            if ( $UserGroupIDSync->{RemoteGroupToLocalGroup} ) {
                for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
                    if ( $Customer{UserGroupID} eq $RemoteGroup ) {
                        $Customer{UserGroupID} = $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup};
                    }
                }
            }

            # Check if group ID or group names should be matched.
            if ( $UserGroupIDSync->{UseGroupNames} ) {
                $Customer{UserGroupID} = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                    Group => $Customer{UserGroupID},
                );
            }

            # Check if any limits are set.
            if ( $Self->{Multitenancy} && !grep { $_ =~ /^$Customer{UserGroupID}$/ } @{ $Self->{UserGroupIDs} } ) {
                # The user does not have access to this information.
                return;
            }
        }
        # If there are no group settings, check if permission on customer company is permitted.
        elsif ( $Self->{Multitenancy} && !$Customer{UserGroupID} && ( !%Company || !$Company{CustomerID} ) ) {
            return;
        }
# EO CustomerMultitenancy

        # return customer data
        return (
            %Company,
            %Customer,
            Source        => "CustomerUser$Count",
            Config        => $ConfigObject->Get("CustomerUser$Count"),
            CompanyConfig => $ConfigObject->Get( $Company{Source} // 'CustomerCompany' ),
        );
    }

    return;
}

=head2 CustomerUserAdd()

to add new customer users

    my $UserLogin = $CustomerUserObject->CustomerUserAdd(
        Source         => 'CustomerUser', # CustomerUser source config
        UserFirstname  => 'Manfred',
        UserLastname   => 'Huber',
        UserCustomerID => 'A124',
        UserLogin      => 'mhuber',
        UserPassword   => 'some-pass', # not required
        UserEmail      => 'manfred.huber@example.com',
        ValidID        => 1,
        UserID         => 123,
    );

=cut

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

    # check data source
    if ( !$Param{Source} ) {
        $Param{Source} = 'CustomerUser';
    }

    # check if user exists
    if ( $Param{UserLogin} ) {
        my %User = $Self->CustomerUserDataGet( User => $Param{UserLogin} );
        if (%User) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => $Kernel::OM->Get('Kernel::Language')->Translate( 'Customer user "%s" already exists.', $Param{UserLogin} ),
            );
            return;
        }
    }

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to add the UserGroupID.
    if ( $Self->{Multitenancy} ) {
        delete $Param{UserGroupID};
    }

    my $UserGroupIDSync = $Self->{$Param{Source}}->{CustomerUserMap}->{UserGroupIDSync};
    if ( $Param{UserGroupID} ) {
        # Check if group ID or group names should be matched.
        if ( $UserGroupIDSync->{UseGroupNames} ) {
            my $GroupName = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                GroupID => $Param{UserGroupID},
            );

            $Param{UserGroupID} = $GroupName;
        }

        # Replace the local group with the associated remote group.
        for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
            if ( $Param{UserGroupID} eq $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup} ) {
                $Param{UserGroupID} = $RemoteGroup;
            }
        }
    }
# EO CustomerMultitenancy

    # store customer user data
    my $Result = $Self->{ $Param{Source} }->CustomerUserAdd(%Param);
    return if !$Result;

    # trigger event
    $Self->EventHandler(
        Event => 'CustomerUserAdd',
        Data  => {
            UserLogin => $Param{UserLogin},
            NewData   => \%Param,
        },
        UserID => $Param{UserID},
    );

    return $Result;

}

=head2 CustomerUserUpdate()

to update customer users

    $CustomerUserObject->CustomerUserUpdate(
        Source        => 'CustomerUser', # CustomerUser source config
        ID            => 'mh'            # current user login
        UserLogin     => 'mhuber',       # new user login
        UserFirstname => 'Huber',
        UserLastname  => 'Manfred',
        UserPassword  => 'some-pass',    # not required
        UserEmail     => 'email@example.com',
        ValidID       => 1,
        UserID        => 123,
    );

=cut

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

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

    # check for UserLogin-renaming and if new UserLogin already exists...
    if ( $Param{ID} && ( lc $Param{UserLogin} ne lc $Param{ID} ) ) {
        my %User = $Self->CustomerUserDataGet( User => $Param{UserLogin} );
        if (%User) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => $Kernel::OM->Get('Kernel::Language')->Translate( 'Customer user "%s" already exists.', $Param{UserLogin} ),
            );
            return;
        }
    }

    # check if user exists
    my %User = $Self->CustomerUserDataGet( User => $Param{ID} || $Param{UserLogin} );
    if ( !%User ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such user '$Param{UserLogin}'!",
        );
        return;
    }

# Rother OSS / CustomerMultitenancy
    # Check if the user has permission to change the UserGroupID.
    if ( $Self->{Multitenancy} ) {
        # Set the UserGroupID to the current UserGroupID.
        if ( $User{UserGroupID} ) {
            $Param{UserGroupID} = $User{UserGroupID};
        }
    }

    my $UserGroupIDSync = $Self->{$User{Source}}->{CustomerUserMap}->{UserGroupIDSync};
    if ( $Param{UserGroupID} ) {
        # Check if group ID or group names should be matched.
        if ( $UserGroupIDSync->{UseGroupNames} ) {
            my $GroupName = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                GroupID => $Param{UserGroupID},
            );

            $Param{UserGroupID} = $GroupName;
        }

        # Replace the local group with the associated remote group.
        for my $RemoteGroup ( keys %{ $UserGroupIDSync->{RemoteGroupToLocalGroup} } ) {
            if ( $Param{UserGroupID} eq $UserGroupIDSync->{RemoteGroupToLocalGroup}{$RemoteGroup} ) {
                $Param{UserGroupID} = $RemoteGroup;
            }
        }
    }
# EO CustomerMultitenancy

    my $Result = $Self->{ $User{Source} }->CustomerUserUpdate(%Param);
    return if !$Result;

    # trigger event
    $Self->EventHandler(
        Event => 'CustomerUserUpdate',
        Data  => {
            UserLogin => $Param{ID} || $Param{UserLogin},
            NewData   => \%Param,
            OldData   => \%User,
        },
        UserID => $Param{UserID},
    );

    return $Result;
}

=head2 SetPassword()

to set customer users passwords

    $CustomerUserObject->SetPassword(
        UserLogin => 'some-login',
        PW        => 'some-new-password'
    );

=cut

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

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

    # check if user exists
    my %User = $Self->CustomerUserDataGet( User => $Param{UserLogin} );
    if ( !%User ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such user '$Param{UserLogin}'!",
        );
        return;
    }
    return $Self->{ $User{Source} }->SetPassword(%Param);
}

=head2 GenerateRandomPassword()

generate a random password

    my $Password = $CustomerUserObject->GenerateRandomPassword();

    or

    my $Password = $CustomerUserObject->GenerateRandomPassword(
        Size => 16,
    );

=cut

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

    return $Self->{CustomerUser}->GenerateRandomPassword(%Param);
}

=head2 SetPreferences()

set customer user preferences

    $CustomerUserObject->SetPreferences(
        Key    => 'UserComment',
        Value  => 'some comment',
        UserID => 'some-login',
    );

=cut

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

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

    # Don't allow overwriting of native user data.
    my %Blacklisted = (
        UserID         => 1,
        UserLogin      => 1,
        UserPassword   => 1,
        UserFirstname  => 1,
        UserLastname   => 1,
        UserFullname   => 1,
        UserStreet     => 1,
        UserCity       => 1,
        UserZip        => 1,
        UserCountry    => 1,
        UserComment    => 1,
        UserCustomerID => 1,
        UserTitle      => 1,
        UserEmail      => 1,
        ChangeTime     => 1,
        ChangeBy       => 1,
        CreateTime     => 1,
        CreateBy       => 1,
        UserPhone      => 1,
        UserMobile     => 1,
        UserFax        => 1,
        UserMailString => 1,
        ValidID        => 1,
    );

    return 0 if $Blacklisted{ $Param{Key} };

    # check if user exists
    my %User = $Self->CustomerUserDataGet( User => $Param{UserID} );
    if ( !%User ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such user '$Param{UserID}'!",
        );
        return;
    }

    # call new api (2.4.8 and higher)
    if ( $Self->{ $User{Source} }->can('SetPreferences') ) {
        return $Self->{ $User{Source} }->SetPreferences(%Param);
    }

    # call old api
    return $Self->{PreferencesObject}->SetPreferences(%Param);
}

=head2 GetPreferences()

get customer user preferences

    my %Preferences = $CustomerUserObject->GetPreferences(
        UserID => 'some-login',
    );

=cut

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

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

    # check if user exists
    my %User = $Self->CustomerUserDataGet( User => $Param{UserID} );
    if ( !%User ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such user '$Param{UserID}'!",
        );
        return;
    }

    # call new api (2.4.8 and higher)
    if ( $Self->{ $User{Source} }->can('GetPreferences') ) {
        return $Self->{ $User{Source} }->GetPreferences(%Param);
    }

    # call old api
    return $Self->{PreferencesObject}->GetPreferences(%Param);
}

=head2 SearchPreferences()

search in user preferences

    my %UserList = $CustomerUserObject->SearchPreferences(
        Key   => 'UserSomeKey',
        Value => 'SomeValue',   # optional, limit to a certain value/pattern
    );

=cut

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

    my %Data;
    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};

        # get customer search result of backend and merge it
        # call new api (2.4.8 and higher)
        my %SubData;
        if ( $Self->{"CustomerUser$Count"}->can('SearchPreferences') ) {
            %SubData = $Self->{"CustomerUser$Count"}->SearchPreferences(%Param);
        }

        # call old api
        else {
            %SubData = $Self->{PreferencesObject}->SearchPreferences(%Param);
        }
        %Data = ( %SubData, %Data );
    }

    return %Data;
}

=head2 TokenGenerate()

generate a random token

    my $Token = $CustomerUserObject->TokenGenerate(
        UserID => 123,
    );

=cut

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

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

    my $Token = $Kernel::OM->Get('Kernel::System::Main')->GenerateRandomString(
        Length => 14,
    );

    # save token in preferences
    $Self->SetPreferences(
        Key    => 'UserToken',
        Value  => $Token,
        UserID => $Param{UserID},
    );

    return $Token;
}

=head2 TokenCheck()

check password token

    my $Valid = $CustomerUserObject>TokenCheck(
        Token  => $Token,
        UserID => 123,
    );

=cut

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

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

    # get preferences token
    my %Preferences = $Self->GetPreferences(
        UserID => $Param{UserID},
    );

    # check requested vs. stored token
    return if !$Preferences{UserToken};
    return if $Preferences{UserToken} ne $Param{Token};

    # reset password token
    $Self->SetPreferences(
        Key    => 'UserToken',
        Value  => '',
        UserID => $Param{UserID},
    );

    return 1;
}

=head2 CustomerUserCacheClear()

clear cache of customer user data

    $CustomerUserObject->CustomerUserCacheClear(
        UserLogin => 'mhuber',
    );

=cut

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

    SOURCE:
    for my $Count ( '', 1 .. 10 ) {

        next SOURCE if !$Self->{"CustomerUser$Count"};
        $Self->{"CustomerUser$Count"}->_CustomerUserCacheClear(
            UserLogin => $Param{UserLogin},
        );
    }

    return 1;
}

=head2 CustomerUserCustomerMemberAdd()

to add a customer user to a customer

    my $Success = $CustomerUserObject->CustomerUserCustomerMemberAdd(
        CustomerUserID => 123,
        CustomerID     => 123,
        Active         => 1,        # optional
        UserID         => 123,
    );

=cut

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

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

    # delete affected caches
    my $CacheKey = 'Cache::CustomerUserCustomerMemberList::';
    $Kernel::OM->Get('Kernel::System::Cache')->Delete(
        Type => $Self->{CacheType},
        Key  => $CacheKey . 'CustomerUserID::' . $Param{CustomerUserID},
    );
    $Kernel::OM->Get('Kernel::System::Cache')->Delete(
        Type => $Self->{CacheType},
        Key  => $CacheKey . 'CustomerID::' . $Param{CustomerID},
    );

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # delete existing relation
    return if !$DBObject->Do(
        SQL => 'DELETE FROM customer_user_customer
            WHERE user_id = ?
            AND customer_id = ?',
        Bind => [ \$Param{CustomerUserID}, \$Param{CustomerID} ],
    );

    # return if relation is not active
    return 1 if !$Param{Active};

    # insert new relation
    return if !$DBObject->Do(
        SQL => '
            INSERT INTO customer_user_customer (user_id, customer_id, create_time, create_by,
            change_time, change_by)
            VALUES (?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [ \$Param{CustomerUserID}, \$Param{CustomerID}, \$Param{UserID}, \$Param{UserID}, ],
    );

    return 1;
}

=head2 CustomerUserCustomerMemberList()

get related customer IDs of a customer user

    my @CustomerIDs = $CustomerUserObject->CustomerUserCustomerMemberList(
        CustomerUserID => 123,
    );

Returns:
    @CustomerIDs = (
        '123',
        '456',
    );

get related customer users of a customer ID

    my @CustomerUsers = $CustomerUserObject->CustomerUserCustomerMemberList(
        CustomerID => 123,
    );

Returns:
    @CustomerUsers = (
        '123',
        '456',
    );

=cut

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

    # check needed stuff
    if ( !$Param{CustomerUserID} && !$Param{CustomerID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Got no CustomerUserID or CustomerID!',
        );
        return;
    }

    # get needed objects
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');
    my $CacheKey = 'Cache::CustomerUserCustomerMemberList::';

    if ( $Param{CustomerUserID} ) {

        # check if this result is present (in cache)
        $CacheKey .= 'CustomerUserID::' . $Param{CustomerUserID};
        my $Cache = $Kernel::OM->Get('Kernel::System::Cache')->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return @{$Cache} if $Cache;

        # get customer ids
        return if !$DBObject->Prepare(
            SQL =>
                'SELECT customer_id
                FROM customer_user_customer
                WHERE user_id = ?
                ORDER BY customer_id',
            Bind => [ \$Param{CustomerUserID}, ],
        );

        # fetch the result
        my @CustomerIDs;
        while ( my @Row = $DBObject->FetchrowArray() ) {
            push @CustomerIDs, $Row[0];
        }

        # cache the result
        $Kernel::OM->Get('Kernel::System::Cache')->Set(
            Type  => $Self->{CacheType},
            TTL   => $Self->{CacheTTL},
            Key   => $CacheKey,
            Value => \@CustomerIDs,

        );

        return @CustomerIDs;
    }
    else {

        # check if this result is present (in cache)
        $CacheKey .= 'CustomerID::' . $Param{CustomerID};
        my $Cache = $Kernel::OM->Get('Kernel::System::Cache')->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return @{$Cache} if $Cache;

        # get customer users
        return if !$DBObject->Prepare(
            SQL =>
                'SELECT user_id
                FROM customer_user_customer WHERE
                customer_id = ?
                ORDER BY user_id',
            Bind => [ \$Param{CustomerID}, ],
        );

        # fetch the result
        my @CustomerUserIDs;
        while ( my @Row = $DBObject->FetchrowArray() ) {
            push @CustomerUserIDs, $Row[0];
        }

        # cache the result
        $Kernel::OM->Get('Kernel::System::Cache')->Set(
            Type  => $Self->{CacheType},
            TTL   => $Self->{CacheTTL},
            Key   => $CacheKey,
            Value => \@CustomerUserIDs,
        );

        return @CustomerUserIDs;
    }
}

=head2 DeleteOnePreference()

get customer user preferences

    my %Preferences = $CustomerUserObject->DeleteOnePreference(
        UserID => 'some-login',
        Key    => 'PreferenceKey',
    );

=cut

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

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

    # check if user exists
    my %User = $Self->CustomerUserDataGet( User => $Param{UserID} );
    if ( !%User ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such user '$Param{UserID}'!",
        );
        return;
    }

    # call new api (2.4.8 and higher)
    if ( $Self->{ $User{Source} }->can('DeleteOnePreference') ) {
        return $Self->{ $User{Source} }->DeleteOnePreference(%Param);
    }

    # call old api
    return $Self->{PreferencesObject}->DeleteOnePreference(%Param);
}

sub DESTROY {
    my $Self = shift;

    # execute all transaction events
    $Self->EventHandlerTransaction();

    return 1;
}

1;
</File>
        <File Location="Kernel/Config/Files/XML/CustomerMultitenancy.xml" Permission="660" Encode="Base64">PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiID8+CjxvdG9ib19jb25maWcgdmVyc2lvbj0iMi4wIiBpbml0PSJBcHBsaWNhdGlvbiI+CiAgICA8U2V0dGluZyBOYW1lPSJNdWx0aXRlbmFuY3kiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+RW5hYmxlcyB0aGUgbXVsdGl0ZW5hbmN5IGZ1bmN0aW9uIGZvciBjdXN0b21lciBhbmQgY3VzdG9tZXIgdXNlci48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxOYXZpZ2F0aW9uPkNvcmU6OlBlcm1pc3Npb248L05hdmlnYXRpb24+CiAgICAgICAgPFZhbHVlPgogICAgICAgICAgICA8SXRlbSBWYWx1ZVR5cGU9IkNoZWNrYm94Ij4xPC9JdGVtPgogICAgICAgIDwvVmFsdWU+CiAgICA8L1NldHRpbmc+CiAgICA8U2V0dGluZyBOYW1lPSJNdWx0aXRlbmFuY3k6OlBlcm1pc3Npb25Hcm91cCIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5TcGVjaWZpZXMgdGhlIGdyb3VwIHRoYXQgY2FuIHNldCBtdWx0aXRlbmFuY3kgZm9yIGN1c3RvbWVyLiBNdWx0aXRlbmFuY3kgZG9lcyBub3QgYXBwbHkgdG8gbWVtYmVycyBvZiB0aGlzIGdyb3VwLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPE5hdmlnYXRpb24+Q29yZTo6UGVybWlzc2lvbjwvTmF2aWdhdGlvbj4KICAgICAgICA8VmFsdWU+CiAgICAgICAgICAgIDxJdGVtIFZhbHVlVHlwZT0iU3RyaW5nIiBWYWx1ZVJlZ2V4PSIiPmFkbWluPC9JdGVtPgogICAgICAgIDwvVmFsdWU+CiAgICA8L1NldHRpbmc+Cjwvb3RvYm9fY29uZmlnPgo=</File>
        <File Location="Kernel/Language/de_CustomerMultitenancy.pm" Permission="660" Encode="Base64">IyAtLQojIE9UT0JPIGlzIGEgd2ViLWJhc2VkIHRpY2tldGluZyBzeXN0ZW0gZm9yIHNlcnZpY2Ugb3JnYW5pc2F0aW9ucy4KIyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIwIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDE5LTIwMjYgUm90aGVyIE9TUyBHbWJILCBodHRwczovL290b2JvLmlvLwojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU6IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkgaXQgdW5kZXIKIyB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZQojIEZvdW5kYXRpb24sIGVpdGhlciB2ZXJzaW9uIDMgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiMgVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsIGJ1dCBXSVRIT1VUCiMgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YgTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MKIyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuIFNlZSB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KIyBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQojIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtLiBJZiBub3QsIHNlZSA8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy8+LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0N1c3RvbWVyTXVsdGl0ZW5hbmN5OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5DdXN0b21lckNvbXBhbnkKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xpbWl0cyB0aGUgYWNjZXNzIG9mIGN1c3RvbWVyIGRhdGEgdG8gY2VydGFpbiBncm91cHMgKG11bHRpdGVuYW5jeSkuIFdpdGhvdXQgc2VsZWN0aW9uLCBjdXN0b21lciBkYXRhIGlzIHZpc2libGUgdG8gYWxsIGFnZW50cy4nfSA9CiAgICAgICAgJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VuYWJsZXMgdGhlIG11bHRpdGVuYW5jeSBmdW5jdGlvbiBmb3IgY3VzdG9tZXIgYW5kIGN1c3RvbWVyIHVzZXIuJ30gPQogICAgICAgICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3BlY2lmaWVzIHRoZSBncm91cCB0aGF0IGNhbiBzZXQgbXVsdGl0ZW5hbmN5IGZvciBjdXN0b21lci4gTXVsdGl0ZW5hbmN5IGRvZXMgbm90IGFwcGx5IHRvIG1lbWJlcnMgb2YgdGhpcyBncm91cC4nfSA9CiAgICAgICAgJyc7CgoKICAgIHB1c2ggQHsgJFNlbGYtPntKYXZhU2NyaXB0U3RyaW5nc30gLy8gW10gfSwgKAogICAgKTsKCn0KCjE7Cg==</File>
        <File Location="doc/en/CustomerMultitenancy.pdf" Permission="644" Encode="Base64">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</File>
    </Filelist>
    <DatabaseInstall Type="post">
        <TableAlter Type="post" Name="customer_user">
            <ColumnAdd Name="group_id" Required="false" Size="100" Type="VARCHAR"></ColumnAdd>
        </TableAlter>
        <TableAlter Type="post" Name="customer_company">
            <ColumnAdd Name="group_id" Required="false" Size="100" Type="VARCHAR"></ColumnAdd>
        </TableAlter>
    </DatabaseInstall>
    <DatabaseUninstall Type="pre">
        <TableAlter Type="pre" Name="customer_user">
            <ColumnDrop Name="group_id"></ColumnDrop>
        </TableAlter>
        <TableAlter Type="pre" Name="customer_company">
            <ColumnDrop Name="group_id"></ColumnDrop>
        </TableAlter>
    </DatabaseUninstall>
</otrs_package>